123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545 |
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.VolatileClassesSchemas = exports.SchemaController = void 0;
- exports.buildMergedSchemaObject = buildMergedSchemaObject;
- exports.classNameIsValid = classNameIsValid;
- exports.defaultColumns = exports.default = exports.convertSchemaToAdapterSchema = void 0;
- exports.fieldNameIsValid = fieldNameIsValid;
- exports.invalidClassNameMessage = invalidClassNameMessage;
- exports.systemClasses = exports.requiredColumns = exports.load = void 0;
- var _StorageAdapter = require("../Adapters/Storage/StorageAdapter");
- var _SchemaCache = _interopRequireDefault(require("../Adapters/Cache/SchemaCache"));
- var _DatabaseController = _interopRequireDefault(require("./DatabaseController"));
- var _Config = _interopRequireDefault(require("../Config"));
- var _deepcopy = _interopRequireDefault(require("deepcopy"));
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
- function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
- function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
- function _objectDestructuringEmpty(t) { if (null == t) throw new TypeError("Cannot destructure " + t); }
- function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
- // This class handles schema validation, persistence, and modification.
- //
- // Each individual Schema object should be immutable. The helpers to
- // do things with the Schema just return a new schema when the schema
- // is changed.
- //
- // The canonical place to store this Schema is in the database itself,
- // in a _SCHEMA collection. This is not the right way to do it for an
- // open source framework, but it's backward compatible, so we're
- // keeping it this way for now.
- //
- // In API-handling code, you should only use the Schema class via the
- // DatabaseController. This will let us replace the schema logic for
- // different databases.
- // TODO: hide all schema logic inside the database adapter.
- // -disable-next
- const Parse = require('parse/node').Parse;
- // -disable-next
- const defaultColumns = exports.defaultColumns = Object.freeze({
- // Contain the default columns for every parse object type (except _Join collection)
- _Default: {
- objectId: {
- type: 'String'
- },
- createdAt: {
- type: 'Date'
- },
- updatedAt: {
- type: 'Date'
- },
- ACL: {
- type: 'ACL'
- }
- },
- // The additional default columns for the _User collection (in addition to DefaultCols)
- _User: {
- username: {
- type: 'String'
- },
- password: {
- type: 'String'
- },
- email: {
- type: 'String'
- },
- emailVerified: {
- type: 'Boolean'
- },
- authData: {
- type: 'Object'
- }
- },
- // The additional default columns for the _Installation collection (in addition to DefaultCols)
- _Installation: {
- installationId: {
- type: 'String'
- },
- deviceToken: {
- type: 'String'
- },
- channels: {
- type: 'Array'
- },
- deviceType: {
- type: 'String'
- },
- pushType: {
- type: 'String'
- },
- GCMSenderId: {
- type: 'String'
- },
- timeZone: {
- type: 'String'
- },
- localeIdentifier: {
- type: 'String'
- },
- badge: {
- type: 'Number'
- },
- appVersion: {
- type: 'String'
- },
- appName: {
- type: 'String'
- },
- appIdentifier: {
- type: 'String'
- },
- parseVersion: {
- type: 'String'
- }
- },
- // The additional default columns for the _Role collection (in addition to DefaultCols)
- _Role: {
- name: {
- type: 'String'
- },
- users: {
- type: 'Relation',
- targetClass: '_User'
- },
- roles: {
- type: 'Relation',
- targetClass: '_Role'
- }
- },
- // The additional default columns for the _Session collection (in addition to DefaultCols)
- _Session: {
- user: {
- type: 'Pointer',
- targetClass: '_User'
- },
- installationId: {
- type: 'String'
- },
- sessionToken: {
- type: 'String'
- },
- expiresAt: {
- type: 'Date'
- },
- createdWith: {
- type: 'Object'
- }
- },
- _Product: {
- productIdentifier: {
- type: 'String'
- },
- download: {
- type: 'File'
- },
- downloadName: {
- type: 'String'
- },
- icon: {
- type: 'File'
- },
- order: {
- type: 'Number'
- },
- title: {
- type: 'String'
- },
- subtitle: {
- type: 'String'
- }
- },
- _PushStatus: {
- pushTime: {
- type: 'String'
- },
- source: {
- type: 'String'
- },
- // rest or webui
- query: {
- type: 'String'
- },
- // the stringified JSON query
- payload: {
- type: 'String'
- },
- // the stringified JSON payload,
- title: {
- type: 'String'
- },
- expiry: {
- type: 'Number'
- },
- expiration_interval: {
- type: 'Number'
- },
- status: {
- type: 'String'
- },
- numSent: {
- type: 'Number'
- },
- numFailed: {
- type: 'Number'
- },
- pushHash: {
- type: 'String'
- },
- errorMessage: {
- type: 'Object'
- },
- sentPerType: {
- type: 'Object'
- },
- failedPerType: {
- type: 'Object'
- },
- sentPerUTCOffset: {
- type: 'Object'
- },
- failedPerUTCOffset: {
- type: 'Object'
- },
- count: {
- type: 'Number'
- } // tracks # of batches queued and pending
- },
- _JobStatus: {
- jobName: {
- type: 'String'
- },
- source: {
- type: 'String'
- },
- status: {
- type: 'String'
- },
- message: {
- type: 'String'
- },
- params: {
- type: 'Object'
- },
- // params received when calling the job
- finishedAt: {
- type: 'Date'
- }
- },
- _JobSchedule: {
- jobName: {
- type: 'String'
- },
- description: {
- type: 'String'
- },
- params: {
- type: 'String'
- },
- startAfter: {
- type: 'String'
- },
- daysOfWeek: {
- type: 'Array'
- },
- timeOfDay: {
- type: 'String'
- },
- lastRun: {
- type: 'Number'
- },
- repeatMinutes: {
- type: 'Number'
- }
- },
- _Hooks: {
- functionName: {
- type: 'String'
- },
- className: {
- type: 'String'
- },
- triggerName: {
- type: 'String'
- },
- url: {
- type: 'String'
- }
- },
- _GlobalConfig: {
- objectId: {
- type: 'String'
- },
- params: {
- type: 'Object'
- },
- masterKeyOnly: {
- type: 'Object'
- }
- },
- _GraphQLConfig: {
- objectId: {
- type: 'String'
- },
- config: {
- type: 'Object'
- }
- },
- _Audience: {
- objectId: {
- type: 'String'
- },
- name: {
- type: 'String'
- },
- query: {
- type: 'String'
- },
- //storing query as JSON string to prevent "Nested keys should not contain the '$' or '.' characters" error
- lastUsed: {
- type: 'Date'
- },
- timesUsed: {
- type: 'Number'
- }
- },
- _Idempotency: {
- reqId: {
- type: 'String'
- },
- expire: {
- type: 'Date'
- }
- }
- });
- // fields required for read or write operations on their respective classes.
- const requiredColumns = exports.requiredColumns = Object.freeze({
- read: {
- _User: ['username']
- },
- write: {
- _Product: ['productIdentifier', 'icon', 'order', 'title', 'subtitle'],
- _Role: ['name', 'ACL']
- }
- });
- const invalidColumns = ['length'];
- const systemClasses = exports.systemClasses = Object.freeze(['_User', '_Installation', '_Role', '_Session', '_Product', '_PushStatus', '_JobStatus', '_JobSchedule', '_Audience', '_Idempotency']);
- const volatileClasses = Object.freeze(['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_GraphQLConfig', '_JobSchedule', '_Audience', '_Idempotency']);
- // Anything that start with role
- const roleRegex = /^role:.*/;
- // Anything that starts with userField (allowed for protected fields only)
- const protectedFieldsPointerRegex = /^userField:.*/;
- // * permission
- const publicRegex = /^\*$/;
- const authenticatedRegex = /^authenticated$/;
- const requiresAuthenticationRegex = /^requiresAuthentication$/;
- const clpPointerRegex = /^pointerFields$/;
- // regex for validating entities in protectedFields object
- const protectedFieldsRegex = Object.freeze([protectedFieldsPointerRegex, publicRegex, authenticatedRegex, roleRegex]);
- // clp regex
- const clpFieldsRegex = Object.freeze([clpPointerRegex, publicRegex, requiresAuthenticationRegex, roleRegex]);
- function validatePermissionKey(key, userIdRegExp) {
- let matchesSome = false;
- for (const regEx of clpFieldsRegex) {
- if (key.match(regEx) !== null) {
- matchesSome = true;
- break;
- }
- }
- // userId depends on startup options so it's dynamic
- const valid = matchesSome || key.match(userIdRegExp) !== null;
- if (!valid) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);
- }
- }
- function validateProtectedFieldsKey(key, userIdRegExp) {
- let matchesSome = false;
- for (const regEx of protectedFieldsRegex) {
- if (key.match(regEx) !== null) {
- matchesSome = true;
- break;
- }
- }
- // userId regex depends on launch options so it's dynamic
- const valid = matchesSome || key.match(userIdRegExp) !== null;
- if (!valid) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);
- }
- }
- const CLPValidKeys = Object.freeze(['find', 'count', 'get', 'create', 'update', 'delete', 'addField', 'readUserFields', 'writeUserFields', 'protectedFields']);
- // validation before setting class-level permissions on collection
- function validateCLP(perms, fields, userIdRegExp) {
- if (!perms) {
- return;
- }
- for (const operationKey in perms) {
- if (CLPValidKeys.indexOf(operationKey) == -1) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `${operationKey} is not a valid operation for class level permissions`);
- }
- const operation = perms[operationKey];
- // proceed with next operationKey
- // throws when root fields are of wrong type
- validateCLPjson(operation, operationKey);
- if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {
- // validate grouped pointer permissions
- // must be an array with field names
- for (const fieldName of operation) {
- validatePointerPermission(fieldName, fields, operationKey);
- }
- // readUserFields and writerUserFields do not have nesdted fields
- // proceed with next operationKey
- continue;
- }
- // validate protected fields
- if (operationKey === 'protectedFields') {
- for (const entity in operation) {
- // throws on unexpected key
- validateProtectedFieldsKey(entity, userIdRegExp);
- const protectedFields = operation[entity];
- if (!Array.isArray(protectedFields)) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`);
- }
- // if the field is in form of array
- for (const field of protectedFields) {
- // do not alloow to protect default fields
- if (defaultColumns._Default[field]) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `Default field '${field}' can not be protected`);
- }
- // field should exist on collection
- if (!Object.prototype.hasOwnProperty.call(fields, field)) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `Field '${field}' in protectedFields:${entity} does not exist`);
- }
- }
- }
- // proceed with next operationKey
- continue;
- }
- // validate other fields
- // Entity can be:
- // "*" - Public,
- // "requiresAuthentication" - authenticated users,
- // "objectId" - _User id,
- // "role:rolename",
- // "pointerFields" - array of field names containing pointers to users
- for (const entity in operation) {
- // throws on unexpected key
- validatePermissionKey(entity, userIdRegExp);
- // entity can be either:
- // "pointerFields": string[]
- if (entity === 'pointerFields') {
- const pointerFields = operation[entity];
- if (Array.isArray(pointerFields)) {
- for (const pointerField of pointerFields) {
- validatePointerPermission(pointerField, fields, operation);
- }
- } else {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`);
- }
- // proceed with next entity key
- continue;
- }
- // or [entity]: boolean
- const permit = operation[entity];
- if (permit !== true) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${permit}' is not a valid value for class level permissions ${operationKey}:${entity}:${permit}`);
- }
- }
- }
- }
- function validateCLPjson(operation, operationKey) {
- if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {
- if (!Array.isArray(operation)) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an array`);
- }
- } else {
- if (typeof operation === 'object' && operation !== null) {
- // ok to proceed
- return;
- } else {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an object`);
- }
- }
- }
- function validatePointerPermission(fieldName, fields, operation) {
- // Uses collection schema to ensure the field is of type:
- // - Pointer<_User> (pointers)
- // - Array
- //
- // It's not possible to enforce type on Array's items in schema
- // so we accept any Array field, and later when applying permissions
- // only items that are pointers to _User are considered.
- if (!(fields[fieldName] && (fields[fieldName].type == 'Pointer' && fields[fieldName].targetClass == '_User' || fields[fieldName].type == 'Array'))) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`);
- }
- }
- const joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;
- const classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;
- function classNameIsValid(className) {
- // Valid classes must:
- return (
- // Be one of _User, _Installation, _Role, _Session OR
- systemClasses.indexOf(className) > -1 ||
- // Be a join table OR
- joinClassRegex.test(className) ||
- // Include only alpha-numeric and underscores, and not start with an underscore or number
- fieldNameIsValid(className, className)
- );
- }
- // Valid fields must be alpha-numeric, and not start with an underscore or number
- // must not be a reserved key
- function fieldNameIsValid(fieldName, className) {
- if (className && className !== '_Hooks') {
- if (fieldName === 'className') {
- return false;
- }
- }
- return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);
- }
- // Checks that it's not trying to clobber one of the default fields of the class.
- function fieldNameIsValidForClass(fieldName, className) {
- if (!fieldNameIsValid(fieldName, className)) {
- return false;
- }
- if (defaultColumns._Default[fieldName]) {
- return false;
- }
- if (defaultColumns[className] && defaultColumns[className][fieldName]) {
- return false;
- }
- return true;
- }
- function invalidClassNameMessage(className) {
- return 'Invalid classname: ' + className + ', classnames can only have alphanumeric characters and _, and must start with an alpha character ';
- }
- const invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');
- const validNonRelationOrPointerTypes = ['Number', 'String', 'Boolean', 'Date', 'Object', 'Array', 'GeoPoint', 'File', 'Bytes', 'Polygon'];
- // Returns an error suitable for throwing if the type is invalid
- const fieldTypeIsInvalid = ({
- type,
- targetClass
- }) => {
- if (['Pointer', 'Relation'].indexOf(type) >= 0) {
- if (!targetClass) {
- return new Parse.Error(135, `type ${type} needs a class name`);
- } else if (typeof targetClass !== 'string') {
- return invalidJsonError;
- } else if (!classNameIsValid(targetClass)) {
- return new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(targetClass));
- } else {
- return undefined;
- }
- }
- if (typeof type !== 'string') {
- return invalidJsonError;
- }
- if (validNonRelationOrPointerTypes.indexOf(type) < 0) {
- return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);
- }
- return undefined;
- };
- const convertSchemaToAdapterSchema = schema => {
- schema = injectDefaultSchema(schema);
- delete schema.fields.ACL;
- schema.fields._rperm = {
- type: 'Array'
- };
- schema.fields._wperm = {
- type: 'Array'
- };
- if (schema.className === '_User') {
- delete schema.fields.password;
- schema.fields._hashed_password = {
- type: 'String'
- };
- }
- return schema;
- };
- exports.convertSchemaToAdapterSchema = convertSchemaToAdapterSchema;
- const convertAdapterSchemaToParseSchema = _ref => {
- let schema = _extends({}, (_objectDestructuringEmpty(_ref), _ref));
- delete schema.fields._rperm;
- delete schema.fields._wperm;
- schema.fields.ACL = {
- type: 'ACL'
- };
- if (schema.className === '_User') {
- delete schema.fields.authData; //Auth data is implicit
- delete schema.fields._hashed_password;
- schema.fields.password = {
- type: 'String'
- };
- }
- if (schema.indexes && Object.keys(schema.indexes).length === 0) {
- delete schema.indexes;
- }
- return schema;
- };
- class SchemaData {
- constructor(allSchemas = [], protectedFields = {}) {
- this.__data = {};
- this.__protectedFields = protectedFields;
- allSchemas.forEach(schema => {
- if (volatileClasses.includes(schema.className)) {
- return;
- }
- Object.defineProperty(this, schema.className, {
- get: () => {
- if (!this.__data[schema.className]) {
- const data = {};
- data.fields = injectDefaultSchema(schema).fields;
- data.classLevelPermissions = (0, _deepcopy.default)(schema.classLevelPermissions);
- data.indexes = schema.indexes;
- const classProtectedFields = this.__protectedFields[schema.className];
- if (classProtectedFields) {
- for (const key in classProtectedFields) {
- const unq = new Set([...(data.classLevelPermissions.protectedFields[key] || []), ...classProtectedFields[key]]);
- data.classLevelPermissions.protectedFields[key] = Array.from(unq);
- }
- }
- this.__data[schema.className] = data;
- }
- return this.__data[schema.className];
- }
- });
- });
- // Inject the in-memory classes
- volatileClasses.forEach(className => {
- Object.defineProperty(this, className, {
- get: () => {
- if (!this.__data[className]) {
- const schema = injectDefaultSchema({
- className,
- fields: {},
- classLevelPermissions: {}
- });
- const data = {};
- data.fields = schema.fields;
- data.classLevelPermissions = schema.classLevelPermissions;
- data.indexes = schema.indexes;
- this.__data[className] = data;
- }
- return this.__data[className];
- }
- });
- });
- }
- }
- const injectDefaultSchema = ({
- className,
- fields,
- classLevelPermissions,
- indexes
- }) => {
- const defaultSchema = {
- className,
- fields: _objectSpread(_objectSpread(_objectSpread({}, defaultColumns._Default), defaultColumns[className] || {}), fields),
- classLevelPermissions
- };
- if (indexes && Object.keys(indexes).length !== 0) {
- defaultSchema.indexes = indexes;
- }
- return defaultSchema;
- };
- const _HooksSchema = {
- className: '_Hooks',
- fields: defaultColumns._Hooks
- };
- const _GlobalConfigSchema = {
- className: '_GlobalConfig',
- fields: defaultColumns._GlobalConfig
- };
- const _GraphQLConfigSchema = {
- className: '_GraphQLConfig',
- fields: defaultColumns._GraphQLConfig
- };
- const _PushStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
- className: '_PushStatus',
- fields: {},
- classLevelPermissions: {}
- }));
- const _JobStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
- className: '_JobStatus',
- fields: {},
- classLevelPermissions: {}
- }));
- const _JobScheduleSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
- className: '_JobSchedule',
- fields: {},
- classLevelPermissions: {}
- }));
- const _AudienceSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
- className: '_Audience',
- fields: defaultColumns._Audience,
- classLevelPermissions: {}
- }));
- const _IdempotencySchema = convertSchemaToAdapterSchema(injectDefaultSchema({
- className: '_Idempotency',
- fields: defaultColumns._Idempotency,
- classLevelPermissions: {}
- }));
- const VolatileClassesSchemas = exports.VolatileClassesSchemas = [_HooksSchema, _JobStatusSchema, _JobScheduleSchema, _PushStatusSchema, _GlobalConfigSchema, _GraphQLConfigSchema, _AudienceSchema, _IdempotencySchema];
- const dbTypeMatchesObjectType = (dbType, objectType) => {
- if (dbType.type !== objectType.type) return false;
- if (dbType.targetClass !== objectType.targetClass) return false;
- if (dbType === objectType.type) return true;
- if (dbType.type === objectType.type) return true;
- return false;
- };
- const typeToString = type => {
- if (typeof type === 'string') {
- return type;
- }
- if (type.targetClass) {
- return `${type.type}<${type.targetClass}>`;
- }
- return `${type.type}`;
- };
- const ttl = {
- date: Date.now(),
- duration: undefined
- };
- // Stores the entire schema of the app in a weird hybrid format somewhere between
- // the mongo format and the Parse format. Soon, this will all be Parse format.
- class SchemaController {
- constructor(databaseAdapter) {
- this._dbAdapter = databaseAdapter;
- const config = _Config.default.get(Parse.applicationId);
- this.schemaData = new SchemaData(_SchemaCache.default.all(), this.protectedFields);
- this.protectedFields = config.protectedFields;
- const customIds = config.allowCustomObjectId;
- const customIdRegEx = /^.{1,}$/u; // 1+ chars
- const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;
- this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;
- this._dbAdapter.watch(() => {
- this.reloadData({
- clearCache: true
- });
- });
- }
- async reloadDataIfNeeded() {
- if (this._dbAdapter.enableSchemaHooks) {
- return;
- }
- const {
- date,
- duration
- } = ttl || {};
- if (!duration) {
- return;
- }
- const now = Date.now();
- if (now - date > duration) {
- ttl.date = now;
- await this.reloadData({
- clearCache: true
- });
- }
- }
- reloadData(options = {
- clearCache: false
- }) {
- if (this.reloadDataPromise && !options.clearCache) {
- return this.reloadDataPromise;
- }
- this.reloadDataPromise = this.getAllClasses(options).then(allSchemas => {
- this.schemaData = new SchemaData(allSchemas, this.protectedFields);
- delete this.reloadDataPromise;
- }, err => {
- this.schemaData = new SchemaData();
- delete this.reloadDataPromise;
- throw err;
- }).then(() => {});
- return this.reloadDataPromise;
- }
- async getAllClasses(options = {
- clearCache: false
- }) {
- if (options.clearCache) {
- return this.setAllClasses();
- }
- await this.reloadDataIfNeeded();
- const cached = _SchemaCache.default.all();
- if (cached && cached.length) {
- return Promise.resolve(cached);
- }
- return this.setAllClasses();
- }
- setAllClasses() {
- return this._dbAdapter.getAllClasses().then(allSchemas => allSchemas.map(injectDefaultSchema)).then(allSchemas => {
- _SchemaCache.default.put(allSchemas);
- return allSchemas;
- });
- }
- getOneSchema(className, allowVolatileClasses = false, options = {
- clearCache: false
- }) {
- if (options.clearCache) {
- _SchemaCache.default.clear();
- }
- if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {
- const data = this.schemaData[className];
- return Promise.resolve({
- className,
- fields: data.fields,
- classLevelPermissions: data.classLevelPermissions,
- indexes: data.indexes
- });
- }
- const cached = _SchemaCache.default.get(className);
- if (cached && !options.clearCache) {
- return Promise.resolve(cached);
- }
- return this.setAllClasses().then(allSchemas => {
- const oneSchema = allSchemas.find(schema => schema.className === className);
- if (!oneSchema) {
- return Promise.reject(undefined);
- }
- return oneSchema;
- });
- }
- // Create a new class that includes the three default fields.
- // ACL is an implicit column that does not get an entry in the
- // _SCHEMAS database. Returns a promise that resolves with the
- // created schema, in mongo format.
- // on success, and rejects with an error on fail. Ensure you
- // have authorization (master key, or client class creation
- // enabled) before calling this function.
- async addClassIfNotExists(className, fields = {}, classLevelPermissions, indexes = {}) {
- var validationError = this.validateNewClass(className, fields, classLevelPermissions);
- if (validationError) {
- if (validationError instanceof Parse.Error) {
- return Promise.reject(validationError);
- } else if (validationError.code && validationError.error) {
- return Promise.reject(new Parse.Error(validationError.code, validationError.error));
- }
- return Promise.reject(validationError);
- }
- try {
- const adapterSchema = await this._dbAdapter.createClass(className, convertSchemaToAdapterSchema({
- fields,
- classLevelPermissions,
- indexes,
- className
- }));
- // TODO: Remove by updating schema cache directly
- await this.reloadData({
- clearCache: true
- });
- const parseSchema = convertAdapterSchemaToParseSchema(adapterSchema);
- return parseSchema;
- } catch (error) {
- if (error && error.code === Parse.Error.DUPLICATE_VALUE) {
- throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);
- } else {
- throw error;
- }
- }
- }
- updateClass(className, submittedFields, classLevelPermissions, indexes, database) {
- return this.getOneSchema(className).then(schema => {
- const existingFields = schema.fields;
- Object.keys(submittedFields).forEach(name => {
- const field = submittedFields[name];
- if (existingFields[name] && existingFields[name].type !== field.type && field.__op !== 'Delete') {
- throw new Parse.Error(255, `Field ${name} exists, cannot update.`);
- }
- if (!existingFields[name] && field.__op === 'Delete') {
- throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);
- }
- });
- delete existingFields._rperm;
- delete existingFields._wperm;
- const newSchema = buildMergedSchemaObject(existingFields, submittedFields);
- const defaultFields = defaultColumns[className] || defaultColumns._Default;
- const fullNewSchema = Object.assign({}, newSchema, defaultFields);
- const validationError = this.validateSchemaData(className, newSchema, classLevelPermissions, Object.keys(existingFields));
- if (validationError) {
- throw new Parse.Error(validationError.code, validationError.error);
- }
- // Finally we have checked to make sure the request is valid and we can start deleting fields.
- // Do all deletions first, then a single save to _SCHEMA collection to handle all additions.
- const deletedFields = [];
- const insertedFields = [];
- Object.keys(submittedFields).forEach(fieldName => {
- if (submittedFields[fieldName].__op === 'Delete') {
- deletedFields.push(fieldName);
- } else {
- insertedFields.push(fieldName);
- }
- });
- let deletePromise = Promise.resolve();
- if (deletedFields.length > 0) {
- deletePromise = this.deleteFields(deletedFields, className, database);
- }
- let enforceFields = [];
- return deletePromise // Delete Everything
- .then(() => this.reloadData({
- clearCache: true
- })) // Reload our Schema, so we have all the new values
- .then(() => {
- const promises = insertedFields.map(fieldName => {
- const type = submittedFields[fieldName];
- return this.enforceFieldExists(className, fieldName, type);
- });
- return Promise.all(promises);
- }).then(results => {
- enforceFields = results.filter(result => !!result);
- return this.setPermissions(className, classLevelPermissions, newSchema);
- }).then(() => this._dbAdapter.setIndexesWithSchemaFormat(className, indexes, schema.indexes, fullNewSchema)).then(() => this.reloadData({
- clearCache: true
- }))
- //TODO: Move this logic into the database adapter
- .then(() => {
- this.ensureFields(enforceFields);
- const schema = this.schemaData[className];
- const reloadedSchema = {
- className: className,
- fields: schema.fields,
- classLevelPermissions: schema.classLevelPermissions
- };
- if (schema.indexes && Object.keys(schema.indexes).length !== 0) {
- reloadedSchema.indexes = schema.indexes;
- }
- return reloadedSchema;
- });
- }).catch(error => {
- if (error === undefined) {
- throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} does not exist.`);
- } else {
- throw error;
- }
- });
- }
- // Returns a promise that resolves successfully to the new schema
- // object or fails with a reason.
- enforceClassExists(className) {
- if (this.schemaData[className]) {
- return Promise.resolve(this);
- }
- // We don't have this class. Update the schema
- return (
- // The schema update succeeded. Reload the schema
- this.addClassIfNotExists(className).catch(() => {
- // The schema update failed. This can be okay - it might
- // have failed because there's a race condition and a different
- // client is making the exact same schema update that we want.
- // So just reload the schema.
- return this.reloadData({
- clearCache: true
- });
- }).then(() => {
- // Ensure that the schema now validates
- if (this.schemaData[className]) {
- return this;
- } else {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);
- }
- }).catch(() => {
- // The schema still doesn't validate. Give up
- throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');
- })
- );
- }
- validateNewClass(className, fields = {}, classLevelPermissions) {
- if (this.schemaData[className]) {
- throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);
- }
- if (!classNameIsValid(className)) {
- return {
- code: Parse.Error.INVALID_CLASS_NAME,
- error: invalidClassNameMessage(className)
- };
- }
- return this.validateSchemaData(className, fields, classLevelPermissions, []);
- }
- validateSchemaData(className, fields, classLevelPermissions, existingFieldNames) {
- for (const fieldName in fields) {
- if (existingFieldNames.indexOf(fieldName) < 0) {
- if (!fieldNameIsValid(fieldName, className)) {
- return {
- code: Parse.Error.INVALID_KEY_NAME,
- error: 'invalid field name: ' + fieldName
- };
- }
- if (!fieldNameIsValidForClass(fieldName, className)) {
- return {
- code: 136,
- error: 'field ' + fieldName + ' cannot be added'
- };
- }
- const fieldType = fields[fieldName];
- const error = fieldTypeIsInvalid(fieldType);
- if (error) return {
- code: error.code,
- error: error.message
- };
- if (fieldType.defaultValue !== undefined) {
- let defaultValueType = getType(fieldType.defaultValue);
- if (typeof defaultValueType === 'string') {
- defaultValueType = {
- type: defaultValueType
- };
- } else if (typeof defaultValueType === 'object' && fieldType.type === 'Relation') {
- return {
- code: Parse.Error.INCORRECT_TYPE,
- error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`
- };
- }
- if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {
- return {
- code: Parse.Error.INCORRECT_TYPE,
- error: `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(fieldType)} but got ${typeToString(defaultValueType)}`
- };
- }
- } else if (fieldType.required) {
- if (typeof fieldType === 'object' && fieldType.type === 'Relation') {
- return {
- code: Parse.Error.INCORRECT_TYPE,
- error: `The 'required' option is not applicable for ${typeToString(fieldType)}`
- };
- }
- }
- }
- }
- for (const fieldName in defaultColumns[className]) {
- fields[fieldName] = defaultColumns[className][fieldName];
- }
- const geoPoints = Object.keys(fields).filter(key => fields[key] && fields[key].type === 'GeoPoint');
- if (geoPoints.length > 1) {
- return {
- code: Parse.Error.INCORRECT_TYPE,
- error: 'currently, only one GeoPoint field may exist in an object. Adding ' + geoPoints[1] + ' when ' + geoPoints[0] + ' already exists.'
- };
- }
- validateCLP(classLevelPermissions, fields, this.userIdRegEx);
- }
- // Sets the Class-level permissions for a given className, which must exist.
- async setPermissions(className, perms, newSchema) {
- if (typeof perms === 'undefined') {
- return Promise.resolve();
- }
- validateCLP(perms, newSchema, this.userIdRegEx);
- await this._dbAdapter.setClassLevelPermissions(className, perms);
- const cached = _SchemaCache.default.get(className);
- if (cached) {
- cached.classLevelPermissions = perms;
- }
- }
- // Returns a promise that resolves successfully to the new schema
- // object if the provided className-fieldName-type tuple is valid.
- // The className must already be validated.
- // If 'freeze' is true, refuse to update the schema for this field.
- enforceFieldExists(className, fieldName, type, isValidation, maintenance) {
- if (fieldName.indexOf('.') > 0) {
- // "<array>.<index>" for Nested Arrays
- // "<embedded document>.<field>" for Nested Objects
- // JSON Arrays are treated as Nested Objects
- const [x, y] = fieldName.split('.');
- fieldName = x;
- const isArrayIndex = Array.from(y).every(c => c >= '0' && c <= '9');
- if (isArrayIndex && !['sentPerUTCOffset', 'failedPerUTCOffset'].includes(fieldName)) {
- type = 'Array';
- } else {
- type = 'Object';
- }
- }
- let fieldNameToValidate = `${fieldName}`;
- if (maintenance && fieldNameToValidate.charAt(0) === '_') {
- fieldNameToValidate = fieldNameToValidate.substring(1);
- }
- if (!fieldNameIsValid(fieldNameToValidate, className)) {
- throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);
- }
- // If someone tries to create a new field with null/undefined as the value, return;
- if (!type) {
- return undefined;
- }
- const expectedType = this.getExpectedType(className, fieldName);
- if (typeof type === 'string') {
- type = {
- type
- };
- }
- if (type.defaultValue !== undefined) {
- let defaultValueType = getType(type.defaultValue);
- if (typeof defaultValueType === 'string') {
- defaultValueType = {
- type: defaultValueType
- };
- }
- if (!dbTypeMatchesObjectType(type, defaultValueType)) {
- throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(type)} but got ${typeToString(defaultValueType)}`);
- }
- }
- if (expectedType) {
- if (!dbTypeMatchesObjectType(expectedType, type)) {
- throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName}; expected ${typeToString(expectedType)} but got ${typeToString(type)}`);
- }
- // If type options do not change
- // we can safely return
- if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {
- return undefined;
- }
- // Field options are may be changed
- // ensure to have an update to date schema field
- return this._dbAdapter.updateFieldOptions(className, fieldName, type);
- }
- return this._dbAdapter.addFieldIfNotExists(className, fieldName, type).catch(error => {
- if (error.code == Parse.Error.INCORRECT_TYPE) {
- // Make sure that we throw errors when it is appropriate to do so.
- throw error;
- }
- // The update failed. This can be okay - it might have been a race
- // condition where another client updated the schema in the same
- // way that we wanted to. So, just reload the schema
- return Promise.resolve();
- }).then(() => {
- return {
- className,
- fieldName,
- type
- };
- });
- }
- ensureFields(fields) {
- for (let i = 0; i < fields.length; i += 1) {
- const {
- className,
- fieldName
- } = fields[i];
- let {
- type
- } = fields[i];
- const expectedType = this.getExpectedType(className, fieldName);
- if (typeof type === 'string') {
- type = {
- type: type
- };
- }
- if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {
- throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);
- }
- }
- }
- // maintain compatibility
- deleteField(fieldName, className, database) {
- return this.deleteFields([fieldName], className, database);
- }
- // Delete fields, and remove that data from all objects. This is intended
- // to remove unused fields, if other writers are writing objects that include
- // this field, the field may reappear. Returns a Promise that resolves with
- // no object on success, or rejects with { code, error } on failure.
- // Passing the database and prefix is necessary in order to drop relation collections
- // and remove fields from objects. Ideally the database would belong to
- // a database adapter and this function would close over it or access it via member.
- deleteFields(fieldNames, className, database) {
- if (!classNameIsValid(className)) {
- throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));
- }
- fieldNames.forEach(fieldName => {
- if (!fieldNameIsValid(fieldName, className)) {
- throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);
- }
- //Don't allow deleting the default fields.
- if (!fieldNameIsValidForClass(fieldName, className)) {
- throw new Parse.Error(136, `field ${fieldName} cannot be changed`);
- }
- });
- return this.getOneSchema(className, false, {
- clearCache: true
- }).catch(error => {
- if (error === undefined) {
- throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} does not exist.`);
- } else {
- throw error;
- }
- }).then(schema => {
- fieldNames.forEach(fieldName => {
- if (!schema.fields[fieldName]) {
- throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);
- }
- });
- const schemaFields = _objectSpread({}, schema.fields);
- return database.adapter.deleteFields(className, schema, fieldNames).then(() => {
- return Promise.all(fieldNames.map(fieldName => {
- const field = schemaFields[fieldName];
- if (field && field.type === 'Relation') {
- //For relations, drop the _Join table
- return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);
- }
- return Promise.resolve();
- }));
- });
- }).then(() => {
- _SchemaCache.default.clear();
- });
- }
- // Validates an object provided in REST format.
- // Returns a promise that resolves to the new schema if this object is
- // valid.
- async validateObject(className, object, query, maintenance) {
- let geocount = 0;
- const schema = await this.enforceClassExists(className);
- const promises = [];
- for (const fieldName in object) {
- if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {
- geocount++;
- }
- if (geocount > 1) {
- return Promise.reject(new Parse.Error(Parse.Error.INCORRECT_TYPE, 'there can only be one geopoint field in a class'));
- }
- }
- for (const fieldName in object) {
- if (object[fieldName] === undefined) {
- continue;
- }
- const expected = getType(object[fieldName]);
- if (!expected) {
- continue;
- }
- if (fieldName === 'ACL') {
- // Every object has ACL implicitly.
- continue;
- }
- promises.push(schema.enforceFieldExists(className, fieldName, expected, true, maintenance));
- }
- const results = await Promise.all(promises);
- const enforceFields = results.filter(result => !!result);
- if (enforceFields.length !== 0) {
- // TODO: Remove by updating schema cache directly
- await this.reloadData({
- clearCache: true
- });
- }
- this.ensureFields(enforceFields);
- const promise = Promise.resolve(schema);
- return thenValidateRequiredColumns(promise, className, object, query);
- }
- // Validates that all the properties are set for the object
- validateRequiredColumns(className, object, query) {
- const columns = requiredColumns.write[className];
- if (!columns || columns.length == 0) {
- return Promise.resolve(this);
- }
- const missingColumns = columns.filter(function (column) {
- if (query && query.objectId) {
- if (object[column] && typeof object[column] === 'object') {
- // Trying to delete a required column
- return object[column].__op == 'Delete';
- }
- // Not trying to do anything there
- return false;
- }
- return !object[column];
- });
- if (missingColumns.length > 0) {
- throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');
- }
- return Promise.resolve(this);
- }
- testPermissionsForClassName(className, aclGroup, operation) {
- return SchemaController.testPermissions(this.getClassLevelPermissions(className), aclGroup, operation);
- }
- // Tests that the class level permission let pass the operation for a given aclGroup
- static testPermissions(classPermissions, aclGroup, operation) {
- if (!classPermissions || !classPermissions[operation]) {
- return true;
- }
- const perms = classPermissions[operation];
- if (perms['*']) {
- return true;
- }
- // Check permissions against the aclGroup provided (array of userId/roles)
- if (aclGroup.some(acl => {
- return perms[acl] === true;
- })) {
- return true;
- }
- return false;
- }
- // Validates an operation passes class-level-permissions set in the schema
- static validatePermission(classPermissions, className, aclGroup, operation, action) {
- if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {
- return Promise.resolve();
- }
- if (!classPermissions || !classPermissions[operation]) {
- return true;
- }
- const perms = classPermissions[operation];
- // If only for authenticated users
- // make sure we have an aclGroup
- if (perms['requiresAuthentication']) {
- // If aclGroup has * (public)
- if (!aclGroup || aclGroup.length == 0) {
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.');
- } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.');
- }
- // requiresAuthentication passed, just move forward
- // probably would be wise at some point to rename to 'authenticatedUser'
- return Promise.resolve();
- }
- // No matching CLP, let's check the Pointer permissions
- // And handle those later
- const permissionField = ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';
- // Reject create when write lockdown
- if (permissionField == 'writeUserFields' && operation == 'create') {
- throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`);
- }
- // Process the readUserFields later
- if (Array.isArray(classPermissions[permissionField]) && classPermissions[permissionField].length > 0) {
- return Promise.resolve();
- }
- const pointerFields = classPermissions[operation].pointerFields;
- if (Array.isArray(pointerFields) && pointerFields.length > 0) {
- // any op except 'addField as part of create' is ok.
- if (operation !== 'addField' || action === 'update') {
- // We can allow adding field on update flow only.
- return Promise.resolve();
- }
- }
- throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`);
- }
- // Validates an operation passes class-level-permissions set in the schema
- validatePermission(className, aclGroup, operation, action) {
- return SchemaController.validatePermission(this.getClassLevelPermissions(className), className, aclGroup, operation, action);
- }
- getClassLevelPermissions(className) {
- return this.schemaData[className] && this.schemaData[className].classLevelPermissions;
- }
- // Returns the expected type for a className+key combination
- // or undefined if the schema is not set
- getExpectedType(className, fieldName) {
- if (this.schemaData[className]) {
- const expectedType = this.schemaData[className].fields[fieldName];
- return expectedType === 'map' ? 'Object' : expectedType;
- }
- return undefined;
- }
- // Checks if a given class is in the schema.
- hasClass(className) {
- if (this.schemaData[className]) {
- return Promise.resolve(true);
- }
- return this.reloadData().then(() => !!this.schemaData[className]);
- }
- }
- // Returns a promise for a new Schema.
- exports.SchemaController = exports.default = SchemaController;
- const load = (dbAdapter, options) => {
- const schema = new SchemaController(dbAdapter);
- ttl.duration = dbAdapter.schemaCacheTtl;
- return schema.reloadData(options).then(() => schema);
- };
- // Builds a new schema (in schema API response format) out of an
- // existing mongo schema + a schemas API put request. This response
- // does not include the default fields, as it is intended to be passed
- // to mongoSchemaFromFieldsAndClassName. No validation is done here, it
- // is done in mongoSchemaFromFieldsAndClassName.
- exports.load = load;
- function buildMergedSchemaObject(existingFields, putRequest) {
- const newSchema = {};
- // -disable-next
- const sysSchemaField = Object.keys(defaultColumns).indexOf(existingFields._id) === -1 ? [] : Object.keys(defaultColumns[existingFields._id]);
- for (const oldField in existingFields) {
- if (oldField !== '_id' && oldField !== 'ACL' && oldField !== 'updatedAt' && oldField !== 'createdAt' && oldField !== 'objectId') {
- if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {
- continue;
- }
- const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete';
- if (!fieldIsDeleted) {
- newSchema[oldField] = existingFields[oldField];
- }
- }
- }
- for (const newField in putRequest) {
- if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {
- if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {
- continue;
- }
- newSchema[newField] = putRequest[newField];
- }
- }
- return newSchema;
- }
- // Given a schema promise, construct another schema promise that
- // validates this field once the schema loads.
- function thenValidateRequiredColumns(schemaPromise, className, object, query) {
- return schemaPromise.then(schema => {
- return schema.validateRequiredColumns(className, object, query);
- });
- }
- // Gets the type from a REST API formatted object, where 'type' is
- // extended past javascript types to include the rest of the Parse
- // type system.
- // The output should be a valid schema value.
- // TODO: ensure that this is compatible with the format used in Open DB
- function getType(obj) {
- const type = typeof obj;
- switch (type) {
- case 'boolean':
- return 'Boolean';
- case 'string':
- return 'String';
- case 'number':
- return 'Number';
- case 'map':
- case 'object':
- if (!obj) {
- return undefined;
- }
- return getObjectType(obj);
- case 'function':
- case 'symbol':
- case 'undefined':
- default:
- throw 'bad obj: ' + obj;
- }
- }
- // This gets the type for non-JSON types like pointers and files, but
- // also gets the appropriate type for $ operators.
- // Returns null if the type is unknown.
- function getObjectType(obj) {
- if (obj instanceof Array) {
- return 'Array';
- }
- if (obj.__type) {
- switch (obj.__type) {
- case 'Pointer':
- if (obj.className) {
- return {
- type: 'Pointer',
- targetClass: obj.className
- };
- }
- break;
- case 'Relation':
- if (obj.className) {
- return {
- type: 'Relation',
- targetClass: obj.className
- };
- }
- break;
- case 'File':
- if (obj.name) {
- return 'File';
- }
- break;
- case 'Date':
- if (obj.iso) {
- return 'Date';
- }
- break;
- case 'GeoPoint':
- if (obj.latitude != null && obj.longitude != null) {
- return 'GeoPoint';
- }
- break;
- case 'Bytes':
- if (obj.base64) {
- return 'Bytes';
- }
- break;
- case 'Polygon':
- if (obj.coordinates) {
- return 'Polygon';
- }
- break;
- }
- throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);
- }
- if (obj['$ne']) {
- return getObjectType(obj['$ne']);
- }
- if (obj.__op) {
- switch (obj.__op) {
- case 'Increment':
- return 'Number';
- case 'Delete':
- return null;
- case 'Add':
- case 'AddUnique':
- case 'Remove':
- return 'Array';
- case 'AddRelation':
- case 'RemoveRelation':
- return {
- type: 'Relation',
- targetClass: obj.objects[0].className
- };
- case 'Batch':
- return getObjectType(obj.ops[0]);
- default:
- throw 'unexpected op: ' + obj.__op;
- }
- }
- return 'Object';
- }
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_StorageAdapter","require","_SchemaCache","_interopRequireDefault","_DatabaseController","_Config","_deepcopy","e","__esModule","default","ownKeys","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty","getOwnPropertyDescriptors","defineProperties","defineProperty","_toPropertyKey","value","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","_objectDestructuringEmpty","_extends","assign","bind","n","hasOwnProperty","Parse","defaultColumns","exports","freeze","_Default","objectId","type","createdAt","updatedAt","ACL","_User","username","password","email","emailVerified","authData","_Installation","installationId","deviceToken","channels","deviceType","pushType","GCMSenderId","timeZone","localeIdentifier","badge","appVersion","appName","appIdentifier","parseVersion","_Role","name","users","targetClass","roles","_Session","user","sessionToken","expiresAt","createdWith","_Product","productIdentifier","download","downloadName","icon","order","title","subtitle","_PushStatus","pushTime","source","query","payload","expiry","expiration_interval","status","numSent","numFailed","pushHash","errorMessage","sentPerType","failedPerType","sentPerUTCOffset","failedPerUTCOffset","count","_JobStatus","jobName","message","params","finishedAt","_JobSchedule","description","startAfter","daysOfWeek","timeOfDay","lastRun","repeatMinutes","_Hooks","functionName","className","triggerName","url","_GlobalConfig","masterKeyOnly","_GraphQLConfig","config","_Audience","lastUsed","timesUsed","_Idempotency","reqId","expire","requiredColumns","read","write","invalidColumns","systemClasses","volatileClasses","roleRegex","protectedFieldsPointerRegex","publicRegex","authenticatedRegex","requiresAuthenticationRegex","clpPointerRegex","protectedFieldsRegex","clpFieldsRegex","validatePermissionKey","key","userIdRegExp","matchesSome","regEx","match","valid","Error","INVALID_JSON","validateProtectedFieldsKey","CLPValidKeys","validateCLP","perms","fields","operationKey","indexOf","operation","validateCLPjson","fieldName","validatePointerPermission","entity","protectedFields","Array","isArray","field","prototype","pointerFields","pointerField","permit","joinClassRegex","classAndFieldRegex","classNameIsValid","test","fieldNameIsValid","includes","fieldNameIsValidForClass","invalidClassNameMessage","invalidJsonError","validNonRelationOrPointerTypes","fieldTypeIsInvalid","INVALID_CLASS_NAME","undefined","INCORRECT_TYPE","convertSchemaToAdapterSchema","schema","injectDefaultSchema","_rperm","_wperm","_hashed_password","convertAdapterSchemaToParseSchema","_ref","indexes","SchemaData","constructor","allSchemas","__data","__protectedFields","get","data","classLevelPermissions","deepcopy","classProtectedFields","unq","Set","from","defaultSchema","_HooksSchema","_GlobalConfigSchema","_GraphQLConfigSchema","_PushStatusSchema","_JobStatusSchema","_JobScheduleSchema","_AudienceSchema","_IdempotencySchema","VolatileClassesSchemas","dbTypeMatchesObjectType","dbType","objectType","typeToString","ttl","date","Date","now","duration","SchemaController","databaseAdapter","_dbAdapter","Config","applicationId","schemaData","SchemaCache","all","customIds","allowCustomObjectId","customIdRegEx","autoIdRegEx","userIdRegEx","watch","reloadData","clearCache","reloadDataIfNeeded","enableSchemaHooks","options","reloadDataPromise","getAllClasses","then","err","setAllClasses","cached","Promise","resolve","map","put","getOneSchema","allowVolatileClasses","clear","oneSchema","find","reject","addClassIfNotExists","validationError","validateNewClass","code","error","adapterSchema","createClass","parseSchema","DUPLICATE_VALUE","updateClass","submittedFields","database","existingFields","__op","newSchema","buildMergedSchemaObject","defaultFields","fullNewSchema","validateSchemaData","deletedFields","insertedFields","deletePromise","deleteFields","enforceFields","promises","enforceFieldExists","results","result","setPermissions","setIndexesWithSchemaFormat","ensureFields","reloadedSchema","catch","enforceClassExists","existingFieldNames","INVALID_KEY_NAME","fieldType","defaultValue","defaultValueType","getType","required","geoPoints","setClassLevelPermissions","isValidation","maintenance","x","y","split","isArrayIndex","every","c","fieldNameToValidate","charAt","substring","expectedType","getExpectedType","JSON","stringify","updateFieldOptions","addFieldIfNotExists","deleteField","fieldNames","schemaFields","adapter","deleteClass","validateObject","object","geocount","expected","promise","thenValidateRequiredColumns","validateRequiredColumns","columns","missingColumns","column","testPermissionsForClassName","aclGroup","testPermissions","getClassLevelPermissions","classPermissions","some","acl","validatePermission","action","OBJECT_NOT_FOUND","permissionField","OPERATION_FORBIDDEN","hasClass","load","dbAdapter","schemaCacheTtl","putRequest","sysSchemaField","_id","oldField","fieldIsDeleted","newField","schemaPromise","obj","getObjectType","__type","iso","latitude","longitude","base64","coordinates","objects","ops"],"sources":["../../src/Controllers/SchemaController.js"],"sourcesContent":["// @flow\n// This class handles schema validation, persistence, and modification.\n//\n// Each individual Schema object should be immutable. The helpers to\n// do things with the Schema just return a new schema when the schema\n// is changed.\n//\n// The canonical place to store this Schema is in the database itself,\n// in a _SCHEMA collection. This is not the right way to do it for an\n// open source framework, but it's backward compatible, so we're\n// keeping it this way for now.\n//\n// In API-handling code, you should only use the Schema class via the\n// DatabaseController. This will let us replace the schema logic for\n// different databases.\n// TODO: hide all schema logic inside the database adapter.\n// @flow-disable-next\nconst Parse = require('parse/node').Parse;\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport SchemaCache from '../Adapters/Cache/SchemaCache';\nimport DatabaseController from './DatabaseController';\nimport Config from '../Config';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport type {\n  Schema,\n  SchemaFields,\n  ClassLevelPermissions,\n  SchemaField,\n  LoadSchemaOptions,\n} from './types';\n\nconst defaultColumns: { [string]: SchemaFields } = Object.freeze({\n  // Contain the default columns for every parse object type (except _Join collection)\n  _Default: {\n    objectId: { type: 'String' },\n    createdAt: { type: 'Date' },\n    updatedAt: { type: 'Date' },\n    ACL: { type: 'ACL' },\n  },\n  // The additional default columns for the _User collection (in addition to DefaultCols)\n  _User: {\n    username: { type: 'String' },\n    password: { type: 'String' },\n    email: { type: 'String' },\n    emailVerified: { type: 'Boolean' },\n    authData: { type: 'Object' },\n  },\n  // The additional default columns for the _Installation collection (in addition to DefaultCols)\n  _Installation: {\n    installationId: { type: 'String' },\n    deviceToken: { type: 'String' },\n    channels: { type: 'Array' },\n    deviceType: { type: 'String' },\n    pushType: { type: 'String' },\n    GCMSenderId: { type: 'String' },\n    timeZone: { type: 'String' },\n    localeIdentifier: { type: 'String' },\n    badge: { type: 'Number' },\n    appVersion: { type: 'String' },\n    appName: { type: 'String' },\n    appIdentifier: { type: 'String' },\n    parseVersion: { type: 'String' },\n  },\n  // The additional default columns for the _Role collection (in addition to DefaultCols)\n  _Role: {\n    name: { type: 'String' },\n    users: { type: 'Relation', targetClass: '_User' },\n    roles: { type: 'Relation', targetClass: '_Role' },\n  },\n  // The additional default columns for the _Session collection (in addition to DefaultCols)\n  _Session: {\n    user: { type: 'Pointer', targetClass: '_User' },\n    installationId: { type: 'String' },\n    sessionToken: { type: 'String' },\n    expiresAt: { type: 'Date' },\n    createdWith: { type: 'Object' },\n  },\n  _Product: {\n    productIdentifier: { type: 'String' },\n    download: { type: 'File' },\n    downloadName: { type: 'String' },\n    icon: { type: 'File' },\n    order: { type: 'Number' },\n    title: { type: 'String' },\n    subtitle: { type: 'String' },\n  },\n  _PushStatus: {\n    pushTime: { type: 'String' },\n    source: { type: 'String' }, // rest or webui\n    query: { type: 'String' }, // the stringified JSON query\n    payload: { type: 'String' }, // the stringified JSON payload,\n    title: { type: 'String' },\n    expiry: { type: 'Number' },\n    expiration_interval: { type: 'Number' },\n    status: { type: 'String' },\n    numSent: { type: 'Number' },\n    numFailed: { type: 'Number' },\n    pushHash: { type: 'String' },\n    errorMessage: { type: 'Object' },\n    sentPerType: { type: 'Object' },\n    failedPerType: { type: 'Object' },\n    sentPerUTCOffset: { type: 'Object' },\n    failedPerUTCOffset: { type: 'Object' },\n    count: { type: 'Number' }, // tracks # of batches queued and pending\n  },\n  _JobStatus: {\n    jobName: { type: 'String' },\n    source: { type: 'String' },\n    status: { type: 'String' },\n    message: { type: 'String' },\n    params: { type: 'Object' }, // params received when calling the job\n    finishedAt: { type: 'Date' },\n  },\n  _JobSchedule: {\n    jobName: { type: 'String' },\n    description: { type: 'String' },\n    params: { type: 'String' },\n    startAfter: { type: 'String' },\n    daysOfWeek: { type: 'Array' },\n    timeOfDay: { type: 'String' },\n    lastRun: { type: 'Number' },\n    repeatMinutes: { type: 'Number' },\n  },\n  _Hooks: {\n    functionName: { type: 'String' },\n    className: { type: 'String' },\n    triggerName: { type: 'String' },\n    url: { type: 'String' },\n  },\n  _GlobalConfig: {\n    objectId: { type: 'String' },\n    params: { type: 'Object' },\n    masterKeyOnly: { type: 'Object' },\n  },\n  _GraphQLConfig: {\n    objectId: { type: 'String' },\n    config: { type: 'Object' },\n  },\n  _Audience: {\n    objectId: { type: 'String' },\n    name: { type: 'String' },\n    query: { type: 'String' }, //storing query as JSON string to prevent \"Nested keys should not contain the '$' or '.' characters\" error\n    lastUsed: { type: 'Date' },\n    timesUsed: { type: 'Number' },\n  },\n  _Idempotency: {\n    reqId: { type: 'String' },\n    expire: { type: 'Date' },\n  },\n});\n\n// fields required for read or write operations on their respective classes.\nconst requiredColumns = Object.freeze({\n  read: {\n    _User: ['username'],\n  },\n  write: {\n    _Product: ['productIdentifier', 'icon', 'order', 'title', 'subtitle'],\n    _Role: ['name', 'ACL'],\n  },\n});\n\nconst invalidColumns = ['length'];\n\nconst systemClasses = Object.freeze([\n  '_User',\n  '_Installation',\n  '_Role',\n  '_Session',\n  '_Product',\n  '_PushStatus',\n  '_JobStatus',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\nconst volatileClasses = Object.freeze([\n  '_JobStatus',\n  '_PushStatus',\n  '_Hooks',\n  '_GlobalConfig',\n  '_GraphQLConfig',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\n// Anything that start with role\nconst roleRegex = /^role:.*/;\n// Anything that starts with userField (allowed for protected fields only)\nconst protectedFieldsPointerRegex = /^userField:.*/;\n// * permission\nconst publicRegex = /^\\*$/;\n\nconst authenticatedRegex = /^authenticated$/;\n\nconst requiresAuthenticationRegex = /^requiresAuthentication$/;\n\nconst clpPointerRegex = /^pointerFields$/;\n\n// regex for validating entities in protectedFields object\nconst protectedFieldsRegex = Object.freeze([\n  protectedFieldsPointerRegex,\n  publicRegex,\n  authenticatedRegex,\n  roleRegex,\n]);\n\n// clp regex\nconst clpFieldsRegex = Object.freeze([\n  clpPointerRegex,\n  publicRegex,\n  requiresAuthenticationRegex,\n  roleRegex,\n]);\n\nfunction validatePermissionKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of clpFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId depends on startup options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nfunction validateProtectedFieldsKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of protectedFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId regex depends on launch options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nconst CLPValidKeys = Object.freeze([\n  'find',\n  'count',\n  'get',\n  'create',\n  'update',\n  'delete',\n  'addField',\n  'readUserFields',\n  'writeUserFields',\n  'protectedFields',\n]);\n\n// validation before setting class-level permissions on collection\nfunction validateCLP(perms: ClassLevelPermissions, fields: SchemaFields, userIdRegExp: RegExp) {\n  if (!perms) {\n    return;\n  }\n  for (const operationKey in perms) {\n    if (CLPValidKeys.indexOf(operationKey) == -1) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `${operationKey} is not a valid operation for class level permissions`\n      );\n    }\n\n    const operation = perms[operationKey];\n    // proceed with next operationKey\n\n    // throws when root fields are of wrong type\n    validateCLPjson(operation, operationKey);\n\n    if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n      // validate grouped pointer permissions\n      // must be an array with field names\n      for (const fieldName of operation) {\n        validatePointerPermission(fieldName, fields, operationKey);\n      }\n      // readUserFields and writerUserFields do not have nesdted fields\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate protected fields\n    if (operationKey === 'protectedFields') {\n      for (const entity in operation) {\n        // throws on unexpected key\n        validateProtectedFieldsKey(entity, userIdRegExp);\n\n        const protectedFields = operation[entity];\n\n        if (!Array.isArray(protectedFields)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`\n          );\n        }\n\n        // if the field is in form of array\n        for (const field of protectedFields) {\n          // do not alloow to protect default fields\n          if (defaultColumns._Default[field]) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Default field '${field}' can not be protected`\n            );\n          }\n          // field should exist on collection\n          if (!Object.prototype.hasOwnProperty.call(fields, field)) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Field '${field}' in protectedFields:${entity} does not exist`\n            );\n          }\n        }\n      }\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate other fields\n    // Entity can be:\n    // \"*\" - Public,\n    // \"requiresAuthentication\" - authenticated users,\n    // \"objectId\" - _User id,\n    // \"role:rolename\",\n    // \"pointerFields\" - array of field names containing pointers to users\n    for (const entity in operation) {\n      // throws on unexpected key\n      validatePermissionKey(entity, userIdRegExp);\n\n      // entity can be either:\n      // \"pointerFields\": string[]\n      if (entity === 'pointerFields') {\n        const pointerFields = operation[entity];\n\n        if (Array.isArray(pointerFields)) {\n          for (const pointerField of pointerFields) {\n            validatePointerPermission(pointerField, fields, operation);\n          }\n        } else {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`\n          );\n        }\n        // proceed with next entity key\n        continue;\n      }\n\n      // or [entity]: boolean\n      const permit = operation[entity];\n\n      if (permit !== true) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `'${permit}' is not a valid value for class level permissions ${operationKey}:${entity}:${permit}`\n        );\n      }\n    }\n  }\n}\n\nfunction validateCLPjson(operation: any, operationKey: string) {\n  if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n    if (!Array.isArray(operation)) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an array`\n      );\n    }\n  } else {\n    if (typeof operation === 'object' && operation !== null) {\n      // ok to proceed\n      return;\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an object`\n      );\n    }\n  }\n}\n\nfunction validatePointerPermission(fieldName: string, fields: Object, operation: string) {\n  // Uses collection schema to ensure the field is of type:\n  // - Pointer<_User> (pointers)\n  // - Array\n  //\n  //    It's not possible to enforce type on Array's items in schema\n  //  so we accept any Array field, and later when applying permissions\n  //  only items that are pointers to _User are considered.\n  if (\n    !(\n      fields[fieldName] &&\n      ((fields[fieldName].type == 'Pointer' && fields[fieldName].targetClass == '_User') ||\n        fields[fieldName].type == 'Array')\n    )\n  ) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`\n    );\n  }\n}\n\nconst joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;\nconst classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;\nfunction classNameIsValid(className: string): boolean {\n  // Valid classes must:\n  return (\n    // Be one of _User, _Installation, _Role, _Session OR\n    systemClasses.indexOf(className) > -1 ||\n    // Be a join table OR\n    joinClassRegex.test(className) ||\n    // Include only alpha-numeric and underscores, and not start with an underscore or number\n    fieldNameIsValid(className, className)\n  );\n}\n\n// Valid fields must be alpha-numeric, and not start with an underscore or number\n// must not be a reserved key\nfunction fieldNameIsValid(fieldName: string, className: string): boolean {\n  if (className && className !== '_Hooks') {\n    if (fieldName === 'className') {\n      return false;\n    }\n  }\n  return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);\n}\n\n// Checks that it's not trying to clobber one of the default fields of the class.\nfunction fieldNameIsValidForClass(fieldName: string, className: string): boolean {\n  if (!fieldNameIsValid(fieldName, className)) {\n    return false;\n  }\n  if (defaultColumns._Default[fieldName]) {\n    return false;\n  }\n  if (defaultColumns[className] && defaultColumns[className][fieldName]) {\n    return false;\n  }\n  return true;\n}\n\nfunction invalidClassNameMessage(className: string): string {\n  return (\n    'Invalid classname: ' +\n    className +\n    ', classnames can only have alphanumeric characters and _, and must start with an alpha character '\n  );\n}\n\nconst invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');\nconst validNonRelationOrPointerTypes = [\n  'Number',\n  'String',\n  'Boolean',\n  'Date',\n  'Object',\n  'Array',\n  'GeoPoint',\n  'File',\n  'Bytes',\n  'Polygon',\n];\n// Returns an error suitable for throwing if the type is invalid\nconst fieldTypeIsInvalid = ({ type, targetClass }) => {\n  if (['Pointer', 'Relation'].indexOf(type) >= 0) {\n    if (!targetClass) {\n      return new Parse.Error(135, `type ${type} needs a class name`);\n    } else if (typeof targetClass !== 'string') {\n      return invalidJsonError;\n    } else if (!classNameIsValid(targetClass)) {\n      return new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(targetClass));\n    } else {\n      return undefined;\n    }\n  }\n  if (typeof type !== 'string') {\n    return invalidJsonError;\n  }\n  if (validNonRelationOrPointerTypes.indexOf(type) < 0) {\n    return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);\n  }\n  return undefined;\n};\n\nconst convertSchemaToAdapterSchema = (schema: any) => {\n  schema = injectDefaultSchema(schema);\n  delete schema.fields.ACL;\n  schema.fields._rperm = { type: 'Array' };\n  schema.fields._wperm = { type: 'Array' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.password;\n    schema.fields._hashed_password = { type: 'String' };\n  }\n\n  return schema;\n};\n\nconst convertAdapterSchemaToParseSchema = ({ ...schema }) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  schema.fields.ACL = { type: 'ACL' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.authData; //Auth data is implicit\n    delete schema.fields._hashed_password;\n    schema.fields.password = { type: 'String' };\n  }\n\n  if (schema.indexes && Object.keys(schema.indexes).length === 0) {\n    delete schema.indexes;\n  }\n\n  return schema;\n};\n\nclass SchemaData {\n  __data: any;\n  __protectedFields: any;\n  constructor(allSchemas = [], protectedFields = {}) {\n    this.__data = {};\n    this.__protectedFields = protectedFields;\n    allSchemas.forEach(schema => {\n      if (volatileClasses.includes(schema.className)) {\n        return;\n      }\n      Object.defineProperty(this, schema.className, {\n        get: () => {\n          if (!this.__data[schema.className]) {\n            const data = {};\n            data.fields = injectDefaultSchema(schema).fields;\n            data.classLevelPermissions = deepcopy(schema.classLevelPermissions);\n            data.indexes = schema.indexes;\n\n            const classProtectedFields = this.__protectedFields[schema.className];\n            if (classProtectedFields) {\n              for (const key in classProtectedFields) {\n                const unq = new Set([\n                  ...(data.classLevelPermissions.protectedFields[key] || []),\n                  ...classProtectedFields[key],\n                ]);\n                data.classLevelPermissions.protectedFields[key] = Array.from(unq);\n              }\n            }\n\n            this.__data[schema.className] = data;\n          }\n          return this.__data[schema.className];\n        },\n      });\n    });\n\n    // Inject the in-memory classes\n    volatileClasses.forEach(className => {\n      Object.defineProperty(this, className, {\n        get: () => {\n          if (!this.__data[className]) {\n            const schema = injectDefaultSchema({\n              className,\n              fields: {},\n              classLevelPermissions: {},\n            });\n            const data = {};\n            data.fields = schema.fields;\n            data.classLevelPermissions = schema.classLevelPermissions;\n            data.indexes = schema.indexes;\n            this.__data[className] = data;\n          }\n          return this.__data[className];\n        },\n      });\n    });\n  }\n}\n\nconst injectDefaultSchema = ({ className, fields, classLevelPermissions, indexes }: Schema) => {\n  const defaultSchema: Schema = {\n    className,\n    fields: {\n      ...defaultColumns._Default,\n      ...(defaultColumns[className] || {}),\n      ...fields,\n    },\n    classLevelPermissions,\n  };\n  if (indexes && Object.keys(indexes).length !== 0) {\n    defaultSchema.indexes = indexes;\n  }\n  return defaultSchema;\n};\n\nconst _HooksSchema = { className: '_Hooks', fields: defaultColumns._Hooks };\nconst _GlobalConfigSchema = {\n  className: '_GlobalConfig',\n  fields: defaultColumns._GlobalConfig,\n};\nconst _GraphQLConfigSchema = {\n  className: '_GraphQLConfig',\n  fields: defaultColumns._GraphQLConfig,\n};\nconst _PushStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_PushStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobScheduleSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobSchedule',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _AudienceSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Audience',\n    fields: defaultColumns._Audience,\n    classLevelPermissions: {},\n  })\n);\nconst _IdempotencySchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Idempotency',\n    fields: defaultColumns._Idempotency,\n    classLevelPermissions: {},\n  })\n);\nconst VolatileClassesSchemas = [\n  _HooksSchema,\n  _JobStatusSchema,\n  _JobScheduleSchema,\n  _PushStatusSchema,\n  _GlobalConfigSchema,\n  _GraphQLConfigSchema,\n  _AudienceSchema,\n  _IdempotencySchema,\n];\n\nconst dbTypeMatchesObjectType = (dbType: SchemaField | string, objectType: SchemaField) => {\n  if (dbType.type !== objectType.type) return false;\n  if (dbType.targetClass !== objectType.targetClass) return false;\n  if (dbType === objectType.type) return true;\n  if (dbType.type === objectType.type) return true;\n  return false;\n};\n\nconst typeToString = (type: SchemaField | string): string => {\n  if (typeof type === 'string') {\n    return type;\n  }\n  if (type.targetClass) {\n    return `${type.type}<${type.targetClass}>`;\n  }\n  return `${type.type}`;\n};\nconst ttl = {\n  date: Date.now(),\n  duration: undefined,\n};\n\n// Stores the entire schema of the app in a weird hybrid format somewhere between\n// the mongo format and the Parse format. Soon, this will all be Parse format.\nexport default class SchemaController {\n  _dbAdapter: StorageAdapter;\n  schemaData: { [string]: Schema };\n  reloadDataPromise: ?Promise<any>;\n  protectedFields: any;\n  userIdRegEx: RegExp;\n\n  constructor(databaseAdapter: StorageAdapter) {\n    this._dbAdapter = databaseAdapter;\n    const config = Config.get(Parse.applicationId);\n    this.schemaData = new SchemaData(SchemaCache.all(), this.protectedFields);\n    this.protectedFields = config.protectedFields;\n\n    const customIds = config.allowCustomObjectId;\n\n    const customIdRegEx = /^.{1,}$/u; // 1+ chars\n    const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;\n\n    this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;\n\n    this._dbAdapter.watch(() => {\n      this.reloadData({ clearCache: true });\n    });\n  }\n\n  async reloadDataIfNeeded() {\n    if (this._dbAdapter.enableSchemaHooks) {\n      return;\n    }\n    const { date, duration } = ttl || {};\n    if (!duration) {\n      return;\n    }\n    const now = Date.now();\n    if (now - date > duration) {\n      ttl.date = now;\n      await this.reloadData({ clearCache: true });\n    }\n  }\n\n  reloadData(options: LoadSchemaOptions = { clearCache: false }): Promise<any> {\n    if (this.reloadDataPromise && !options.clearCache) {\n      return this.reloadDataPromise;\n    }\n    this.reloadDataPromise = this.getAllClasses(options)\n      .then(\n        allSchemas => {\n          this.schemaData = new SchemaData(allSchemas, this.protectedFields);\n          delete this.reloadDataPromise;\n        },\n        err => {\n          this.schemaData = new SchemaData();\n          delete this.reloadDataPromise;\n          throw err;\n        }\n      )\n      .then(() => {});\n    return this.reloadDataPromise;\n  }\n\n  async getAllClasses(options: LoadSchemaOptions = { clearCache: false }): Promise<Array<Schema>> {\n    if (options.clearCache) {\n      return this.setAllClasses();\n    }\n    await this.reloadDataIfNeeded();\n    const cached = SchemaCache.all();\n    if (cached && cached.length) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses();\n  }\n\n  setAllClasses(): Promise<Array<Schema>> {\n    return this._dbAdapter\n      .getAllClasses()\n      .then(allSchemas => allSchemas.map(injectDefaultSchema))\n      .then(allSchemas => {\n        SchemaCache.put(allSchemas);\n        return allSchemas;\n      });\n  }\n\n  getOneSchema(\n    className: string,\n    allowVolatileClasses: boolean = false,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<Schema> {\n    if (options.clearCache) {\n      SchemaCache.clear();\n    }\n    if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {\n      const data = this.schemaData[className];\n      return Promise.resolve({\n        className,\n        fields: data.fields,\n        classLevelPermissions: data.classLevelPermissions,\n        indexes: data.indexes,\n      });\n    }\n    const cached = SchemaCache.get(className);\n    if (cached && !options.clearCache) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses().then(allSchemas => {\n      const oneSchema = allSchemas.find(schema => schema.className === className);\n      if (!oneSchema) {\n        return Promise.reject(undefined);\n      }\n      return oneSchema;\n    });\n  }\n\n  // Create a new class that includes the three default fields.\n  // ACL is an implicit column that does not get an entry in the\n  // _SCHEMAS database. Returns a promise that resolves with the\n  // created schema, in mongo format.\n  // on success, and rejects with an error on fail. Ensure you\n  // have authorization (master key, or client class creation\n  // enabled) before calling this function.\n  async addClassIfNotExists(\n    className: string,\n    fields: SchemaFields = {},\n    classLevelPermissions: any,\n    indexes: any = {}\n  ): Promise<void | Schema> {\n    var validationError = this.validateNewClass(className, fields, classLevelPermissions);\n    if (validationError) {\n      if (validationError instanceof Parse.Error) {\n        return Promise.reject(validationError);\n      } else if (validationError.code && validationError.error) {\n        return Promise.reject(new Parse.Error(validationError.code, validationError.error));\n      }\n      return Promise.reject(validationError);\n    }\n    try {\n      const adapterSchema = await this._dbAdapter.createClass(\n        className,\n        convertSchemaToAdapterSchema({\n          fields,\n          classLevelPermissions,\n          indexes,\n          className,\n        })\n      );\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n      const parseSchema = convertAdapterSchemaToParseSchema(adapterSchema);\n      return parseSchema;\n    } catch (error) {\n      if (error && error.code === Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n      } else {\n        throw error;\n      }\n    }\n  }\n\n  updateClass(\n    className: string,\n    submittedFields: SchemaFields,\n    classLevelPermissions: any,\n    indexes: any,\n    database: DatabaseController\n  ) {\n    return this.getOneSchema(className)\n      .then(schema => {\n        const existingFields = schema.fields;\n        Object.keys(submittedFields).forEach(name => {\n          const field = submittedFields[name];\n          if (\n            existingFields[name] &&\n            existingFields[name].type !== field.type &&\n            field.__op !== 'Delete'\n          ) {\n            throw new Parse.Error(255, `Field ${name} exists, cannot update.`);\n          }\n          if (!existingFields[name] && field.__op === 'Delete') {\n            throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);\n          }\n        });\n\n        delete existingFields._rperm;\n        delete existingFields._wperm;\n        const newSchema = buildMergedSchemaObject(existingFields, submittedFields);\n        const defaultFields = defaultColumns[className] || defaultColumns._Default;\n        const fullNewSchema = Object.assign({}, newSchema, defaultFields);\n        const validationError = this.validateSchemaData(\n          className,\n          newSchema,\n          classLevelPermissions,\n          Object.keys(existingFields)\n        );\n        if (validationError) {\n          throw new Parse.Error(validationError.code, validationError.error);\n        }\n\n        // Finally we have checked to make sure the request is valid and we can start deleting fields.\n        // Do all deletions first, then a single save to _SCHEMA collection to handle all additions.\n        const deletedFields: string[] = [];\n        const insertedFields = [];\n        Object.keys(submittedFields).forEach(fieldName => {\n          if (submittedFields[fieldName].__op === 'Delete') {\n            deletedFields.push(fieldName);\n          } else {\n            insertedFields.push(fieldName);\n          }\n        });\n\n        let deletePromise = Promise.resolve();\n        if (deletedFields.length > 0) {\n          deletePromise = this.deleteFields(deletedFields, className, database);\n        }\n        let enforceFields = [];\n        return (\n          deletePromise // Delete Everything\n            .then(() => this.reloadData({ clearCache: true })) // Reload our Schema, so we have all the new values\n            .then(() => {\n              const promises = insertedFields.map(fieldName => {\n                const type = submittedFields[fieldName];\n                return this.enforceFieldExists(className, fieldName, type);\n              });\n              return Promise.all(promises);\n            })\n            .then(results => {\n              enforceFields = results.filter(result => !!result);\n              return this.setPermissions(className, classLevelPermissions, newSchema);\n            })\n            .then(() =>\n              this._dbAdapter.setIndexesWithSchemaFormat(\n                className,\n                indexes,\n                schema.indexes,\n                fullNewSchema\n              )\n            )\n            .then(() => this.reloadData({ clearCache: true }))\n            //TODO: Move this logic into the database adapter\n            .then(() => {\n              this.ensureFields(enforceFields);\n              const schema = this.schemaData[className];\n              const reloadedSchema: Schema = {\n                className: className,\n                fields: schema.fields,\n                classLevelPermissions: schema.classLevelPermissions,\n              };\n              if (schema.indexes && Object.keys(schema.indexes).length !== 0) {\n                reloadedSchema.indexes = schema.indexes;\n              }\n              return reloadedSchema;\n            })\n        );\n      })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      });\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object or fails with a reason.\n  enforceClassExists(className: string): Promise<SchemaController> {\n    if (this.schemaData[className]) {\n      return Promise.resolve(this);\n    }\n    // We don't have this class. Update the schema\n    return (\n      // The schema update succeeded. Reload the schema\n      this.addClassIfNotExists(className)\n        .catch(() => {\n          // The schema update failed. This can be okay - it might\n          // have failed because there's a race condition and a different\n          // client is making the exact same schema update that we want.\n          // So just reload the schema.\n          return this.reloadData({ clearCache: true });\n        })\n        .then(() => {\n          // Ensure that the schema now validates\n          if (this.schemaData[className]) {\n            return this;\n          } else {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);\n          }\n        })\n        .catch(() => {\n          // The schema still doesn't validate. Give up\n          throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');\n        })\n    );\n  }\n\n  validateNewClass(className: string, fields: SchemaFields = {}, classLevelPermissions: any): any {\n    if (this.schemaData[className]) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n    }\n    if (!classNameIsValid(className)) {\n      return {\n        code: Parse.Error.INVALID_CLASS_NAME,\n        error: invalidClassNameMessage(className),\n      };\n    }\n    return this.validateSchemaData(className, fields, classLevelPermissions, []);\n  }\n\n  validateSchemaData(\n    className: string,\n    fields: SchemaFields,\n    classLevelPermissions: ClassLevelPermissions,\n    existingFieldNames: Array<string>\n  ) {\n    for (const fieldName in fields) {\n      if (existingFieldNames.indexOf(fieldName) < 0) {\n        if (!fieldNameIsValid(fieldName, className)) {\n          return {\n            code: Parse.Error.INVALID_KEY_NAME,\n            error: 'invalid field name: ' + fieldName,\n          };\n        }\n        if (!fieldNameIsValidForClass(fieldName, className)) {\n          return {\n            code: 136,\n            error: 'field ' + fieldName + ' cannot be added',\n          };\n        }\n        const fieldType = fields[fieldName];\n        const error = fieldTypeIsInvalid(fieldType);\n        if (error) return { code: error.code, error: error.message };\n        if (fieldType.defaultValue !== undefined) {\n          let defaultValueType = getType(fieldType.defaultValue);\n          if (typeof defaultValueType === 'string') {\n            defaultValueType = { type: defaultValueType };\n          } else if (typeof defaultValueType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n          if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n                fieldType\n              )} but got ${typeToString(defaultValueType)}`,\n            };\n          }\n        } else if (fieldType.required) {\n          if (typeof fieldType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'required' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n        }\n      }\n    }\n\n    for (const fieldName in defaultColumns[className]) {\n      fields[fieldName] = defaultColumns[className][fieldName];\n    }\n\n    const geoPoints = Object.keys(fields).filter(\n      key => fields[key] && fields[key].type === 'GeoPoint'\n    );\n    if (geoPoints.length > 1) {\n      return {\n        code: Parse.Error.INCORRECT_TYPE,\n        error:\n          'currently, only one GeoPoint field may exist in an object. Adding ' +\n          geoPoints[1] +\n          ' when ' +\n          geoPoints[0] +\n          ' already exists.',\n      };\n    }\n    validateCLP(classLevelPermissions, fields, this.userIdRegEx);\n  }\n\n  // Sets the Class-level permissions for a given className, which must exist.\n  async setPermissions(className: string, perms: any, newSchema: SchemaFields) {\n    if (typeof perms === 'undefined') {\n      return Promise.resolve();\n    }\n    validateCLP(perms, newSchema, this.userIdRegEx);\n    await this._dbAdapter.setClassLevelPermissions(className, perms);\n    const cached = SchemaCache.get(className);\n    if (cached) {\n      cached.classLevelPermissions = perms;\n    }\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object if the provided className-fieldName-type tuple is valid.\n  // The className must already be validated.\n  // If 'freeze' is true, refuse to update the schema for this field.\n  enforceFieldExists(\n    className: string,\n    fieldName: string,\n    type: string | SchemaField,\n    isValidation?: boolean,\n    maintenance?: boolean\n  ) {\n    if (fieldName.indexOf('.') > 0) {\n      // \"<array>.<index>\" for Nested Arrays\n      // \"<embedded document>.<field>\" for Nested Objects\n      // JSON Arrays are treated as Nested Objects\n      const [x, y] = fieldName.split('.');\n      fieldName = x;\n      const isArrayIndex = Array.from(y).every(c => c >= '0' && c <= '9');\n      if (isArrayIndex && !['sentPerUTCOffset', 'failedPerUTCOffset'].includes(fieldName)) {\n        type = 'Array';\n      } else {\n        type = 'Object';\n      }\n    }\n    let fieldNameToValidate = `${fieldName}`;\n    if (maintenance && fieldNameToValidate.charAt(0) === '_') {\n      fieldNameToValidate = fieldNameToValidate.substring(1);\n    }\n    if (!fieldNameIsValid(fieldNameToValidate, className)) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);\n    }\n\n    // If someone tries to create a new field with null/undefined as the value, return;\n    if (!type) {\n      return undefined;\n    }\n\n    const expectedType = this.getExpectedType(className, fieldName);\n    if (typeof type === 'string') {\n      type = ({ type }: SchemaField);\n    }\n\n    if (type.defaultValue !== undefined) {\n      let defaultValueType = getType(type.defaultValue);\n      if (typeof defaultValueType === 'string') {\n        defaultValueType = { type: defaultValueType };\n      }\n      if (!dbTypeMatchesObjectType(type, defaultValueType)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n            type\n          )} but got ${typeToString(defaultValueType)}`\n        );\n      }\n    }\n\n    if (expectedType) {\n      if (!dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName}; expected ${typeToString(\n            expectedType\n          )} but got ${typeToString(type)}`\n        );\n      }\n      // If type options do not change\n      // we can safely return\n      if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {\n        return undefined;\n      }\n      // Field options are may be changed\n      // ensure to have an update to date schema field\n      return this._dbAdapter.updateFieldOptions(className, fieldName, type);\n    }\n\n    return this._dbAdapter\n      .addFieldIfNotExists(className, fieldName, type)\n      .catch(error => {\n        if (error.code == Parse.Error.INCORRECT_TYPE) {\n          // Make sure that we throw errors when it is appropriate to do so.\n          throw error;\n        }\n        // The update failed. This can be okay - it might have been a race\n        // condition where another client updated the schema in the same\n        // way that we wanted to. So, just reload the schema\n        return Promise.resolve();\n      })\n      .then(() => {\n        return {\n          className,\n          fieldName,\n          type,\n        };\n      });\n  }\n\n  ensureFields(fields: any) {\n    for (let i = 0; i < fields.length; i += 1) {\n      const { className, fieldName } = fields[i];\n      let { type } = fields[i];\n      const expectedType = this.getExpectedType(className, fieldName);\n      if (typeof type === 'string') {\n        type = { type: type };\n      }\n      if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);\n      }\n    }\n  }\n\n  // maintain compatibility\n  deleteField(fieldName: string, className: string, database: DatabaseController) {\n    return this.deleteFields([fieldName], className, database);\n  }\n\n  // Delete fields, and remove that data from all objects. This is intended\n  // to remove unused fields, if other writers are writing objects that include\n  // this field, the field may reappear. Returns a Promise that resolves with\n  // no object on success, or rejects with { code, error } on failure.\n  // Passing the database and prefix is necessary in order to drop relation collections\n  // and remove fields from objects. Ideally the database would belong to\n  // a database adapter and this function would close over it or access it via member.\n  deleteFields(fieldNames: Array<string>, className: string, database: DatabaseController) {\n    if (!classNameIsValid(className)) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));\n    }\n\n    fieldNames.forEach(fieldName => {\n      if (!fieldNameIsValid(fieldName, className)) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);\n      }\n      //Don't allow deleting the default fields.\n      if (!fieldNameIsValidForClass(fieldName, className)) {\n        throw new Parse.Error(136, `field ${fieldName} cannot be changed`);\n      }\n    });\n\n    return this.getOneSchema(className, false, { clearCache: true })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      })\n      .then(schema => {\n        fieldNames.forEach(fieldName => {\n          if (!schema.fields[fieldName]) {\n            throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);\n          }\n        });\n\n        const schemaFields = { ...schema.fields };\n        return database.adapter.deleteFields(className, schema, fieldNames).then(() => {\n          return Promise.all(\n            fieldNames.map(fieldName => {\n              const field = schemaFields[fieldName];\n              if (field && field.type === 'Relation') {\n                //For relations, drop the _Join table\n                return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);\n              }\n              return Promise.resolve();\n            })\n          );\n        });\n      })\n      .then(() => {\n        SchemaCache.clear();\n      });\n  }\n\n  // Validates an object provided in REST format.\n  // Returns a promise that resolves to the new schema if this object is\n  // valid.\n  async validateObject(className: string, object: any, query: any, maintenance: boolean) {\n    let geocount = 0;\n    const schema = await this.enforceClassExists(className);\n    const promises = [];\n\n    for (const fieldName in object) {\n      if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {\n        geocount++;\n      }\n      if (geocount > 1) {\n        return Promise.reject(\n          new Parse.Error(\n            Parse.Error.INCORRECT_TYPE,\n            'there can only be one geopoint field in a class'\n          )\n        );\n      }\n    }\n    for (const fieldName in object) {\n      if (object[fieldName] === undefined) {\n        continue;\n      }\n      const expected = getType(object[fieldName]);\n      if (!expected) {\n        continue;\n      }\n      if (fieldName === 'ACL') {\n        // Every object has ACL implicitly.\n        continue;\n      }\n      promises.push(schema.enforceFieldExists(className, fieldName, expected, true, maintenance));\n    }\n    const results = await Promise.all(promises);\n    const enforceFields = results.filter(result => !!result);\n\n    if (enforceFields.length !== 0) {\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n    }\n    this.ensureFields(enforceFields);\n\n    const promise = Promise.resolve(schema);\n    return thenValidateRequiredColumns(promise, className, object, query);\n  }\n\n  // Validates that all the properties are set for the object\n  validateRequiredColumns(className: string, object: any, query: any) {\n    const columns = requiredColumns.write[className];\n    if (!columns || columns.length == 0) {\n      return Promise.resolve(this);\n    }\n\n    const missingColumns = columns.filter(function (column) {\n      if (query && query.objectId) {\n        if (object[column] && typeof object[column] === 'object') {\n          // Trying to delete a required column\n          return object[column].__op == 'Delete';\n        }\n        // Not trying to do anything there\n        return false;\n      }\n      return !object[column];\n    });\n\n    if (missingColumns.length > 0) {\n      throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');\n    }\n    return Promise.resolve(this);\n  }\n\n  testPermissionsForClassName(className: string, aclGroup: string[], operation: string) {\n    return SchemaController.testPermissions(\n      this.getClassLevelPermissions(className),\n      aclGroup,\n      operation\n    );\n  }\n\n  // Tests that the class level permission let pass the operation for a given aclGroup\n  static testPermissions(classPermissions: ?any, aclGroup: string[], operation: string): boolean {\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    if (perms['*']) {\n      return true;\n    }\n    // Check permissions against the aclGroup provided (array of userId/roles)\n    if (\n      aclGroup.some(acl => {\n        return perms[acl] === true;\n      })\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  static validatePermission(\n    classPermissions: ?any,\n    className: string,\n    aclGroup: string[],\n    operation: string,\n    action?: string\n  ) {\n    if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {\n      return Promise.resolve();\n    }\n\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    // If only for authenticated users\n    // make sure we have an aclGroup\n    if (perms['requiresAuthentication']) {\n      // If aclGroup has * (public)\n      if (!aclGroup || aclGroup.length == 0) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.'\n        );\n      } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.'\n        );\n      }\n      // requiresAuthentication passed, just move forward\n      // probably would be wise at some point to rename to 'authenticatedUser'\n      return Promise.resolve();\n    }\n\n    // No matching CLP, let's check the Pointer permissions\n    // And handle those later\n    const permissionField =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    // Reject create when write lockdown\n    if (permissionField == 'writeUserFields' && operation == 'create') {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `Permission denied for action ${operation} on class ${className}.`\n      );\n    }\n\n    // Process the readUserFields later\n    if (\n      Array.isArray(classPermissions[permissionField]) &&\n      classPermissions[permissionField].length > 0\n    ) {\n      return Promise.resolve();\n    }\n\n    const pointerFields = classPermissions[operation].pointerFields;\n    if (Array.isArray(pointerFields) && pointerFields.length > 0) {\n      // any op except 'addField as part of create' is ok.\n      if (operation !== 'addField' || action === 'update') {\n        // We can allow adding field on update flow only.\n        return Promise.resolve();\n      }\n    }\n\n    throw new Parse.Error(\n      Parse.Error.OPERATION_FORBIDDEN,\n      `Permission denied for action ${operation} on class ${className}.`\n    );\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  validatePermission(className: string, aclGroup: string[], operation: string, action?: string) {\n    return SchemaController.validatePermission(\n      this.getClassLevelPermissions(className),\n      className,\n      aclGroup,\n      operation,\n      action\n    );\n  }\n\n  getClassLevelPermissions(className: string): any {\n    return this.schemaData[className] && this.schemaData[className].classLevelPermissions;\n  }\n\n  // Returns the expected type for a className+key combination\n  // or undefined if the schema is not set\n  getExpectedType(className: string, fieldName: string): ?(SchemaField | string) {\n    if (this.schemaData[className]) {\n      const expectedType = this.schemaData[className].fields[fieldName];\n      return expectedType === 'map' ? 'Object' : expectedType;\n    }\n    return undefined;\n  }\n\n  // Checks if a given class is in the schema.\n  hasClass(className: string) {\n    if (this.schemaData[className]) {\n      return Promise.resolve(true);\n    }\n    return this.reloadData().then(() => !!this.schemaData[className]);\n  }\n}\n\n// Returns a promise for a new Schema.\nconst load = (dbAdapter: StorageAdapter, options: any): Promise<SchemaController> => {\n  const schema = new SchemaController(dbAdapter);\n  ttl.duration = dbAdapter.schemaCacheTtl;\n  return schema.reloadData(options).then(() => schema);\n};\n\n// Builds a new schema (in schema API response format) out of an\n// existing mongo schema + a schemas API put request. This response\n// does not include the default fields, as it is intended to be passed\n// to mongoSchemaFromFieldsAndClassName. No validation is done here, it\n// is done in mongoSchemaFromFieldsAndClassName.\nfunction buildMergedSchemaObject(existingFields: SchemaFields, putRequest: any): SchemaFields {\n  const newSchema = {};\n  // @flow-disable-next\n  const sysSchemaField =\n    Object.keys(defaultColumns).indexOf(existingFields._id) === -1\n      ? []\n      : Object.keys(defaultColumns[existingFields._id]);\n  for (const oldField in existingFields) {\n    if (\n      oldField !== '_id' &&\n      oldField !== 'ACL' &&\n      oldField !== 'updatedAt' &&\n      oldField !== 'createdAt' &&\n      oldField !== 'objectId'\n    ) {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {\n        continue;\n      }\n      const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete';\n      if (!fieldIsDeleted) {\n        newSchema[oldField] = existingFields[oldField];\n      }\n    }\n  }\n  for (const newField in putRequest) {\n    if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {\n        continue;\n      }\n      newSchema[newField] = putRequest[newField];\n    }\n  }\n  return newSchema;\n}\n\n// Given a schema promise, construct another schema promise that\n// validates this field once the schema loads.\nfunction thenValidateRequiredColumns(schemaPromise, className, object, query) {\n  return schemaPromise.then(schema => {\n    return schema.validateRequiredColumns(className, object, query);\n  });\n}\n\n// Gets the type from a REST API formatted object, where 'type' is\n// extended past javascript types to include the rest of the Parse\n// type system.\n// The output should be a valid schema value.\n// TODO: ensure that this is compatible with the format used in Open DB\nfunction getType(obj: any): ?(SchemaField | string) {\n  const type = typeof obj;\n  switch (type) {\n    case 'boolean':\n      return 'Boolean';\n    case 'string':\n      return 'String';\n    case 'number':\n      return 'Number';\n    case 'map':\n    case 'object':\n      if (!obj) {\n        return undefined;\n      }\n      return getObjectType(obj);\n    case 'function':\n    case 'symbol':\n    case 'undefined':\n    default:\n      throw 'bad obj: ' + obj;\n  }\n}\n\n// This gets the type for non-JSON types like pointers and files, but\n// also gets the appropriate type for $ operators.\n// Returns null if the type is unknown.\nfunction getObjectType(obj): ?(SchemaField | string) {\n  if (obj instanceof Array) {\n    return 'Array';\n  }\n  if (obj.__type) {\n    switch (obj.__type) {\n      case 'Pointer':\n        if (obj.className) {\n          return {\n            type: 'Pointer',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'Relation':\n        if (obj.className) {\n          return {\n            type: 'Relation',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'File':\n        if (obj.name) {\n          return 'File';\n        }\n        break;\n      case 'Date':\n        if (obj.iso) {\n          return 'Date';\n        }\n        break;\n      case 'GeoPoint':\n        if (obj.latitude != null && obj.longitude != null) {\n          return 'GeoPoint';\n        }\n        break;\n      case 'Bytes':\n        if (obj.base64) {\n          return 'Bytes';\n        }\n        break;\n      case 'Polygon':\n        if (obj.coordinates) {\n          return 'Polygon';\n        }\n        break;\n    }\n    throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);\n  }\n  if (obj['$ne']) {\n    return getObjectType(obj['$ne']);\n  }\n  if (obj.__op) {\n    switch (obj.__op) {\n      case 'Increment':\n        return 'Number';\n      case 'Delete':\n        return null;\n      case 'Add':\n      case 'AddUnique':\n      case 'Remove':\n        return 'Array';\n      case 'AddRelation':\n      case 'RemoveRelation':\n        return {\n          type: 'Relation',\n          targetClass: obj.objects[0].className,\n        };\n      case 'Batch':\n        return getObjectType(obj.ops[0]);\n      default:\n        throw 'unexpected op: ' + obj.__op;\n    }\n  }\n  return 'Object';\n}\n\nexport {\n  load,\n  classNameIsValid,\n  fieldNameIsValid,\n  invalidClassNameMessage,\n  buildMergedSchemaObject,\n  systemClasses,\n  defaultColumns,\n  convertSchemaToAdapterSchema,\n  VolatileClassesSchemas,\n  SchemaController,\n  requiredColumns,\n};\n"],"mappings":";;;;;;;;;;;;AAkBA,IAAAA,eAAA,GAAAC,OAAA;AACA,IAAAC,YAAA,GAAAC,sBAAA,CAAAF,OAAA;AACA,IAAAG,mBAAA,GAAAD,sBAAA,CAAAF,OAAA;AACA,IAAAI,OAAA,GAAAF,sBAAA,CAAAF,OAAA;AAEA,IAAAK,SAAA,GAAAH,sBAAA,CAAAF,OAAA;AAAgC,SAAAE,uBAAAI,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAAA,SAAAG,QAAAH,CAAA,EAAAI,CAAA,QAAAC,CAAA,GAAAC,MAAA,CAAAC,IAAA,CAAAP,CAAA,OAAAM,MAAA,CAAAE,qBAAA,QAAAC,CAAA,GAAAH,MAAA,CAAAE,qBAAA,CAAAR,CAAA,GAAAI,CAAA,KAAAK,CAAA,GAAAA,CAAA,CAAAC,MAAA,WAAAN,CAAA,WAAAE,MAAA,CAAAK,wBAAA,CAAAX,CAAA,EAAAI,CAAA,EAAAQ,UAAA,OAAAP,CAAA,CAAAQ,IAAA,CAAAC,KAAA,CAAAT,CAAA,EAAAI,CAAA,YAAAJ,CAAA;AAAA,SAAAU,cAAAf,CAAA,aAAAI,CAAA,MAAAA,CAAA,GAAAY,SAAA,CAAAC,MAAA,EAAAb,CAAA,UAAAC,CAAA,WAAAW,SAAA,CAAAZ,CAAA,IAAAY,SAAA,CAAAZ,CAAA,QAAAA,CAAA,OAAAD,OAAA,CAAAG,MAAA,CAAAD,CAAA,OAAAa,OAAA,WAAAd,CAAA,IAAAe,eAAA,CAAAnB,CAAA,EAAAI,CAAA,EAAAC,CAAA,CAAAD,CAAA,SAAAE,MAAA,CAAAc,yBAAA,GAAAd,MAAA,CAAAe,gBAAA,CAAArB,CAAA,EAAAM,MAAA,CAAAc,yBAAA,CAAAf,CAAA,KAAAF,OAAA,CAAAG,MAAA,CAAAD,CAAA,GAAAa,OAAA,WAAAd,CAAA,IAAAE,MAAA,CAAAgB,cAAA,CAAAtB,CAAA,EAAAI,CAAA,EAAAE,MAAA,CAAAK,wBAAA,CAAAN,CAAA,EAAAD,CAAA,iBAAAJ,CAAA;AAAA,SAAAmB,gBAAAnB,CAAA,EAAAI,CAAA,EAAAC,CAAA,YAAAD,CAAA,GAAAmB,cAAA,CAAAnB,CAAA,MAAAJ,CAAA,GAAAM,MAAA,CAAAgB,cAAA,CAAAtB,CAAA,EAAAI,CAAA,IAAAoB,KAAA,EAAAnB,CAAA,EAAAO,UAAA,MAAAa,YAAA,MAAAC,QAAA,UAAA1B,CAAA,CAAAI,CAAA,IAAAC,CAAA,EAAAL,CAAA;AAAA,SAAAuB,eAAAlB,CAAA,QAAAsB,CAAA,GAAAC,YAAA,CAAAvB,CAAA,uCAAAsB,CAAA,GAAAA,CAAA,GAAAA,CAAA;AAAA,SAAAC,aAAAvB,CAAA,EAAAD,CAAA,2BAAAC,CAAA,KAAAA,CAAA,SAAAA,CAAA,MAAAL,CAAA,GAAAK,CAAA,CAAAwB,MAAA,CAAAC,WAAA,kBAAA9B,CAAA,QAAA2B,CAAA,GAAA3B,CAAA,CAAA+B,IAAA,CAAA1B,CAAA,EAAAD,CAAA,uCAAAuB,CAAA,SAAAA,CAAA,YAAAK,SAAA,yEAAA5B,CAAA,GAAA6B,MAAA,GAAAC,MAAA,EAAA7B,CAAA;AAAA,SAAA8B,0BAAA9B,CAAA,gBAAAA,CAAA,YAAA2B,SAAA,yBAAA3B,CAAA;AAAA,SAAA+B,SAAA,WAAAA,QAAA,GAAA9B,MAAA,CAAA+B,MAAA,GAAA/B,MAAA,CAAA+B,MAAA,CAAAC,IAAA,eAAAC,CAAA,aAAAvC,CAAA,MAAAA,CAAA,GAAAgB,SAAA,CAAAC,MAAA,EAAAjB,CAAA,UAAAK,CAAA,GAAAW,SAAA,CAAAhB,CAAA,YAAAI,CAAA,IAAAC,CAAA,OAAAmC,cAAA,CAAAT,IAAA,CAAA1B,CAAA,EAAAD,CAAA,MAAAmC,CAAA,CAAAnC,CAAA,IAAAC,CAAA,CAAAD,CAAA,aAAAmC,CAAA,KAAAH,QAAA,CAAAtB,KAAA,OAAAE,SAAA;AAtBhC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMyB,KAAK,GAAG/C,OAAO,CAAC,YAAY,CAAC,CAAC+C,KAAK;;AAKzC;;AAUA,MAAMC,cAA0C,GAAAC,OAAA,CAAAD,cAAA,GAAGpC,MAAM,CAACsC,MAAM,CAAC;EAC/D;EACAC,QAAQ,EAAE;IACRC,QAAQ,EAAE;MAAEC,IAAI,EAAE;IAAS,CAAC;IAC5BC,SAAS,EAAE;MAAED,IAAI,EAAE;IAAO,CAAC;IAC3BE,SAAS,EAAE;MAAEF,IAAI,EAAE;IAAO,CAAC;IAC3BG,GAAG,EAAE;MAAEH,IAAI,EAAE;IAAM;EACrB,CAAC;EACD;EACAI,KAAK,EAAE;IACLC,QAAQ,EAAE;MAAEL,IAAI,EAAE;IAAS,CAAC;IAC5BM,QAAQ,EAAE;MAAEN,IAAI,EAAE;IAAS,CAAC;IAC5BO,KAAK,EAAE;MAAEP,IAAI,EAAE;IAAS,CAAC;IACzBQ,aAAa,EAAE;MAAER,IAAI,EAAE;IAAU,CAAC;IAClCS,QAAQ,EAAE;MAAET,IAAI,EAAE;IAAS;EAC7B,CAAC;EACD;EACAU,aAAa,EAAE;IACbC,cAAc,EAAE;MAAEX,IAAI,EAAE;IAAS,CAAC;IAClCY,WAAW,EAAE;MAAEZ,IAAI,EAAE;IAAS,CAAC;IAC/Ba,QAAQ,EAAE;MAAEb,IAAI,EAAE;IAAQ,CAAC;IAC3Bc,UAAU,EAAE;MAAEd,IAAI,EAAE;IAAS,CAAC;IAC9Be,QAAQ,EAAE;MAAEf,IAAI,EAAE;IAAS,CAAC;IAC5BgB,WAAW,EAAE;MAAEhB,IAAI,EAAE;IAAS,CAAC;IAC/BiB,QAAQ,EAAE;MAAEjB,IAAI,EAAE;IAAS,CAAC;IAC5BkB,gBAAgB,EAAE;MAAElB,IAAI,EAAE;IAAS,CAAC;IACpCmB,KAAK,EAAE;MAAEnB,IAAI,EAAE;IAAS,CAAC;IACzBoB,UAAU,EAAE;MAAEpB,IAAI,EAAE;IAAS,CAAC;IAC9BqB,OAAO,EAAE;MAAErB,IAAI,EAAE;IAAS,CAAC;IAC3BsB,aAAa,EAAE;MAAEtB,IAAI,EAAE;IAAS,CAAC;IACjCuB,YAAY,EAAE;MAAEvB,IAAI,EAAE;IAAS;EACjC,CAAC;EACD;EACAwB,KAAK,EAAE;IACLC,IAAI,EAAE;MAAEzB,IAAI,EAAE;IAAS,CAAC;IACxB0B,KAAK,EAAE;MAAE1B,IAAI,EAAE,UAAU;MAAE2B,WAAW,EAAE;IAAQ,CAAC;IACjDC,KAAK,EAAE;MAAE5B,IAAI,EAAE,UAAU;MAAE2B,WAAW,EAAE;IAAQ;EAClD,CAAC;EACD;EACAE,QAAQ,EAAE;IACRC,IAAI,EAAE;MAAE9B,IAAI,EAAE,SAAS;MAAE2B,WAAW,EAAE;IAAQ,CAAC;IAC/ChB,cAAc,EAAE;MAAEX,IAAI,EAAE;IAAS,CAAC;IAClC+B,YAAY,EAAE;MAAE/B,IAAI,EAAE;IAAS,CAAC;IAChCgC,SAAS,EAAE;MAAEhC,IAAI,EAAE;IAAO,CAAC;IAC3BiC,WAAW,EAAE;MAAEjC,IAAI,EAAE;IAAS;EAChC,CAAC;EACDkC,QAAQ,EAAE;IACRC,iBAAiB,EAAE;MAAEnC,IAAI,EAAE;IAAS,CAAC;IACrCoC,QAAQ,EAAE;MAAEpC,IAAI,EAAE;IAAO,CAAC;IAC1BqC,YAAY,EAAE;MAAErC,IAAI,EAAE;IAAS,CAAC;IAChCsC,IAAI,EAAE;MAAEtC,IAAI,EAAE;IAAO,CAAC;IACtBuC,KAAK,EAAE;MAAEvC,IAAI,EAAE;IAAS,CAAC;IACzBwC,KAAK,EAAE;MAAExC,IAAI,EAAE;IAAS,CAAC;IACzByC,QAAQ,EAAE;MAAEzC,IAAI,EAAE;IAAS;EAC7B,CAAC;EACD0C,WAAW,EAAE;IACXC,QAAQ,EAAE;MAAE3C,IAAI,EAAE;IAAS,CAAC;IAC5B4C,MAAM,EAAE;MAAE5C,IAAI,EAAE;IAAS,CAAC;IAAE;IAC5B6C,KAAK,EAAE;MAAE7C,IAAI,EAAE;IAAS,CAAC;IAAE;IAC3B8C,OAAO,EAAE;MAAE9C,IAAI,EAAE;IAAS,CAAC;IAAE;IAC7BwC,KAAK,EAAE;MAAExC,IAAI,EAAE;IAAS,CAAC;IACzB+C,MAAM,EAAE;MAAE/C,IAAI,EAAE;IAAS,CAAC;IAC1BgD,mBAAmB,EAAE;MAAEhD,IAAI,EAAE;IAAS,CAAC;IACvCiD,MAAM,EAAE;MAAEjD,IAAI,EAAE;IAAS,CAAC;IAC1BkD,OAAO,EAAE;MAAElD,IAAI,EAAE;IAAS,CAAC;IAC3BmD,SAAS,EAAE;MAAEnD,IAAI,EAAE;IAAS,CAAC;IAC7BoD,QAAQ,EAAE;MAAEpD,IAAI,EAAE;IAAS,CAAC;IAC5BqD,YAAY,EAAE;MAAErD,IAAI,EAAE;IAAS,CAAC;IAChCsD,WAAW,EAAE;MAAEtD,IAAI,EAAE;IAAS,CAAC;IAC/BuD,aAAa,EAAE;MAAEvD,IAAI,EAAE;IAAS,CAAC;IACjCwD,gBAAgB,EAAE;MAAExD,IAAI,EAAE;IAAS,CAAC;IACpCyD,kBAAkB,EAAE;MAAEzD,IAAI,EAAE;IAAS,CAAC;IACtC0D,KAAK,EAAE;MAAE1D,IAAI,EAAE;IAAS,CAAC,CAAE;EAC7B,CAAC;EACD2D,UAAU,EAAE;IACVC,OAAO,EAAE;MAAE5D,IAAI,EAAE;IAAS,CAAC;IAC3B4C,MAAM,EAAE;MAAE5C,IAAI,EAAE;IAAS,CAAC;IAC1BiD,MAAM,EAAE;MAAEjD,IAAI,EAAE;IAAS,CAAC;IAC1B6D,OAAO,EAAE;MAAE7D,IAAI,EAAE;IAAS,CAAC;IAC3B8D,MAAM,EAAE;MAAE9D,IAAI,EAAE;IAAS,CAAC;IAAE;IAC5B+D,UAAU,EAAE;MAAE/D,IAAI,EAAE;IAAO;EAC7B,CAAC;EACDgE,YAAY,EAAE;IACZJ,OAAO,EAAE;MAAE5D,IAAI,EAAE;IAAS,CAAC;IAC3BiE,WAAW,EAAE;MAAEjE,IAAI,EAAE;IAAS,CAAC;IAC/B8D,MAAM,EAAE;MAAE9D,IAAI,EAAE;IAAS,CAAC;IAC1BkE,UAAU,EAAE;MAAElE,IAAI,EAAE;IAAS,CAAC;IAC9BmE,UAAU,EAAE;MAAEnE,IAAI,EAAE;IAAQ,CAAC;IAC7BoE,SAAS,EAAE;MAAEpE,IAAI,EAAE;IAAS,CAAC;IAC7BqE,OAAO,EAAE;MAAErE,IAAI,EAAE;IAAS,CAAC;IAC3BsE,aAAa,EAAE;MAAEtE,IAAI,EAAE;IAAS;EAClC,CAAC;EACDuE,MAAM,EAAE;IACNC,YAAY,EAAE;MAAExE,IAAI,EAAE;IAAS,CAAC;IAChCyE,SAAS,EAAE;MAAEzE,IAAI,EAAE;IAAS,CAAC;IAC7B0E,WAAW,EAAE;MAAE1E,IAAI,EAAE;IAAS,CAAC;IAC/B2E,GAAG,EAAE;MAAE3E,IAAI,EAAE;IAAS;EACxB,CAAC;EACD4E,aAAa,EAAE;IACb7E,QAAQ,EAAE;MAAEC,IAAI,EAAE;IAAS,CAAC;IAC5B8D,MAAM,EAAE;MAAE9D,IAAI,EAAE;IAAS,CAAC;IAC1B6E,aAAa,EAAE;MAAE7E,IAAI,EAAE;IAAS;EAClC,CAAC;EACD8E,cAAc,EAAE;IACd/E,QAAQ,EAAE;MAAEC,IAAI,EAAE;IAAS,CAAC;IAC5B+E,MAAM,EAAE;MAAE/E,IAAI,EAAE;IAAS;EAC3B,CAAC;EACDgF,SAAS,EAAE;IACTjF,QAAQ,EAAE;MAAEC,IAAI,EAAE;IAAS,CAAC;IAC5ByB,IAAI,EAAE;MAAEzB,IAAI,EAAE;IAAS,CAAC;IACxB6C,KAAK,EAAE;MAAE7C,IAAI,EAAE;IAAS,CAAC;IAAE;IAC3BiF,QAAQ,EAAE;MAAEjF,IAAI,EAAE;IAAO,CAAC;IAC1BkF,SAAS,EAAE;MAAElF,IAAI,EAAE;IAAS;EAC9B,CAAC;EACDmF,YAAY,EAAE;IACZC,KAAK,EAAE;MAAEpF,IAAI,EAAE;IAAS,CAAC;IACzBqF,MAAM,EAAE;MAAErF,IAAI,EAAE;IAAO;EACzB;AACF,CAAC,CAAC;;AAEF;AACA,MAAMsF,eAAe,GAAA1F,OAAA,CAAA0F,eAAA,GAAG/H,MAAM,CAACsC,MAAM,CAAC;EACpC0F,IAAI,EAAE;IACJnF,KAAK,EAAE,CAAC,UAAU;EACpB,CAAC;EACDoF,KAAK,EAAE;IACLtD,QAAQ,EAAE,CAAC,mBAAmB,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,UAAU,CAAC;IACrEV,KAAK,EAAE,CAAC,MAAM,EAAE,KAAK;EACvB;AACF,CAAC,CAAC;AAEF,MAAMiE,cAAc,GAAG,CAAC,QAAQ,CAAC;AAEjC,MAAMC,aAAa,GAAA9F,OAAA,CAAA8F,aAAA,GAAGnI,MAAM,CAACsC,MAAM,CAAC,CAClC,OAAO,EACP,eAAe,EACf,OAAO,EACP,UAAU,EACV,UAAU,EACV,aAAa,EACb,YAAY,EACZ,cAAc,EACd,WAAW,EACX,cAAc,CACf,CAAC;AAEF,MAAM8F,eAAe,GAAGpI,MAAM,CAACsC,MAAM,CAAC,CACpC,YAAY,EACZ,aAAa,EACb,QAAQ,EACR,eAAe,EACf,gBAAgB,EAChB,cAAc,EACd,WAAW,EACX,cAAc,CACf,CAAC;;AAEF;AACA,MAAM+F,SAAS,GAAG,UAAU;AAC5B;AACA,MAAMC,2BAA2B,GAAG,eAAe;AACnD;AACA,MAAMC,WAAW,GAAG,MAAM;AAE1B,MAAMC,kBAAkB,GAAG,iBAAiB;AAE5C,MAAMC,2BAA2B,GAAG,0BAA0B;AAE9D,MAAMC,eAAe,GAAG,iBAAiB;;AAEzC;AACA,MAAMC,oBAAoB,GAAG3I,MAAM,CAACsC,MAAM,CAAC,CACzCgG,2BAA2B,EAC3BC,WAAW,EACXC,kBAAkB,EAClBH,SAAS,CACV,CAAC;;AAEF;AACA,MAAMO,cAAc,GAAG5I,MAAM,CAACsC,MAAM,CAAC,CACnCoG,eAAe,EACfH,WAAW,EACXE,2BAA2B,EAC3BJ,SAAS,CACV,CAAC;AAEF,SAASQ,qBAAqBA,CAACC,GAAG,EAAEC,YAAY,EAAE;EAChD,IAAIC,WAAW,GAAG,KAAK;EACvB,KAAK,MAAMC,KAAK,IAAIL,cAAc,EAAE;IAClC,IAAIE,GAAG,CAACI,KAAK,CAACD,KAAK,CAAC,KAAK,IAAI,EAAE;MAC7BD,WAAW,GAAG,IAAI;MAClB;IACF;EACF;;EAEA;EACA,MAAMG,KAAK,GAAGH,WAAW,IAAIF,GAAG,CAACI,KAAK,CAACH,YAAY,CAAC,KAAK,IAAI;EAC7D,IAAI,CAACI,KAAK,EAAE;IACV,MAAM,IAAIhH,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIP,GAAG,kDACT,CAAC;EACH;AACF;AAEA,SAASQ,0BAA0BA,CAACR,GAAG,EAAEC,YAAY,EAAE;EACrD,IAAIC,WAAW,GAAG,KAAK;EACvB,KAAK,MAAMC,KAAK,IAAIN,oBAAoB,EAAE;IACxC,IAAIG,GAAG,CAACI,KAAK,CAACD,KAAK,CAAC,KAAK,IAAI,EAAE;MAC7BD,WAAW,GAAG,IAAI;MAClB;IACF;EACF;;EAEA;EACA,MAAMG,KAAK,GAAGH,WAAW,IAAIF,GAAG,CAACI,KAAK,CAACH,YAAY,CAAC,KAAK,IAAI;EAC7D,IAAI,CAACI,KAAK,EAAE;IACV,MAAM,IAAIhH,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIP,GAAG,kDACT,CAAC;EACH;AACF;AAEA,MAAMS,YAAY,GAAGvJ,MAAM,CAACsC,MAAM,CAAC,CACjC,MAAM,EACN,OAAO,EACP,KAAK,EACL,QAAQ,EACR,QAAQ,EACR,QAAQ,EACR,UAAU,EACV,gBAAgB,EAChB,iBAAiB,EACjB,iBAAiB,CAClB,CAAC;;AAEF;AACA,SAASkH,WAAWA,CAACC,KAA4B,EAAEC,MAAoB,EAAEX,YAAoB,EAAE;EAC7F,IAAI,CAACU,KAAK,EAAE;IACV;EACF;EACA,KAAK,MAAME,YAAY,IAAIF,KAAK,EAAE;IAChC,IAAIF,YAAY,CAACK,OAAO,CAACD,YAAY,CAAC,IAAI,CAAC,CAAC,EAAE;MAC5C,MAAM,IAAIxH,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,GAAGM,YAAY,uDACjB,CAAC;IACH;IAEA,MAAME,SAAS,GAAGJ,KAAK,CAACE,YAAY,CAAC;IACrC;;IAEA;IACAG,eAAe,CAACD,SAAS,EAAEF,YAAY,CAAC;IAExC,IAAIA,YAAY,KAAK,gBAAgB,IAAIA,YAAY,KAAK,iBAAiB,EAAE;MAC3E;MACA;MACA,KAAK,MAAMI,SAAS,IAAIF,SAAS,EAAE;QACjCG,yBAAyB,CAACD,SAAS,EAAEL,MAAM,EAAEC,YAAY,CAAC;MAC5D;MACA;MACA;MACA;IACF;;IAEA;IACA,IAAIA,YAAY,KAAK,iBAAiB,EAAE;MACtC,KAAK,MAAMM,MAAM,IAAIJ,SAAS,EAAE;QAC9B;QACAP,0BAA0B,CAACW,MAAM,EAAElB,YAAY,CAAC;QAEhD,MAAMmB,eAAe,GAAGL,SAAS,CAACI,MAAM,CAAC;QAEzC,IAAI,CAACE,KAAK,CAACC,OAAO,CAACF,eAAe,CAAC,EAAE;UACnC,MAAM,IAAI/H,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIa,eAAe,8CAA8CD,MAAM,wBACzE,CAAC;QACH;;QAEA;QACA,KAAK,MAAMI,KAAK,IAAIH,eAAe,EAAE;UACnC;UACA,IAAI9H,cAAc,CAACG,QAAQ,CAAC8H,KAAK,CAAC,EAAE;YAClC,MAAM,IAAIlI,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,kBAAkBgB,KAAK,wBACzB,CAAC;UACH;UACA;UACA,IAAI,CAACrK,MAAM,CAACsK,SAAS,CAACpI,cAAc,CAACT,IAAI,CAACiI,MAAM,EAAEW,KAAK,CAAC,EAAE;YACxD,MAAM,IAAIlI,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,UAAUgB,KAAK,wBAAwBJ,MAAM,iBAC/C,CAAC;UACH;QACF;MACF;MACA;MACA;IACF;;IAEA;IACA;IACA;IACA;IACA;IACA;IACA;IACA,KAAK,MAAMA,MAAM,IAAIJ,SAAS,EAAE;MAC9B;MACAhB,qBAAqB,CAACoB,MAAM,EAAElB,YAAY,CAAC;;MAE3C;MACA;MACA,IAAIkB,MAAM,KAAK,eAAe,EAAE;QAC9B,MAAMM,aAAa,GAAGV,SAAS,CAACI,MAAM,CAAC;QAEvC,IAAIE,KAAK,CAACC,OAAO,CAACG,aAAa,CAAC,EAAE;UAChC,KAAK,MAAMC,YAAY,IAAID,aAAa,EAAE;YACxCP,yBAAyB,CAACQ,YAAY,EAAEd,MAAM,EAAEG,SAAS,CAAC;UAC5D;QACF,CAAC,MAAM;UACL,MAAM,IAAI1H,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIkB,aAAa,8BAA8BZ,YAAY,IAAIM,MAAM,wBACvE,CAAC;QACH;QACA;QACA;MACF;;MAEA;MACA,MAAMQ,MAAM,GAAGZ,SAAS,CAACI,MAAM,CAAC;MAEhC,IAAIQ,MAAM,KAAK,IAAI,EAAE;QACnB,MAAM,IAAItI,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIoB,MAAM,sDAAsDd,YAAY,IAAIM,MAAM,IAAIQ,MAAM,EAClG,CAAC;MACH;IACF;EACF;AACF;AAEA,SAASX,eAAeA,CAACD,SAAc,EAAEF,YAAoB,EAAE;EAC7D,IAAIA,YAAY,KAAK,gBAAgB,IAAIA,YAAY,KAAK,iBAAiB,EAAE;IAC3E,IAAI,CAACQ,KAAK,CAACC,OAAO,CAACP,SAAS,CAAC,EAAE;MAC7B,MAAM,IAAI1H,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIQ,SAAS,sDAAsDF,YAAY,qBACjF,CAAC;IACH;EACF,CAAC,MAAM;IACL,IAAI,OAAOE,SAAS,KAAK,QAAQ,IAAIA,SAAS,KAAK,IAAI,EAAE;MACvD;MACA;IACF,CAAC,MAAM;MACL,MAAM,IAAI1H,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIQ,SAAS,sDAAsDF,YAAY,sBACjF,CAAC;IACH;EACF;AACF;AAEA,SAASK,yBAAyBA,CAACD,SAAiB,EAAEL,MAAc,EAAEG,SAAiB,EAAE;EACvF;EACA;EACA;EACA;EACA;EACA;EACA;EACA,IACE,EACEH,MAAM,CAACK,SAAS,CAAC,KACfL,MAAM,CAACK,SAAS,CAAC,CAACtH,IAAI,IAAI,SAAS,IAAIiH,MAAM,CAACK,SAAS,CAAC,CAAC3F,WAAW,IAAI,OAAO,IAC/EsF,MAAM,CAACK,SAAS,CAAC,CAACtH,IAAI,IAAI,OAAO,CAAC,CACrC,EACD;IACA,MAAM,IAAIN,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EACxB,IAAIU,SAAS,+DAA+DF,SAAS,EACvF,CAAC;EACH;AACF;AAEA,MAAMa,cAAc,GAAG,oCAAoC;AAC3D,MAAMC,kBAAkB,GAAG,yBAAyB;AACpD,SAASC,gBAAgBA,CAAC1D,SAAiB,EAAW;EACpD;EACA;IACE;IACAiB,aAAa,CAACyB,OAAO,CAAC1C,SAAS,CAAC,GAAG,CAAC,CAAC;IACrC;IACAwD,cAAc,CAACG,IAAI,CAAC3D,SAAS,CAAC;IAC9B;IACA4D,gBAAgB,CAAC5D,SAAS,EAAEA,SAAS;EAAC;AAE1C;;AAEA;AACA;AACA,SAAS4D,gBAAgBA,CAACf,SAAiB,EAAE7C,SAAiB,EAAW;EACvE,IAAIA,SAAS,IAAIA,SAAS,KAAK,QAAQ,EAAE;IACvC,IAAI6C,SAAS,KAAK,WAAW,EAAE;MAC7B,OAAO,KAAK;IACd;EACF;EACA,OAAOY,kBAAkB,CAACE,IAAI,CAACd,SAAS,CAAC,IAAI,CAAC7B,cAAc,CAAC6C,QAAQ,CAAChB,SAAS,CAAC;AAClF;;AAEA;AACA,SAASiB,wBAAwBA,CAACjB,SAAiB,EAAE7C,SAAiB,EAAW;EAC/E,IAAI,CAAC4D,gBAAgB,CAACf,SAAS,EAAE7C,SAAS,CAAC,EAAE;IAC3C,OAAO,KAAK;EACd;EACA,IAAI9E,cAAc,CAACG,QAAQ,CAACwH,SAAS,CAAC,EAAE;IACtC,OAAO,KAAK;EACd;EACA,IAAI3H,cAAc,CAAC8E,SAAS,CAAC,IAAI9E,cAAc,CAAC8E,SAAS,CAAC,CAAC6C,SAAS,CAAC,EAAE;IACrE,OAAO,KAAK;EACd;EACA,OAAO,IAAI;AACb;AAEA,SAASkB,uBAAuBA,CAAC/D,SAAiB,EAAU;EAC1D,OACE,qBAAqB,GACrBA,SAAS,GACT,mGAAmG;AAEvG;AAEA,MAAMgE,gBAAgB,GAAG,IAAI/I,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EAAE,cAAc,CAAC;AAClF,MAAM8B,8BAA8B,GAAG,CACrC,QAAQ,EACR,QAAQ,EACR,SAAS,EACT,MAAM,EACN,QAAQ,EACR,OAAO,EACP,UAAU,EACV,MAAM,EACN,OAAO,EACP,SAAS,CACV;AACD;AACA,MAAMC,kBAAkB,GAAGA,CAAC;EAAE3I,IAAI;EAAE2B;AAAY,CAAC,KAAK;EACpD,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,CAACwF,OAAO,CAACnH,IAAI,CAAC,IAAI,CAAC,EAAE;IAC9C,IAAI,CAAC2B,WAAW,EAAE;MAChB,OAAO,IAAIjC,KAAK,CAACiH,KAAK,CAAC,GAAG,EAAE,QAAQ3G,IAAI,qBAAqB,CAAC;IAChE,CAAC,MAAM,IAAI,OAAO2B,WAAW,KAAK,QAAQ,EAAE;MAC1C,OAAO8G,gBAAgB;IACzB,CAAC,MAAM,IAAI,CAACN,gBAAgB,CAACxG,WAAW,CAAC,EAAE;MACzC,OAAO,IAAIjC,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACiC,kBAAkB,EAAEJ,uBAAuB,CAAC7G,WAAW,CAAC,CAAC;IAC9F,CAAC,MAAM;MACL,OAAOkH,SAAS;IAClB;EACF;EACA,IAAI,OAAO7I,IAAI,KAAK,QAAQ,EAAE;IAC5B,OAAOyI,gBAAgB;EACzB;EACA,IAAIC,8BAA8B,CAACvB,OAAO,CAACnH,IAAI,CAAC,GAAG,CAAC,EAAE;IACpD,OAAO,IAAIN,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAAE,uBAAuB9I,IAAI,EAAE,CAAC;EACnF;EACA,OAAO6I,SAAS;AAClB,CAAC;AAED,MAAME,4BAA4B,GAAIC,MAAW,IAAK;EACpDA,MAAM,GAAGC,mBAAmB,CAACD,MAAM,CAAC;EACpC,OAAOA,MAAM,CAAC/B,MAAM,CAAC9G,GAAG;EACxB6I,MAAM,CAAC/B,MAAM,CAACiC,MAAM,GAAG;IAAElJ,IAAI,EAAE;EAAQ,CAAC;EACxCgJ,MAAM,CAAC/B,MAAM,CAACkC,MAAM,GAAG;IAAEnJ,IAAI,EAAE;EAAQ,CAAC;EAExC,IAAIgJ,MAAM,CAACvE,SAAS,KAAK,OAAO,EAAE;IAChC,OAAOuE,MAAM,CAAC/B,MAAM,CAAC3G,QAAQ;IAC7B0I,MAAM,CAAC/B,MAAM,CAACmC,gBAAgB,GAAG;MAAEpJ,IAAI,EAAE;IAAS,CAAC;EACrD;EAEA,OAAOgJ,MAAM;AACf,CAAC;AAACpJ,OAAA,CAAAmJ,4BAAA,GAAAA,4BAAA;AAEF,MAAMM,iCAAiC,GAAGC,IAAA,IAAmB;EAAA,IAAbN,MAAM,GAAA3J,QAAA,MAAAD,yBAAA,CAAAkK,IAAA,GAAAA,IAAA;EACpD,OAAON,MAAM,CAAC/B,MAAM,CAACiC,MAAM;EAC3B,OAAOF,MAAM,CAAC/B,MAAM,CAACkC,MAAM;EAE3BH,MAAM,CAAC/B,MAAM,CAAC9G,GAAG,GAAG;IAAEH,IAAI,EAAE;EAAM,CAAC;EAEnC,IAAIgJ,MAAM,CAACvE,SAAS,KAAK,OAAO,EAAE;IAChC,OAAOuE,MAAM,CAAC/B,MAAM,CAACxG,QAAQ,CAAC,CAAC;IAC/B,OAAOuI,MAAM,CAAC/B,MAAM,CAACmC,gBAAgB;IACrCJ,MAAM,CAAC/B,MAAM,CAAC3G,QAAQ,GAAG;MAAEN,IAAI,EAAE;IAAS,CAAC;EAC7C;EAEA,IAAIgJ,MAAM,CAACO,OAAO,IAAIhM,MAAM,CAACC,IAAI,CAACwL,MAAM,CAACO,OAAO,CAAC,CAACrL,MAAM,KAAK,CAAC,EAAE;IAC9D,OAAO8K,MAAM,CAACO,OAAO;EACvB;EAEA,OAAOP,MAAM;AACf,CAAC;AAED,MAAMQ,UAAU,CAAC;EAGfC,WAAWA,CAACC,UAAU,GAAG,EAAE,EAAEjC,eAAe,GAAG,CAAC,CAAC,EAAE;IACjD,IAAI,CAACkC,MAAM,GAAG,CAAC,CAAC;IAChB,IAAI,CAACC,iBAAiB,GAAGnC,eAAe;IACxCiC,UAAU,CAACvL,OAAO,CAAC6K,MAAM,IAAI;MAC3B,IAAIrD,eAAe,CAAC2C,QAAQ,CAACU,MAAM,CAACvE,SAAS,CAAC,EAAE;QAC9C;MACF;MACAlH,MAAM,CAACgB,cAAc,CAAC,IAAI,EAAEyK,MAAM,CAACvE,SAAS,EAAE;QAC5CoF,GAAG,EAAEA,CAAA,KAAM;UACT,IAAI,CAAC,IAAI,CAACF,MAAM,CAACX,MAAM,CAACvE,SAAS,CAAC,EAAE;YAClC,MAAMqF,IAAI,GAAG,CAAC,CAAC;YACfA,IAAI,CAAC7C,MAAM,GAAGgC,mBAAmB,CAACD,MAAM,CAAC,CAAC/B,MAAM;YAChD6C,IAAI,CAACC,qBAAqB,GAAG,IAAAC,iBAAQ,EAAChB,MAAM,CAACe,qBAAqB,CAAC;YACnED,IAAI,CAACP,OAAO,GAAGP,MAAM,CAACO,OAAO;YAE7B,MAAMU,oBAAoB,GAAG,IAAI,CAACL,iBAAiB,CAACZ,MAAM,CAACvE,SAAS,CAAC;YACrE,IAAIwF,oBAAoB,EAAE;cACxB,KAAK,MAAM5D,GAAG,IAAI4D,oBAAoB,EAAE;gBACtC,MAAMC,GAAG,GAAG,IAAIC,GAAG,CAAC,CAClB,IAAIL,IAAI,CAACC,qBAAqB,CAACtC,eAAe,CAACpB,GAAG,CAAC,IAAI,EAAE,CAAC,EAC1D,GAAG4D,oBAAoB,CAAC5D,GAAG,CAAC,CAC7B,CAAC;gBACFyD,IAAI,CAACC,qBAAqB,CAACtC,eAAe,CAACpB,GAAG,CAAC,GAAGqB,KAAK,CAAC0C,IAAI,CAACF,GAAG,CAAC;cACnE;YACF;YAEA,IAAI,CAACP,MAAM,CAACX,MAAM,CAACvE,SAAS,CAAC,GAAGqF,IAAI;UACtC;UACA,OAAO,IAAI,CAACH,MAAM,CAACX,MAAM,CAACvE,SAAS,CAAC;QACtC;MACF,CAAC,CAAC;IACJ,CAAC,CAAC;;IAEF;IACAkB,eAAe,CAACxH,OAAO,CAACsG,SAAS,IAAI;MACnClH,MAAM,CAACgB,cAAc,CAAC,IAAI,EAAEkG,SAAS,EAAE;QACrCoF,GAAG,EAAEA,CAAA,KAAM;UACT,IAAI,CAAC,IAAI,CAACF,MAAM,CAAClF,SAAS,CAAC,EAAE;YAC3B,MAAMuE,MAAM,GAAGC,mBAAmB,CAAC;cACjCxE,SAAS;cACTwC,MAAM,EAAE,CAAC,CAAC;cACV8C,qBAAqB,EAAE,CAAC;YAC1B,CAAC,CAAC;YACF,MAAMD,IAAI,GAAG,CAAC,CAAC;YACfA,IAAI,CAAC7C,MAAM,GAAG+B,MAAM,CAAC/B,MAAM;YAC3B6C,IAAI,CAACC,qBAAqB,GAAGf,MAAM,CAACe,qBAAqB;YACzDD,IAAI,CAACP,OAAO,GAAGP,MAAM,CAACO,OAAO;YAC7B,IAAI,CAACI,MAAM,CAAClF,SAAS,CAAC,GAAGqF,IAAI;UAC/B;UACA,OAAO,IAAI,CAACH,MAAM,CAAClF,SAAS,CAAC;QAC/B;MACF,CAAC,CAAC;IACJ,CAAC,CAAC;EACJ;AACF;AAEA,MAAMwE,mBAAmB,GAAGA,CAAC;EAAExE,SAAS;EAAEwC,MAAM;EAAE8C,qBAAqB;EAAER;AAAgB,CAAC,KAAK;EAC7F,MAAMc,aAAqB,GAAG;IAC5B5F,SAAS;IACTwC,MAAM,EAAAjJ,aAAA,CAAAA,aAAA,CAAAA,aAAA,KACD2B,cAAc,CAACG,QAAQ,GACtBH,cAAc,CAAC8E,SAAS,CAAC,IAAI,CAAC,CAAC,GAChCwC,MAAM,CACV;IACD8C;EACF,CAAC;EACD,IAAIR,OAAO,IAAIhM,MAAM,CAACC,IAAI,CAAC+L,OAAO,CAAC,CAACrL,MAAM,KAAK,CAAC,EAAE;IAChDmM,aAAa,CAACd,OAAO,GAAGA,OAAO;EACjC;EACA,OAAOc,aAAa;AACtB,CAAC;AAED,MAAMC,YAAY,GAAG;EAAE7F,SAAS,EAAE,QAAQ;EAAEwC,MAAM,EAAEtH,cAAc,CAAC4E;AAAO,CAAC;AAC3E,MAAMgG,mBAAmB,GAAG;EAC1B9F,SAAS,EAAE,eAAe;EAC1BwC,MAAM,EAAEtH,cAAc,CAACiF;AACzB,CAAC;AACD,MAAM4F,oBAAoB,GAAG;EAC3B/F,SAAS,EAAE,gBAAgB;EAC3BwC,MAAM,EAAEtH,cAAc,CAACmF;AACzB,CAAC;AACD,MAAM2F,iBAAiB,GAAG1B,4BAA4B,CACpDE,mBAAmB,CAAC;EAClBxE,SAAS,EAAE,aAAa;EACxBwC,MAAM,EAAE,CAAC,CAAC;EACV8C,qBAAqB,EAAE,CAAC;AAC1B,CAAC,CACH,CAAC;AACD,MAAMW,gBAAgB,GAAG3B,4BAA4B,CACnDE,mBAAmB,CAAC;EAClBxE,SAAS,EAAE,YAAY;EACvBwC,MAAM,EAAE,CAAC,CAAC;EACV8C,qBAAqB,EAAE,CAAC;AAC1B,CAAC,CACH,CAAC;AACD,MAAMY,kBAAkB,GAAG5B,4BAA4B,CACrDE,mBAAmB,CAAC;EAClBxE,SAAS,EAAE,cAAc;EACzBwC,MAAM,EAAE,CAAC,CAAC;EACV8C,qBAAqB,EAAE,CAAC;AAC1B,CAAC,CACH,CAAC;AACD,MAAMa,eAAe,GAAG7B,4BAA4B,CAClDE,mBAAmB,CAAC;EAClBxE,SAAS,EAAE,WAAW;EACtBwC,MAAM,EAAEtH,cAAc,CAACqF,SAAS;EAChC+E,qBAAqB,EAAE,CAAC;AAC1B,CAAC,CACH,CAAC;AACD,MAAMc,kBAAkB,GAAG9B,4BAA4B,CACrDE,mBAAmB,CAAC;EAClBxE,SAAS,EAAE,cAAc;EACzBwC,MAAM,EAAEtH,cAAc,CAACwF,YAAY;EACnC4E,qBAAqB,EAAE,CAAC;AAC1B,CAAC,CACH,CAAC;AACD,MAAMe,sBAAsB,GAAAlL,OAAA,CAAAkL,sBAAA,GAAG,CAC7BR,YAAY,EACZI,gBAAgB,EAChBC,kBAAkB,EAClBF,iBAAiB,EACjBF,mBAAmB,EACnBC,oBAAoB,EACpBI,eAAe,EACfC,kBAAkB,CACnB;AAED,MAAME,uBAAuB,GAAGA,CAACC,MAA4B,EAAEC,UAAuB,KAAK;EACzF,IAAID,MAAM,CAAChL,IAAI,KAAKiL,UAAU,CAACjL,IAAI,EAAE,OAAO,KAAK;EACjD,IAAIgL,MAAM,CAACrJ,WAAW,KAAKsJ,UAAU,CAACtJ,WAAW,EAAE,OAAO,KAAK;EAC/D,IAAIqJ,MAAM,KAAKC,UAAU,CAACjL,IAAI,EAAE,OAAO,IAAI;EAC3C,IAAIgL,MAAM,CAAChL,IAAI,KAAKiL,UAAU,CAACjL,IAAI,EAAE,OAAO,IAAI;EAChD,OAAO,KAAK;AACd,CAAC;AAED,MAAMkL,YAAY,GAAIlL,IAA0B,IAAa;EAC3D,IAAI,OAAOA,IAAI,KAAK,QAAQ,EAAE;IAC5B,OAAOA,IAAI;EACb;EACA,IAAIA,IAAI,CAAC2B,WAAW,EAAE;IACpB,OAAO,GAAG3B,IAAI,CAACA,IAAI,IAAIA,IAAI,CAAC2B,WAAW,GAAG;EAC5C;EACA,OAAO,GAAG3B,IAAI,CAACA,IAAI,EAAE;AACvB,CAAC;AACD,MAAMmL,GAAG,GAAG;EACVC,IAAI,EAAEC,IAAI,CAACC,GAAG,CAAC,CAAC;EAChBC,QAAQ,EAAE1C;AACZ,CAAC;;AAED;AACA;AACe,MAAM2C,gBAAgB,CAAC;EAOpC/B,WAAWA,CAACgC,eAA+B,EAAE;IAC3C,IAAI,CAACC,UAAU,GAAGD,eAAe;IACjC,MAAM1G,MAAM,GAAG4G,eAAM,CAAC9B,GAAG,CAACnK,KAAK,CAACkM,aAAa,CAAC;IAC9C,IAAI,CAACC,UAAU,GAAG,IAAIrC,UAAU,CAACsC,oBAAW,CAACC,GAAG,CAAC,CAAC,EAAE,IAAI,CAACtE,eAAe,CAAC;IACzE,IAAI,CAACA,eAAe,GAAG1C,MAAM,CAAC0C,eAAe;IAE7C,MAAMuE,SAAS,GAAGjH,MAAM,CAACkH,mBAAmB;IAE5C,MAAMC,aAAa,GAAG,UAAU,CAAC,CAAC;IAClC,MAAMC,WAAW,GAAG,mBAAmB;IAEvC,IAAI,CAACC,WAAW,GAAGJ,SAAS,GAAGE,aAAa,GAAGC,WAAW;IAE1D,IAAI,CAACT,UAAU,CAACW,KAAK,CAAC,MAAM;MAC1B,IAAI,CAACC,UAAU,CAAC;QAAEC,UAAU,EAAE;MAAK,CAAC,CAAC;IACvC,CAAC,CAAC;EACJ;EAEA,MAAMC,kBAAkBA,CAAA,EAAG;IACzB,IAAI,IAAI,CAACd,UAAU,CAACe,iBAAiB,EAAE;MACrC;IACF;IACA,MAAM;MAAErB,IAAI;MAAEG;IAAS,CAAC,GAAGJ,GAAG,IAAI,CAAC,CAAC;IACpC,IAAI,CAACI,QAAQ,EAAE;MACb;IACF;IACA,MAAMD,GAAG,GAAGD,IAAI,CAACC,GAAG,CAAC,CAAC;IACtB,IAAIA,GAAG,GAAGF,IAAI,GAAGG,QAAQ,EAAE;MACzBJ,GAAG,CAACC,IAAI,GAAGE,GAAG;MACd,MAAM,IAAI,CAACgB,UAAU,CAAC;QAAEC,UAAU,EAAE;MAAK,CAAC,CAAC;IAC7C;EACF;EAEAD,UAAUA,CAACI,OAA0B,GAAG;IAAEH,UAAU,EAAE;EAAM,CAAC,EAAgB;IAC3E,IAAI,IAAI,CAACI,iBAAiB,IAAI,CAACD,OAAO,CAACH,UAAU,EAAE;MACjD,OAAO,IAAI,CAACI,iBAAiB;IAC/B;IACA,IAAI,CAACA,iBAAiB,GAAG,IAAI,CAACC,aAAa,CAACF,OAAO,CAAC,CACjDG,IAAI,CACHnD,UAAU,IAAI;MACZ,IAAI,CAACmC,UAAU,GAAG,IAAIrC,UAAU,CAACE,UAAU,EAAE,IAAI,CAACjC,eAAe,CAAC;MAClE,OAAO,IAAI,CAACkF,iBAAiB;IAC/B,CAAC,EACDG,GAAG,IAAI;MACL,IAAI,CAACjB,UAAU,GAAG,IAAIrC,UAAU,CAAC,CAAC;MAClC,OAAO,IAAI,CAACmD,iBAAiB;MAC7B,MAAMG,GAAG;IACX,CACF,CAAC,CACAD,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;IACjB,OAAO,IAAI,CAACF,iBAAiB;EAC/B;EAEA,MAAMC,aAAaA,CAACF,OAA0B,GAAG;IAAEH,UAAU,EAAE;EAAM,CAAC,EAA0B;IAC9F,IAAIG,OAAO,CAACH,UAAU,EAAE;MACtB,OAAO,IAAI,CAACQ,aAAa,CAAC,CAAC;IAC7B;IACA,MAAM,IAAI,CAACP,kBAAkB,CAAC,CAAC;IAC/B,MAAMQ,MAAM,GAAGlB,oBAAW,CAACC,GAAG,CAAC,CAAC;IAChC,IAAIiB,MAAM,IAAIA,MAAM,CAAC9O,MAAM,EAAE;MAC3B,OAAO+O,OAAO,CAACC,OAAO,CAACF,MAAM,CAAC;IAChC;IACA,OAAO,IAAI,CAACD,aAAa,CAAC,CAAC;EAC7B;EAEAA,aAAaA,CAAA,EAA2B;IACtC,OAAO,IAAI,CAACrB,UAAU,CACnBkB,aAAa,CAAC,CAAC,CACfC,IAAI,CAACnD,UAAU,IAAIA,UAAU,CAACyD,GAAG,CAAClE,mBAAmB,CAAC,CAAC,CACvD4D,IAAI,CAACnD,UAAU,IAAI;MAClBoC,oBAAW,CAACsB,GAAG,CAAC1D,UAAU,CAAC;MAC3B,OAAOA,UAAU;IACnB,CAAC,CAAC;EACN;EAEA2D,YAAYA,CACV5I,SAAiB,EACjB6I,oBAA6B,GAAG,KAAK,EACrCZ,OAA0B,GAAG;IAAEH,UAAU,EAAE;EAAM,CAAC,EACjC;IACjB,IAAIG,OAAO,CAACH,UAAU,EAAE;MACtBT,oBAAW,CAACyB,KAAK,CAAC,CAAC;IACrB;IACA,IAAID,oBAAoB,IAAI3H,eAAe,CAACwB,OAAO,CAAC1C,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE;MACnE,MAAMqF,IAAI,GAAG,IAAI,CAAC+B,UAAU,CAACpH,SAAS,CAAC;MACvC,OAAOwI,OAAO,CAACC,OAAO,CAAC;QACrBzI,SAAS;QACTwC,MAAM,EAAE6C,IAAI,CAAC7C,MAAM;QACnB8C,qBAAqB,EAAED,IAAI,CAACC,qBAAqB;QACjDR,OAAO,EAAEO,IAAI,CAACP;MAChB,CAAC,CAAC;IACJ;IACA,MAAMyD,MAAM,GAAGlB,oBAAW,CAACjC,GAAG,CAACpF,SAAS,CAAC;IACzC,IAAIuI,MAAM,IAAI,CAACN,OAAO,CAACH,UAAU,EAAE;MACjC,OAAOU,OAAO,CAACC,OAAO,CAACF,MAAM,CAAC;IAChC;IACA,OAAO,IAAI,CAACD,aAAa,CAAC,CAAC,CAACF,IAAI,CAACnD,UAAU,IAAI;MAC7C,MAAM8D,SAAS,GAAG9D,UAAU,CAAC+D,IAAI,CAACzE,MAAM,IAAIA,MAAM,CAACvE,SAAS,KAAKA,SAAS,CAAC;MAC3E,IAAI,CAAC+I,SAAS,EAAE;QACd,OAAOP,OAAO,CAACS,MAAM,CAAC7E,SAAS,CAAC;MAClC;MACA,OAAO2E,SAAS;IAClB,CAAC,CAAC;EACJ;;EAEA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,MAAMG,mBAAmBA,CACvBlJ,SAAiB,EACjBwC,MAAoB,GAAG,CAAC,CAAC,EACzB8C,qBAA0B,EAC1BR,OAAY,GAAG,CAAC,CAAC,EACO;IACxB,IAAIqE,eAAe,GAAG,IAAI,CAACC,gBAAgB,CAACpJ,SAAS,EAAEwC,MAAM,EAAE8C,qBAAqB,CAAC;IACrF,IAAI6D,eAAe,EAAE;MACnB,IAAIA,eAAe,YAAYlO,KAAK,CAACiH,KAAK,EAAE;QAC1C,OAAOsG,OAAO,CAACS,MAAM,CAACE,eAAe,CAAC;MACxC,CAAC,MAAM,IAAIA,eAAe,CAACE,IAAI,IAAIF,eAAe,CAACG,KAAK,EAAE;QACxD,OAAOd,OAAO,CAACS,MAAM,CAAC,IAAIhO,KAAK,CAACiH,KAAK,CAACiH,eAAe,CAACE,IAAI,EAAEF,eAAe,CAACG,KAAK,CAAC,CAAC;MACrF;MACA,OAAOd,OAAO,CAACS,MAAM,CAACE,eAAe,CAAC;IACxC;IACA,IAAI;MACF,MAAMI,aAAa,GAAG,MAAM,IAAI,CAACtC,UAAU,CAACuC,WAAW,CACrDxJ,SAAS,EACTsE,4BAA4B,CAAC;QAC3B9B,MAAM;QACN8C,qBAAqB;QACrBR,OAAO;QACP9E;MACF,CAAC,CACH,CAAC;MACD;MACA,MAAM,IAAI,CAAC6H,UAAU,CAAC;QAAEC,UAAU,EAAE;MAAK,CAAC,CAAC;MAC3C,MAAM2B,WAAW,GAAG7E,iCAAiC,CAAC2E,aAAa,CAAC;MACpE,OAAOE,WAAW;IACpB,CAAC,CAAC,OAAOH,KAAK,EAAE;MACd,IAAIA,KAAK,IAAIA,KAAK,CAACD,IAAI,KAAKpO,KAAK,CAACiH,KAAK,CAACwH,eAAe,EAAE;QACvD,MAAM,IAAIzO,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACiC,kBAAkB,EAAE,SAASnE,SAAS,kBAAkB,CAAC;MAC7F,CAAC,MAAM;QACL,MAAMsJ,KAAK;MACb;IACF;EACF;EAEAK,WAAWA,CACT3J,SAAiB,EACjB4J,eAA6B,EAC7BtE,qBAA0B,EAC1BR,OAAY,EACZ+E,QAA4B,EAC5B;IACA,OAAO,IAAI,CAACjB,YAAY,CAAC5I,SAAS,CAAC,CAChCoI,IAAI,CAAC7D,MAAM,IAAI;MACd,MAAMuF,cAAc,GAAGvF,MAAM,CAAC/B,MAAM;MACpC1J,MAAM,CAACC,IAAI,CAAC6Q,eAAe,CAAC,CAAClQ,OAAO,CAACsD,IAAI,IAAI;QAC3C,MAAMmG,KAAK,GAAGyG,eAAe,CAAC5M,IAAI,CAAC;QACnC,IACE8M,cAAc,CAAC9M,IAAI,CAAC,IACpB8M,cAAc,CAAC9M,IAAI,CAAC,CAACzB,IAAI,KAAK4H,KAAK,CAAC5H,IAAI,IACxC4H,KAAK,CAAC4G,IAAI,KAAK,QAAQ,EACvB;UACA,MAAM,IAAI9O,KAAK,CAACiH,KAAK,CAAC,GAAG,EAAE,SAASlF,IAAI,yBAAyB,CAAC;QACpE;QACA,IAAI,CAAC8M,cAAc,CAAC9M,IAAI,CAAC,IAAImG,KAAK,CAAC4G,IAAI,KAAK,QAAQ,EAAE;UACpD,MAAM,IAAI9O,KAAK,CAACiH,KAAK,CAAC,GAAG,EAAE,SAASlF,IAAI,iCAAiC,CAAC;QAC5E;MACF,CAAC,CAAC;MAEF,OAAO8M,cAAc,CAACrF,MAAM;MAC5B,OAAOqF,cAAc,CAACpF,MAAM;MAC5B,MAAMsF,SAAS,GAAGC,uBAAuB,CAACH,cAAc,EAAEF,eAAe,CAAC;MAC1E,MAAMM,aAAa,GAAGhP,cAAc,CAAC8E,SAAS,CAAC,IAAI9E,cAAc,CAACG,QAAQ;MAC1E,MAAM8O,aAAa,GAAGrR,MAAM,CAAC+B,MAAM,CAAC,CAAC,CAAC,EAAEmP,SAAS,EAAEE,aAAa,CAAC;MACjE,MAAMf,eAAe,GAAG,IAAI,CAACiB,kBAAkB,CAC7CpK,SAAS,EACTgK,SAAS,EACT1E,qBAAqB,EACrBxM,MAAM,CAACC,IAAI,CAAC+Q,cAAc,CAC5B,CAAC;MACD,IAAIX,eAAe,EAAE;QACnB,MAAM,IAAIlO,KAAK,CAACiH,KAAK,CAACiH,eAAe,CAACE,IAAI,EAAEF,eAAe,CAACG,KAAK,CAAC;MACpE;;MAEA;MACA;MACA,MAAMe,aAAuB,GAAG,EAAE;MAClC,MAAMC,cAAc,GAAG,EAAE;MACzBxR,MAAM,CAACC,IAAI,CAAC6Q,eAAe,CAAC,CAAClQ,OAAO,CAACmJ,SAAS,IAAI;QAChD,IAAI+G,eAAe,CAAC/G,SAAS,CAAC,CAACkH,IAAI,KAAK,QAAQ,EAAE;UAChDM,aAAa,CAAChR,IAAI,CAACwJ,SAAS,CAAC;QAC/B,CAAC,MAAM;UACLyH,cAAc,CAACjR,IAAI,CAACwJ,SAAS,CAAC;QAChC;MACF,CAAC,CAAC;MAEF,IAAI0H,aAAa,GAAG/B,OAAO,CAACC,OAAO,CAAC,CAAC;MACrC,IAAI4B,aAAa,CAAC5Q,MAAM,GAAG,CAAC,EAAE;QAC5B8Q,aAAa,GAAG,IAAI,CAACC,YAAY,CAACH,aAAa,EAAErK,SAAS,EAAE6J,QAAQ,CAAC;MACvE;MACA,IAAIY,aAAa,GAAG,EAAE;MACtB,OACEF,aAAa,CAAC;MAAA,CACXnC,IAAI,CAAC,MAAM,IAAI,CAACP,UAAU,CAAC;QAAEC,UAAU,EAAE;MAAK,CAAC,CAAC,CAAC,CAAC;MAAA,CAClDM,IAAI,CAAC,MAAM;QACV,MAAMsC,QAAQ,GAAGJ,cAAc,CAAC5B,GAAG,CAAC7F,SAAS,IAAI;UAC/C,MAAMtH,IAAI,GAAGqO,eAAe,CAAC/G,SAAS,CAAC;UACvC,OAAO,IAAI,CAAC8H,kBAAkB,CAAC3K,SAAS,EAAE6C,SAAS,EAAEtH,IAAI,CAAC;QAC5D,CAAC,CAAC;QACF,OAAOiN,OAAO,CAAClB,GAAG,CAACoD,QAAQ,CAAC;MAC9B,CAAC,CAAC,CACDtC,IAAI,CAACwC,OAAO,IAAI;QACfH,aAAa,GAAGG,OAAO,CAAC1R,MAAM,CAAC2R,MAAM,IAAI,CAAC,CAACA,MAAM,CAAC;QAClD,OAAO,IAAI,CAACC,cAAc,CAAC9K,SAAS,EAAEsF,qBAAqB,EAAE0E,SAAS,CAAC;MACzE,CAAC,CAAC,CACD5B,IAAI,CAAC,MACJ,IAAI,CAACnB,UAAU,CAAC8D,0BAA0B,CACxC/K,SAAS,EACT8E,OAAO,EACPP,MAAM,CAACO,OAAO,EACdqF,aACF,CACF,CAAC,CACA/B,IAAI,CAAC,MAAM,IAAI,CAACP,UAAU,CAAC;QAAEC,UAAU,EAAE;MAAK,CAAC,CAAC;MACjD;MAAA,CACCM,IAAI,CAAC,MAAM;QACV,IAAI,CAAC4C,YAAY,CAACP,aAAa,CAAC;QAChC,MAAMlG,MAAM,GAAG,IAAI,CAAC6C,UAAU,CAACpH,SAAS,CAAC;QACzC,MAAMiL,cAAsB,GAAG;UAC7BjL,SAAS,EAAEA,SAAS;UACpBwC,MAAM,EAAE+B,MAAM,CAAC/B,MAAM;UACrB8C,qBAAqB,EAAEf,MAAM,CAACe;QAChC,CAAC;QACD,IAAIf,MAAM,CAACO,OAAO,IAAIhM,MAAM,CAACC,IAAI,CAACwL,MAAM,CAACO,OAAO,CAAC,CAACrL,MAAM,KAAK,CAAC,EAAE;UAC9DwR,cAAc,CAACnG,OAAO,GAAGP,MAAM,CAACO,OAAO;QACzC;QACA,OAAOmG,cAAc;MACvB,CAAC,CAAC;IAER,CAAC,CAAC,CACDC,KAAK,CAAC5B,KAAK,IAAI;MACd,IAAIA,KAAK,KAAKlF,SAAS,EAAE;QACvB,MAAM,IAAInJ,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACiC,kBAAkB,EAC9B,SAASnE,SAAS,kBACpB,CAAC;MACH,CAAC,MAAM;QACL,MAAMsJ,KAAK;MACb;IACF,CAAC,CAAC;EACN;;EAEA;EACA;EACA6B,kBAAkBA,CAACnL,SAAiB,EAA6B;IAC/D,IAAI,IAAI,CAACoH,UAAU,CAACpH,SAAS,CAAC,EAAE;MAC9B,OAAOwI,OAAO,CAACC,OAAO,CAAC,IAAI,CAAC;IAC9B;IACA;IACA;MACE;MACA,IAAI,CAACS,mBAAmB,CAAClJ,SAAS,CAAC,CAChCkL,KAAK,CAAC,MAAM;QACX;QACA;QACA;QACA;QACA,OAAO,IAAI,CAACrD,UAAU,CAAC;UAAEC,UAAU,EAAE;QAAK,CAAC,CAAC;MAC9C,CAAC,CAAC,CACDM,IAAI,CAAC,MAAM;QACV;QACA,IAAI,IAAI,CAAChB,UAAU,CAACpH,SAAS,CAAC,EAAE;UAC9B,OAAO,IAAI;QACb,CAAC,MAAM;UACL,MAAM,IAAI/E,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EAAE,iBAAiBnC,SAAS,EAAE,CAAC;QAC/E;MACF,CAAC,CAAC,CACDkL,KAAK,CAAC,MAAM;QACX;QACA,MAAM,IAAIjQ,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EAAE,uCAAuC,CAAC;MAC1F,CAAC;IAAC;EAER;EAEAiH,gBAAgBA,CAACpJ,SAAiB,EAAEwC,MAAoB,GAAG,CAAC,CAAC,EAAE8C,qBAA0B,EAAO;IAC9F,IAAI,IAAI,CAAC8B,UAAU,CAACpH,SAAS,CAAC,EAAE;MAC9B,MAAM,IAAI/E,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACiC,kBAAkB,EAAE,SAASnE,SAAS,kBAAkB,CAAC;IAC7F;IACA,IAAI,CAAC0D,gBAAgB,CAAC1D,SAAS,CAAC,EAAE;MAChC,OAAO;QACLqJ,IAAI,EAAEpO,KAAK,CAACiH,KAAK,CAACiC,kBAAkB;QACpCmF,KAAK,EAAEvF,uBAAuB,CAAC/D,SAAS;MAC1C,CAAC;IACH;IACA,OAAO,IAAI,CAACoK,kBAAkB,CAACpK,SAAS,EAAEwC,MAAM,EAAE8C,qBAAqB,EAAE,EAAE,CAAC;EAC9E;EAEA8E,kBAAkBA,CAChBpK,SAAiB,EACjBwC,MAAoB,EACpB8C,qBAA4C,EAC5C8F,kBAAiC,EACjC;IACA,KAAK,MAAMvI,SAAS,IAAIL,MAAM,EAAE;MAC9B,IAAI4I,kBAAkB,CAAC1I,OAAO,CAACG,SAAS,CAAC,GAAG,CAAC,EAAE;QAC7C,IAAI,CAACe,gBAAgB,CAACf,SAAS,EAAE7C,SAAS,CAAC,EAAE;UAC3C,OAAO;YACLqJ,IAAI,EAAEpO,KAAK,CAACiH,KAAK,CAACmJ,gBAAgB;YAClC/B,KAAK,EAAE,sBAAsB,GAAGzG;UAClC,CAAC;QACH;QACA,IAAI,CAACiB,wBAAwB,CAACjB,SAAS,EAAE7C,SAAS,CAAC,EAAE;UACnD,OAAO;YACLqJ,IAAI,EAAE,GAAG;YACTC,KAAK,EAAE,QAAQ,GAAGzG,SAAS,GAAG;UAChC,CAAC;QACH;QACA,MAAMyI,SAAS,GAAG9I,MAAM,CAACK,SAAS,CAAC;QACnC,MAAMyG,KAAK,GAAGpF,kBAAkB,CAACoH,SAAS,CAAC;QAC3C,IAAIhC,KAAK,EAAE,OAAO;UAAED,IAAI,EAAEC,KAAK,CAACD,IAAI;UAAEC,KAAK,EAAEA,KAAK,CAAClK;QAAQ,CAAC;QAC5D,IAAIkM,SAAS,CAACC,YAAY,KAAKnH,SAAS,EAAE;UACxC,IAAIoH,gBAAgB,GAAGC,OAAO,CAACH,SAAS,CAACC,YAAY,CAAC;UACtD,IAAI,OAAOC,gBAAgB,KAAK,QAAQ,EAAE;YACxCA,gBAAgB,GAAG;cAAEjQ,IAAI,EAAEiQ;YAAiB,CAAC;UAC/C,CAAC,MAAM,IAAI,OAAOA,gBAAgB,KAAK,QAAQ,IAAIF,SAAS,CAAC/P,IAAI,KAAK,UAAU,EAAE;YAChF,OAAO;cACL8N,IAAI,EAAEpO,KAAK,CAACiH,KAAK,CAACmC,cAAc;cAChCiF,KAAK,EAAE,oDAAoD7C,YAAY,CAAC6E,SAAS,CAAC;YACpF,CAAC;UACH;UACA,IAAI,CAAChF,uBAAuB,CAACgF,SAAS,EAAEE,gBAAgB,CAAC,EAAE;YACzD,OAAO;cACLnC,IAAI,EAAEpO,KAAK,CAACiH,KAAK,CAACmC,cAAc;cAChCiF,KAAK,EAAE,uBAAuBtJ,SAAS,IAAI6C,SAAS,4BAA4B4D,YAAY,CAC1F6E,SACF,CAAC,YAAY7E,YAAY,CAAC+E,gBAAgB,CAAC;YAC7C,CAAC;UACH;QACF,CAAC,MAAM,IAAIF,SAAS,CAACI,QAAQ,EAAE;UAC7B,IAAI,OAAOJ,SAAS,KAAK,QAAQ,IAAIA,SAAS,CAAC/P,IAAI,KAAK,UAAU,EAAE;YAClE,OAAO;cACL8N,IAAI,EAAEpO,KAAK,CAACiH,KAAK,CAACmC,cAAc;cAChCiF,KAAK,EAAE,+CAA+C7C,YAAY,CAAC6E,SAAS,CAAC;YAC/E,CAAC;UACH;QACF;MACF;IACF;IAEA,KAAK,MAAMzI,SAAS,IAAI3H,cAAc,CAAC8E,SAAS,CAAC,EAAE;MACjDwC,MAAM,CAACK,SAAS,CAAC,GAAG3H,cAAc,CAAC8E,SAAS,CAAC,CAAC6C,SAAS,CAAC;IAC1D;IAEA,MAAM8I,SAAS,GAAG7S,MAAM,CAACC,IAAI,CAACyJ,MAAM,CAAC,CAACtJ,MAAM,CAC1C0I,GAAG,IAAIY,MAAM,CAACZ,GAAG,CAAC,IAAIY,MAAM,CAACZ,GAAG,CAAC,CAACrG,IAAI,KAAK,UAC7C,CAAC;IACD,IAAIoQ,SAAS,CAAClS,MAAM,GAAG,CAAC,EAAE;MACxB,OAAO;QACL4P,IAAI,EAAEpO,KAAK,CAACiH,KAAK,CAACmC,cAAc;QAChCiF,KAAK,EACH,oEAAoE,GACpEqC,SAAS,CAAC,CAAC,CAAC,GACZ,QAAQ,GACRA,SAAS,CAAC,CAAC,CAAC,GACZ;MACJ,CAAC;IACH;IACArJ,WAAW,CAACgD,qBAAqB,EAAE9C,MAAM,EAAE,IAAI,CAACmF,WAAW,CAAC;EAC9D;;EAEA;EACA,MAAMmD,cAAcA,CAAC9K,SAAiB,EAAEuC,KAAU,EAAEyH,SAAuB,EAAE;IAC3E,IAAI,OAAOzH,KAAK,KAAK,WAAW,EAAE;MAChC,OAAOiG,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B;IACAnG,WAAW,CAACC,KAAK,EAAEyH,SAAS,EAAE,IAAI,CAACrC,WAAW,CAAC;IAC/C,MAAM,IAAI,CAACV,UAAU,CAAC2E,wBAAwB,CAAC5L,SAAS,EAAEuC,KAAK,CAAC;IAChE,MAAMgG,MAAM,GAAGlB,oBAAW,CAACjC,GAAG,CAACpF,SAAS,CAAC;IACzC,IAAIuI,MAAM,EAAE;MACVA,MAAM,CAACjD,qBAAqB,GAAG/C,KAAK;IACtC;EACF;;EAEA;EACA;EACA;EACA;EACAoI,kBAAkBA,CAChB3K,SAAiB,EACjB6C,SAAiB,EACjBtH,IAA0B,EAC1BsQ,YAAsB,EACtBC,WAAqB,EACrB;IACA,IAAIjJ,SAAS,CAACH,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;MAC9B;MACA;MACA;MACA,MAAM,CAACqJ,CAAC,EAAEC,CAAC,CAAC,GAAGnJ,SAAS,CAACoJ,KAAK,CAAC,GAAG,CAAC;MACnCpJ,SAAS,GAAGkJ,CAAC;MACb,MAAMG,YAAY,GAAGjJ,KAAK,CAAC0C,IAAI,CAACqG,CAAC,CAAC,CAACG,KAAK,CAACC,CAAC,IAAIA,CAAC,IAAI,GAAG,IAAIA,CAAC,IAAI,GAAG,CAAC;MACnE,IAAIF,YAAY,IAAI,CAAC,CAAC,kBAAkB,EAAE,oBAAoB,CAAC,CAACrI,QAAQ,CAAChB,SAAS,CAAC,EAAE;QACnFtH,IAAI,GAAG,OAAO;MAChB,CAAC,MAAM;QACLA,IAAI,GAAG,QAAQ;MACjB;IACF;IACA,IAAI8Q,mBAAmB,GAAG,GAAGxJ,SAAS,EAAE;IACxC,IAAIiJ,WAAW,IAAIO,mBAAmB,CAACC,MAAM,CAAC,CAAC,CAAC,KAAK,GAAG,EAAE;MACxDD,mBAAmB,GAAGA,mBAAmB,CAACE,SAAS,CAAC,CAAC,CAAC;IACxD;IACA,IAAI,CAAC3I,gBAAgB,CAACyI,mBAAmB,EAAErM,SAAS,CAAC,EAAE;MACrD,MAAM,IAAI/E,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACmJ,gBAAgB,EAAE,uBAAuBxI,SAAS,GAAG,CAAC;IAC1F;;IAEA;IACA,IAAI,CAACtH,IAAI,EAAE;MACT,OAAO6I,SAAS;IAClB;IAEA,MAAMoI,YAAY,GAAG,IAAI,CAACC,eAAe,CAACzM,SAAS,EAAE6C,SAAS,CAAC;IAC/D,IAAI,OAAOtH,IAAI,KAAK,QAAQ,EAAE;MAC5BA,IAAI,GAAI;QAAEA;MAAK,CAAe;IAChC;IAEA,IAAIA,IAAI,CAACgQ,YAAY,KAAKnH,SAAS,EAAE;MACnC,IAAIoH,gBAAgB,GAAGC,OAAO,CAAClQ,IAAI,CAACgQ,YAAY,CAAC;MACjD,IAAI,OAAOC,gBAAgB,KAAK,QAAQ,EAAE;QACxCA,gBAAgB,GAAG;UAAEjQ,IAAI,EAAEiQ;QAAiB,CAAC;MAC/C;MACA,IAAI,CAAClF,uBAAuB,CAAC/K,IAAI,EAAEiQ,gBAAgB,CAAC,EAAE;QACpD,MAAM,IAAIvQ,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAC1B,uBAAuBrE,SAAS,IAAI6C,SAAS,4BAA4B4D,YAAY,CACnFlL,IACF,CAAC,YAAYkL,YAAY,CAAC+E,gBAAgB,CAAC,EAC7C,CAAC;MACH;IACF;IAEA,IAAIgB,YAAY,EAAE;MAChB,IAAI,CAAClG,uBAAuB,CAACkG,YAAY,EAAEjR,IAAI,CAAC,EAAE;QAChD,MAAM,IAAIN,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAC1B,uBAAuBrE,SAAS,IAAI6C,SAAS,cAAc4D,YAAY,CACrE+F,YACF,CAAC,YAAY/F,YAAY,CAAClL,IAAI,CAAC,EACjC,CAAC;MACH;MACA;MACA;MACA,IAAIsQ,YAAY,IAAIa,IAAI,CAACC,SAAS,CAACH,YAAY,CAAC,KAAKE,IAAI,CAACC,SAAS,CAACpR,IAAI,CAAC,EAAE;QACzE,OAAO6I,SAAS;MAClB;MACA;MACA;MACA,OAAO,IAAI,CAAC6C,UAAU,CAAC2F,kBAAkB,CAAC5M,SAAS,EAAE6C,SAAS,EAAEtH,IAAI,CAAC;IACvE;IAEA,OAAO,IAAI,CAAC0L,UAAU,CACnB4F,mBAAmB,CAAC7M,SAAS,EAAE6C,SAAS,EAAEtH,IAAI,CAAC,CAC/C2P,KAAK,CAAC5B,KAAK,IAAI;MACd,IAAIA,KAAK,CAACD,IAAI,IAAIpO,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAAE;QAC5C;QACA,MAAMiF,KAAK;MACb;MACA;MACA;MACA;MACA,OAAOd,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B,CAAC,CAAC,CACDL,IAAI,CAAC,MAAM;MACV,OAAO;QACLpI,SAAS;QACT6C,SAAS;QACTtH;MACF,CAAC;IACH,CAAC,CAAC;EACN;EAEAyP,YAAYA,CAACxI,MAAW,EAAE;IACxB,KAAK,IAAIrI,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGqI,MAAM,CAAC/I,MAAM,EAAEU,CAAC,IAAI,CAAC,EAAE;MACzC,MAAM;QAAE6F,SAAS;QAAE6C;MAAU,CAAC,GAAGL,MAAM,CAACrI,CAAC,CAAC;MAC1C,IAAI;QAAEoB;MAAK,CAAC,GAAGiH,MAAM,CAACrI,CAAC,CAAC;MACxB,MAAMqS,YAAY,GAAG,IAAI,CAACC,eAAe,CAACzM,SAAS,EAAE6C,SAAS,CAAC;MAC/D,IAAI,OAAOtH,IAAI,KAAK,QAAQ,EAAE;QAC5BA,IAAI,GAAG;UAAEA,IAAI,EAAEA;QAAK,CAAC;MACvB;MACA,IAAI,CAACiR,YAAY,IAAI,CAAClG,uBAAuB,CAACkG,YAAY,EAAEjR,IAAI,CAAC,EAAE;QACjE,MAAM,IAAIN,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACC,YAAY,EAAE,uBAAuBU,SAAS,EAAE,CAAC;MACrF;IACF;EACF;;EAEA;EACAiK,WAAWA,CAACjK,SAAiB,EAAE7C,SAAiB,EAAE6J,QAA4B,EAAE;IAC9E,OAAO,IAAI,CAACW,YAAY,CAAC,CAAC3H,SAAS,CAAC,EAAE7C,SAAS,EAAE6J,QAAQ,CAAC;EAC5D;;EAEA;EACA;EACA;EACA;EACA;EACA;EACA;EACAW,YAAYA,CAACuC,UAAyB,EAAE/M,SAAiB,EAAE6J,QAA4B,EAAE;IACvF,IAAI,CAACnG,gBAAgB,CAAC1D,SAAS,CAAC,EAAE;MAChC,MAAM,IAAI/E,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACiC,kBAAkB,EAAEJ,uBAAuB,CAAC/D,SAAS,CAAC,CAAC;IAC3F;IAEA+M,UAAU,CAACrT,OAAO,CAACmJ,SAAS,IAAI;MAC9B,IAAI,CAACe,gBAAgB,CAACf,SAAS,EAAE7C,SAAS,CAAC,EAAE;QAC3C,MAAM,IAAI/E,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACmJ,gBAAgB,EAAE,uBAAuBxI,SAAS,EAAE,CAAC;MACzF;MACA;MACA,IAAI,CAACiB,wBAAwB,CAACjB,SAAS,EAAE7C,SAAS,CAAC,EAAE;QACnD,MAAM,IAAI/E,KAAK,CAACiH,KAAK,CAAC,GAAG,EAAE,SAASW,SAAS,oBAAoB,CAAC;MACpE;IACF,CAAC,CAAC;IAEF,OAAO,IAAI,CAAC+F,YAAY,CAAC5I,SAAS,EAAE,KAAK,EAAE;MAAE8H,UAAU,EAAE;IAAK,CAAC,CAAC,CAC7DoD,KAAK,CAAC5B,KAAK,IAAI;MACd,IAAIA,KAAK,KAAKlF,SAAS,EAAE;QACvB,MAAM,IAAInJ,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACiC,kBAAkB,EAC9B,SAASnE,SAAS,kBACpB,CAAC;MACH,CAAC,MAAM;QACL,MAAMsJ,KAAK;MACb;IACF,CAAC,CAAC,CACDlB,IAAI,CAAC7D,MAAM,IAAI;MACdwI,UAAU,CAACrT,OAAO,CAACmJ,SAAS,IAAI;QAC9B,IAAI,CAAC0B,MAAM,CAAC/B,MAAM,CAACK,SAAS,CAAC,EAAE;UAC7B,MAAM,IAAI5H,KAAK,CAACiH,KAAK,CAAC,GAAG,EAAE,SAASW,SAAS,iCAAiC,CAAC;QACjF;MACF,CAAC,CAAC;MAEF,MAAMmK,YAAY,GAAAzT,aAAA,KAAQgL,MAAM,CAAC/B,MAAM,CAAE;MACzC,OAAOqH,QAAQ,CAACoD,OAAO,CAACzC,YAAY,CAACxK,SAAS,EAAEuE,MAAM,EAAEwI,UAAU,CAAC,CAAC3E,IAAI,CAAC,MAAM;QAC7E,OAAOI,OAAO,CAAClB,GAAG,CAChByF,UAAU,CAACrE,GAAG,CAAC7F,SAAS,IAAI;UAC1B,MAAMM,KAAK,GAAG6J,YAAY,CAACnK,SAAS,CAAC;UACrC,IAAIM,KAAK,IAAIA,KAAK,CAAC5H,IAAI,KAAK,UAAU,EAAE;YACtC;YACA,OAAOsO,QAAQ,CAACoD,OAAO,CAACC,WAAW,CAAC,SAASrK,SAAS,IAAI7C,SAAS,EAAE,CAAC;UACxE;UACA,OAAOwI,OAAO,CAACC,OAAO,CAAC,CAAC;QAC1B,CAAC,CACH,CAAC;MACH,CAAC,CAAC;IACJ,CAAC,CAAC,CACDL,IAAI,CAAC,MAAM;MACVf,oBAAW,CAACyB,KAAK,CAAC,CAAC;IACrB,CAAC,CAAC;EACN;;EAEA;EACA;EACA;EACA,MAAMqE,cAAcA,CAACnN,SAAiB,EAAEoN,MAAW,EAAEhP,KAAU,EAAE0N,WAAoB,EAAE;IACrF,IAAIuB,QAAQ,GAAG,CAAC;IAChB,MAAM9I,MAAM,GAAG,MAAM,IAAI,CAAC4G,kBAAkB,CAACnL,SAAS,CAAC;IACvD,MAAM0K,QAAQ,GAAG,EAAE;IAEnB,KAAK,MAAM7H,SAAS,IAAIuK,MAAM,EAAE;MAC9B,IAAIA,MAAM,CAACvK,SAAS,CAAC,IAAI4I,OAAO,CAAC2B,MAAM,CAACvK,SAAS,CAAC,CAAC,KAAK,UAAU,EAAE;QAClEwK,QAAQ,EAAE;MACZ;MACA,IAAIA,QAAQ,GAAG,CAAC,EAAE;QAChB,OAAO7E,OAAO,CAACS,MAAM,CACnB,IAAIhO,KAAK,CAACiH,KAAK,CACbjH,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAC1B,iDACF,CACF,CAAC;MACH;IACF;IACA,KAAK,MAAMxB,SAAS,IAAIuK,MAAM,EAAE;MAC9B,IAAIA,MAAM,CAACvK,SAAS,CAAC,KAAKuB,SAAS,EAAE;QACnC;MACF;MACA,MAAMkJ,QAAQ,GAAG7B,OAAO,CAAC2B,MAAM,CAACvK,SAAS,CAAC,CAAC;MAC3C,IAAI,CAACyK,QAAQ,EAAE;QACb;MACF;MACA,IAAIzK,SAAS,KAAK,KAAK,EAAE;QACvB;QACA;MACF;MACA6H,QAAQ,CAACrR,IAAI,CAACkL,MAAM,CAACoG,kBAAkB,CAAC3K,SAAS,EAAE6C,SAAS,EAAEyK,QAAQ,EAAE,IAAI,EAAExB,WAAW,CAAC,CAAC;IAC7F;IACA,MAAMlB,OAAO,GAAG,MAAMpC,OAAO,CAAClB,GAAG,CAACoD,QAAQ,CAAC;IAC3C,MAAMD,aAAa,GAAGG,OAAO,CAAC1R,MAAM,CAAC2R,MAAM,IAAI,CAAC,CAACA,MAAM,CAAC;IAExD,IAAIJ,aAAa,CAAChR,MAAM,KAAK,CAAC,EAAE;MAC9B;MACA,MAAM,IAAI,CAACoO,UAAU,CAAC;QAAEC,UAAU,EAAE;MAAK,CAAC,CAAC;IAC7C;IACA,IAAI,CAACkD,YAAY,CAACP,aAAa,CAAC;IAEhC,MAAM8C,OAAO,GAAG/E,OAAO,CAACC,OAAO,CAAClE,MAAM,CAAC;IACvC,OAAOiJ,2BAA2B,CAACD,OAAO,EAAEvN,SAAS,EAAEoN,MAAM,EAAEhP,KAAK,CAAC;EACvE;;EAEA;EACAqP,uBAAuBA,CAACzN,SAAiB,EAAEoN,MAAW,EAAEhP,KAAU,EAAE;IAClE,MAAMsP,OAAO,GAAG7M,eAAe,CAACE,KAAK,CAACf,SAAS,CAAC;IAChD,IAAI,CAAC0N,OAAO,IAAIA,OAAO,CAACjU,MAAM,IAAI,CAAC,EAAE;MACnC,OAAO+O,OAAO,CAACC,OAAO,CAAC,IAAI,CAAC;IAC9B;IAEA,MAAMkF,cAAc,GAAGD,OAAO,CAACxU,MAAM,CAAC,UAAU0U,MAAM,EAAE;MACtD,IAAIxP,KAAK,IAAIA,KAAK,CAAC9C,QAAQ,EAAE;QAC3B,IAAI8R,MAAM,CAACQ,MAAM,CAAC,IAAI,OAAOR,MAAM,CAACQ,MAAM,CAAC,KAAK,QAAQ,EAAE;UACxD;UACA,OAAOR,MAAM,CAACQ,MAAM,CAAC,CAAC7D,IAAI,IAAI,QAAQ;QACxC;QACA;QACA,OAAO,KAAK;MACd;MACA,OAAO,CAACqD,MAAM,CAACQ,MAAM,CAAC;IACxB,CAAC,CAAC;IAEF,IAAID,cAAc,CAAClU,MAAM,GAAG,CAAC,EAAE;MAC7B,MAAM,IAAIwB,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAAEsJ,cAAc,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC;IACxF;IACA,OAAOnF,OAAO,CAACC,OAAO,CAAC,IAAI,CAAC;EAC9B;EAEAoF,2BAA2BA,CAAC7N,SAAiB,EAAE8N,QAAkB,EAAEnL,SAAiB,EAAE;IACpF,OAAOoE,gBAAgB,CAACgH,eAAe,CACrC,IAAI,CAACC,wBAAwB,CAAChO,SAAS,CAAC,EACxC8N,QAAQ,EACRnL,SACF,CAAC;EACH;;EAEA;EACA,OAAOoL,eAAeA,CAACE,gBAAsB,EAAEH,QAAkB,EAAEnL,SAAiB,EAAW;IAC7F,IAAI,CAACsL,gBAAgB,IAAI,CAACA,gBAAgB,CAACtL,SAAS,CAAC,EAAE;MACrD,OAAO,IAAI;IACb;IACA,MAAMJ,KAAK,GAAG0L,gBAAgB,CAACtL,SAAS,CAAC;IACzC,IAAIJ,KAAK,CAAC,GAAG,CAAC,EAAE;MACd,OAAO,IAAI;IACb;IACA;IACA,IACEuL,QAAQ,CAACI,IAAI,CAACC,GAAG,IAAI;MACnB,OAAO5L,KAAK,CAAC4L,GAAG,CAAC,KAAK,IAAI;IAC5B,CAAC,CAAC,EACF;MACA,OAAO,IAAI;IACb;IACA,OAAO,KAAK;EACd;;EAEA;EACA,OAAOC,kBAAkBA,CACvBH,gBAAsB,EACtBjO,SAAiB,EACjB8N,QAAkB,EAClBnL,SAAiB,EACjB0L,MAAe,EACf;IACA,IAAItH,gBAAgB,CAACgH,eAAe,CAACE,gBAAgB,EAAEH,QAAQ,EAAEnL,SAAS,CAAC,EAAE;MAC3E,OAAO6F,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B;IAEA,IAAI,CAACwF,gBAAgB,IAAI,CAACA,gBAAgB,CAACtL,SAAS,CAAC,EAAE;MACrD,OAAO,IAAI;IACb;IACA,MAAMJ,KAAK,GAAG0L,gBAAgB,CAACtL,SAAS,CAAC;IACzC;IACA;IACA,IAAIJ,KAAK,CAAC,wBAAwB,CAAC,EAAE;MACnC;MACA,IAAI,CAACuL,QAAQ,IAAIA,QAAQ,CAACrU,MAAM,IAAI,CAAC,EAAE;QACrC,MAAM,IAAIwB,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACoM,gBAAgB,EAC5B,oDACF,CAAC;MACH,CAAC,MAAM,IAAIR,QAAQ,CAACpL,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,IAAIoL,QAAQ,CAACrU,MAAM,IAAI,CAAC,EAAE;QAC7D,MAAM,IAAIwB,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACoM,gBAAgB,EAC5B,oDACF,CAAC;MACH;MACA;MACA;MACA,OAAO9F,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B;;IAEA;IACA;IACA,MAAM8F,eAAe,GACnB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC7L,OAAO,CAACC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,gBAAgB,GAAG,iBAAiB;;IAEzF;IACA,IAAI4L,eAAe,IAAI,iBAAiB,IAAI5L,SAAS,IAAI,QAAQ,EAAE;MACjE,MAAM,IAAI1H,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACsM,mBAAmB,EAC/B,gCAAgC7L,SAAS,aAAa3C,SAAS,GACjE,CAAC;IACH;;IAEA;IACA,IACEiD,KAAK,CAACC,OAAO,CAAC+K,gBAAgB,CAACM,eAAe,CAAC,CAAC,IAChDN,gBAAgB,CAACM,eAAe,CAAC,CAAC9U,MAAM,GAAG,CAAC,EAC5C;MACA,OAAO+O,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B;IAEA,MAAMpF,aAAa,GAAG4K,gBAAgB,CAACtL,SAAS,CAAC,CAACU,aAAa;IAC/D,IAAIJ,KAAK,CAACC,OAAO,CAACG,aAAa,CAAC,IAAIA,aAAa,CAAC5J,MAAM,GAAG,CAAC,EAAE;MAC5D;MACA,IAAIkJ,SAAS,KAAK,UAAU,IAAI0L,MAAM,KAAK,QAAQ,EAAE;QACnD;QACA,OAAO7F,OAAO,CAACC,OAAO,CAAC,CAAC;MAC1B;IACF;IAEA,MAAM,IAAIxN,KAAK,CAACiH,KAAK,CACnBjH,KAAK,CAACiH,KAAK,CAACsM,mBAAmB,EAC/B,gCAAgC7L,SAAS,aAAa3C,SAAS,GACjE,CAAC;EACH;;EAEA;EACAoO,kBAAkBA,CAACpO,SAAiB,EAAE8N,QAAkB,EAAEnL,SAAiB,EAAE0L,MAAe,EAAE;IAC5F,OAAOtH,gBAAgB,CAACqH,kBAAkB,CACxC,IAAI,CAACJ,wBAAwB,CAAChO,SAAS,CAAC,EACxCA,SAAS,EACT8N,QAAQ,EACRnL,SAAS,EACT0L,MACF,CAAC;EACH;EAEAL,wBAAwBA,CAAChO,SAAiB,EAAO;IAC/C,OAAO,IAAI,CAACoH,UAAU,CAACpH,SAAS,CAAC,IAAI,IAAI,CAACoH,UAAU,CAACpH,SAAS,CAAC,CAACsF,qBAAqB;EACvF;;EAEA;EACA;EACAmH,eAAeA,CAACzM,SAAiB,EAAE6C,SAAiB,EAA2B;IAC7E,IAAI,IAAI,CAACuE,UAAU,CAACpH,SAAS,CAAC,EAAE;MAC9B,MAAMwM,YAAY,GAAG,IAAI,CAACpF,UAAU,CAACpH,SAAS,CAAC,CAACwC,MAAM,CAACK,SAAS,CAAC;MACjE,OAAO2J,YAAY,KAAK,KAAK,GAAG,QAAQ,GAAGA,YAAY;IACzD;IACA,OAAOpI,SAAS;EAClB;;EAEA;EACAqK,QAAQA,CAACzO,SAAiB,EAAE;IAC1B,IAAI,IAAI,CAACoH,UAAU,CAACpH,SAAS,CAAC,EAAE;MAC9B,OAAOwI,OAAO,CAACC,OAAO,CAAC,IAAI,CAAC;IAC9B;IACA,OAAO,IAAI,CAACZ,UAAU,CAAC,CAAC,CAACO,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAChB,UAAU,CAACpH,SAAS,CAAC,CAAC;EACnE;AACF;;AAEA;AAAA7E,OAAA,CAAA4L,gBAAA,GAAA5L,OAAA,CAAAzC,OAAA,GAAAqO,gBAAA;AACA,MAAM2H,IAAI,GAAGA,CAACC,SAAyB,EAAE1G,OAAY,KAAgC;EACnF,MAAM1D,MAAM,GAAG,IAAIwC,gBAAgB,CAAC4H,SAAS,CAAC;EAC9CjI,GAAG,CAACI,QAAQ,GAAG6H,SAAS,CAACC,cAAc;EACvC,OAAOrK,MAAM,CAACsD,UAAU,CAACI,OAAO,CAAC,CAACG,IAAI,CAAC,MAAM7D,MAAM,CAAC;AACtD,CAAC;;AAED;AACA;AACA;AACA;AACA;AAAApJ,OAAA,CAAAuT,IAAA,GAAAA,IAAA;AACA,SAASzE,uBAAuBA,CAACH,cAA4B,EAAE+E,UAAe,EAAgB;EAC5F,MAAM7E,SAAS,GAAG,CAAC,CAAC;EACpB;EACA,MAAM8E,cAAc,GAClBhW,MAAM,CAACC,IAAI,CAACmC,cAAc,CAAC,CAACwH,OAAO,CAACoH,cAAc,CAACiF,GAAG,CAAC,KAAK,CAAC,CAAC,GAC1D,EAAE,GACFjW,MAAM,CAACC,IAAI,CAACmC,cAAc,CAAC4O,cAAc,CAACiF,GAAG,CAAC,CAAC;EACrD,KAAK,MAAMC,QAAQ,IAAIlF,cAAc,EAAE;IACrC,IACEkF,QAAQ,KAAK,KAAK,IAClBA,QAAQ,KAAK,KAAK,IAClBA,QAAQ,KAAK,WAAW,IACxBA,QAAQ,KAAK,WAAW,IACxBA,QAAQ,KAAK,UAAU,EACvB;MACA,IAAIF,cAAc,CAACrV,MAAM,GAAG,CAAC,IAAIqV,cAAc,CAACpM,OAAO,CAACsM,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE;QACxE;MACF;MACA,MAAMC,cAAc,GAAGJ,UAAU,CAACG,QAAQ,CAAC,IAAIH,UAAU,CAACG,QAAQ,CAAC,CAACjF,IAAI,KAAK,QAAQ;MACrF,IAAI,CAACkF,cAAc,EAAE;QACnBjF,SAAS,CAACgF,QAAQ,CAAC,GAAGlF,cAAc,CAACkF,QAAQ,CAAC;MAChD;IACF;EACF;EACA,KAAK,MAAME,QAAQ,IAAIL,UAAU,EAAE;IACjC,IAAIK,QAAQ,KAAK,UAAU,IAAIL,UAAU,CAACK,QAAQ,CAAC,CAACnF,IAAI,KAAK,QAAQ,EAAE;MACrE,IAAI+E,cAAc,CAACrV,MAAM,GAAG,CAAC,IAAIqV,cAAc,CAACpM,OAAO,CAACwM,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE;QACxE;MACF;MACAlF,SAAS,CAACkF,QAAQ,CAAC,GAAGL,UAAU,CAACK,QAAQ,CAAC;IAC5C;EACF;EACA,OAAOlF,SAAS;AAClB;;AAEA;AACA;AACA,SAASwD,2BAA2BA,CAAC2B,aAAa,EAAEnP,SAAS,EAAEoN,MAAM,EAAEhP,KAAK,EAAE;EAC5E,OAAO+Q,aAAa,CAAC/G,IAAI,CAAC7D,MAAM,IAAI;IAClC,OAAOA,MAAM,CAACkJ,uBAAuB,CAACzN,SAAS,EAAEoN,MAAM,EAAEhP,KAAK,CAAC;EACjE,CAAC,CAAC;AACJ;;AAEA;AACA;AACA;AACA;AACA;AACA,SAASqN,OAAOA,CAAC2D,GAAQ,EAA2B;EAClD,MAAM7T,IAAI,GAAG,OAAO6T,GAAG;EACvB,QAAQ7T,IAAI;IACV,KAAK,SAAS;MACZ,OAAO,SAAS;IAClB,KAAK,QAAQ;MACX,OAAO,QAAQ;IACjB,KAAK,QAAQ;MACX,OAAO,QAAQ;IACjB,KAAK,KAAK;IACV,KAAK,QAAQ;MACX,IAAI,CAAC6T,GAAG,EAAE;QACR,OAAOhL,SAAS;MAClB;MACA,OAAOiL,aAAa,CAACD,GAAG,CAAC;IAC3B,KAAK,UAAU;IACf,KAAK,QAAQ;IACb,KAAK,WAAW;IAChB;MACE,MAAM,WAAW,GAAGA,GAAG;EAC3B;AACF;;AAEA;AACA;AACA;AACA,SAASC,aAAaA,CAACD,GAAG,EAA2B;EACnD,IAAIA,GAAG,YAAYnM,KAAK,EAAE;IACxB,OAAO,OAAO;EAChB;EACA,IAAImM,GAAG,CAACE,MAAM,EAAE;IACd,QAAQF,GAAG,CAACE,MAAM;MAChB,KAAK,SAAS;QACZ,IAAIF,GAAG,CAACpP,SAAS,EAAE;UACjB,OAAO;YACLzE,IAAI,EAAE,SAAS;YACf2B,WAAW,EAAEkS,GAAG,CAACpP;UACnB,CAAC;QACH;QACA;MACF,KAAK,UAAU;QACb,IAAIoP,GAAG,CAACpP,SAAS,EAAE;UACjB,OAAO;YACLzE,IAAI,EAAE,UAAU;YAChB2B,WAAW,EAAEkS,GAAG,CAACpP;UACnB,CAAC;QACH;QACA;MACF,KAAK,MAAM;QACT,IAAIoP,GAAG,CAACpS,IAAI,EAAE;UACZ,OAAO,MAAM;QACf;QACA;MACF,KAAK,MAAM;QACT,IAAIoS,GAAG,CAACG,GAAG,EAAE;UACX,OAAO,MAAM;QACf;QACA;MACF,KAAK,UAAU;QACb,IAAIH,GAAG,CAACI,QAAQ,IAAI,IAAI,IAAIJ,GAAG,CAACK,SAAS,IAAI,IAAI,EAAE;UACjD,OAAO,UAAU;QACnB;QACA;MACF,KAAK,OAAO;QACV,IAAIL,GAAG,CAACM,MAAM,EAAE;UACd,OAAO,OAAO;QAChB;QACA;MACF,KAAK,SAAS;QACZ,IAAIN,GAAG,CAACO,WAAW,EAAE;UACnB,OAAO,SAAS;QAClB;QACA;IACJ;IACA,MAAM,IAAI1U,KAAK,CAACiH,KAAK,CAACjH,KAAK,CAACiH,KAAK,CAACmC,cAAc,EAAE,sBAAsB,GAAG+K,GAAG,CAACE,MAAM,CAAC;EACxF;EACA,IAAIF,GAAG,CAAC,KAAK,CAAC,EAAE;IACd,OAAOC,aAAa,CAACD,GAAG,CAAC,KAAK,CAAC,CAAC;EAClC;EACA,IAAIA,GAAG,CAACrF,IAAI,EAAE;IACZ,QAAQqF,GAAG,CAACrF,IAAI;MACd,KAAK,WAAW;QACd,OAAO,QAAQ;MACjB,KAAK,QAAQ;QACX,OAAO,IAAI;MACb,KAAK,KAAK;MACV,KAAK,WAAW;MAChB,KAAK,QAAQ;QACX,OAAO,OAAO;MAChB,KAAK,aAAa;MAClB,KAAK,gBAAgB;QACnB,OAAO;UACLxO,IAAI,EAAE,UAAU;UAChB2B,WAAW,EAAEkS,GAAG,CAACQ,OAAO,CAAC,CAAC,CAAC,CAAC5P;QAC9B,CAAC;MACH,KAAK,OAAO;QACV,OAAOqP,aAAa,CAACD,GAAG,CAACS,GAAG,CAAC,CAAC,CAAC,CAAC;MAClC;QACE,MAAM,iBAAiB,GAAGT,GAAG,CAACrF,IAAI;IACtC;EACF;EACA,OAAO,QAAQ;AACjB","ignoreList":[]}
|