"use strict"; var _node = require("parse/node"); var _lodash = _interopRequireDefault(require("lodash")); var _intersect = _interopRequireDefault(require("intersect")); var _deepcopy = _interopRequireDefault(require("deepcopy")); var _logger = _interopRequireDefault(require("../logger")); var _Utils = _interopRequireDefault(require("../Utils")); var SchemaController = _interopRequireWildcard(require("./SchemaController")); var _StorageAdapter = require("../Adapters/Storage/StorageAdapter"); var _MongoStorageAdapter = _interopRequireDefault(require("../Adapters/Storage/Mongo/MongoStorageAdapter")); var _PostgresStorageAdapter = _interopRequireDefault(require("../Adapters/Storage/Postgres/PostgresStorageAdapter")); var _SchemaCache = _interopRequireDefault(require("../Adapters/Cache/SchemaCache")); const _excluded = ["ACL"], _excluded2 = ["_rperm", "_wperm"]; // A database adapter that works with data exported from the hosted // Parse database. // -disable-next // -disable-next // -disable-next // -disable-next function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; } 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 _objectWithoutProperties(e, t) { if (null == e) return {}; var o, r, i = _objectWithoutPropertiesLoose(e, t); if (Object.getOwnPropertySymbols) { var n = Object.getOwnPropertySymbols(e); for (r = 0; r < n.length; r++) o = n[r], t.indexOf(o) >= 0 || {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]); } return i; } function _objectWithoutPropertiesLoose(r, e) { if (null == r) return {}; var t = {}; for (var n in r) if ({}.hasOwnProperty.call(r, n)) { if (e.indexOf(n) >= 0) continue; t[n] = r[n]; } return t; } function addWriteACL(query, acl) { const newQuery = _lodash.default.cloneDeep(query); //Can't be any existing '_wperm' query, we don't allow client queries on that, no need to $and newQuery._wperm = { $in: [null, ...acl] }; return newQuery; } function addReadACL(query, acl) { const newQuery = _lodash.default.cloneDeep(query); //Can't be any existing '_rperm' query, we don't allow client queries on that, no need to $and newQuery._rperm = { $in: [null, '*', ...acl] }; return newQuery; } // Transforms a REST API formatted ACL object to our two-field mongo format. const transformObjectACL = _ref => { let { ACL } = _ref, result = _objectWithoutProperties(_ref, _excluded); if (!ACL) { return result; } result._wperm = []; result._rperm = []; for (const entry in ACL) { if (ACL[entry].read) { result._rperm.push(entry); } if (ACL[entry].write) { result._wperm.push(entry); } } return result; }; const specialQueryKeys = ['$and', '$or', '$nor', '_rperm', '_wperm']; const specialMasterQueryKeys = [...specialQueryKeys, '_email_verify_token', '_perishable_token', '_tombstone', '_email_verify_token_expires_at', '_failed_login_count', '_account_lockout_expires_at', '_password_changed_at', '_password_history']; const validateQuery = (query, isMaster, isMaintenance, update) => { if (isMaintenance) { isMaster = true; } if (query.ACL) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Cannot query on ACL.'); } if (query.$or) { if (query.$or instanceof Array) { query.$or.forEach(value => validateQuery(value, isMaster, isMaintenance, update)); } else { throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Bad $or format - use an array value.'); } } if (query.$and) { if (query.$and instanceof Array) { query.$and.forEach(value => validateQuery(value, isMaster, isMaintenance, update)); } else { throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Bad $and format - use an array value.'); } } if (query.$nor) { if (query.$nor instanceof Array && query.$nor.length > 0) { query.$nor.forEach(value => validateQuery(value, isMaster, isMaintenance, update)); } else { throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Bad $nor format - use an array of at least 1 value.'); } } Object.keys(query).forEach(key => { if (query && query[key] && query[key].$regex) { if (typeof query[key].$options === 'string') { if (!query[key].$options.match(/^[imxs]+$/)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, `Bad $options value for query: ${query[key].$options}`); } } } if (!key.match(/^[a-zA-Z][a-zA-Z0-9_\.]*$/) && (!specialQueryKeys.includes(key) && !isMaster && !update || update && isMaster && !specialMasterQueryKeys.includes(key))) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Invalid key name: ${key}`); } }); }; // Filters out any data that shouldn't be on this REST-formatted object. const filterSensitiveData = (isMaster, isMaintenance, aclGroup, auth, operation, schema, className, protectedFields, object) => { let userId = null; if (auth && auth.user) userId = auth.user.id; // replace protectedFields when using pointer-permissions const perms = schema && schema.getClassLevelPermissions ? schema.getClassLevelPermissions(className) : {}; if (perms) { const isReadOperation = ['get', 'find'].indexOf(operation) > -1; if (isReadOperation && perms.protectedFields) { // extract protectedFields added with the pointer-permission prefix const protectedFieldsPointerPerm = Object.keys(perms.protectedFields).filter(key => key.startsWith('userField:')).map(key => { return { key: key.substring(10), value: perms.protectedFields[key] }; }); const newProtectedFields = []; let overrideProtectedFields = false; // check if the object grants the current user access based on the extracted fields protectedFieldsPointerPerm.forEach(pointerPerm => { let pointerPermIncludesUser = false; const readUserFieldValue = object[pointerPerm.key]; if (readUserFieldValue) { if (Array.isArray(readUserFieldValue)) { pointerPermIncludesUser = readUserFieldValue.some(user => user.objectId && user.objectId === userId); } else { pointerPermIncludesUser = readUserFieldValue.objectId && readUserFieldValue.objectId === userId; } } if (pointerPermIncludesUser) { overrideProtectedFields = true; newProtectedFields.push(pointerPerm.value); } }); // if at least one pointer-permission affected the current user // intersect vs protectedFields from previous stage (@see addProtectedFields) // Sets theory (intersections): A x (B x C) == (A x B) x C if (overrideProtectedFields && protectedFields) { newProtectedFields.push(protectedFields); } // intersect all sets of protectedFields newProtectedFields.forEach(fields => { if (fields) { // if there're no protctedFields by other criteria ( id / role / auth) // then we must intersect each set (per userField) if (!protectedFields) { protectedFields = fields; } else { protectedFields = protectedFields.filter(v => fields.includes(v)); } } }); } } const isUserClass = className === '_User'; if (isUserClass) { object.password = object._hashed_password; delete object._hashed_password; delete object.sessionToken; } if (isMaintenance) { return object; } /* special treat for the user class: don't filter protectedFields if currently loggedin user is the retrieved user */ if (!(isUserClass && userId && object.objectId === userId)) { var _perms$protectedField; protectedFields && protectedFields.forEach(k => delete object[k]); // fields not requested by client (excluded), // but were needed to apply protectedFields perms === null || perms === void 0 || (_perms$protectedField = perms.protectedFields) === null || _perms$protectedField === void 0 || (_perms$protectedField = _perms$protectedField.temporaryKeys) === null || _perms$protectedField === void 0 || _perms$protectedField.forEach(k => delete object[k]); } for (const key in object) { if (key.charAt(0) === '_') { delete object[key]; } } if (!isUserClass || isMaster) { return object; } if (aclGroup.indexOf(object.objectId) > -1) { return object; } delete object.authData; return object; }; // Runs an update on the database. // Returns a promise for an object with the new values for field // modifications that don't know their results ahead of time, like // 'increment'. // Options: // acl: a list of strings. If the object to be updated has an ACL, // one of the provided strings must provide the caller with // write permissions. const specialKeysForUpdate = ['_hashed_password', '_perishable_token', '_email_verify_token', '_email_verify_token_expires_at', '_account_lockout_expires_at', '_failed_login_count', '_perishable_token_expires_at', '_password_changed_at', '_password_history']; const isSpecialUpdateKey = key => { return specialKeysForUpdate.indexOf(key) >= 0; }; function joinTableName(className, key) { return `_Join:${key}:${className}`; } const flattenUpdateOperatorsForCreate = object => { for (const key in object) { if (object[key] && object[key].__op) { switch (object[key].__op) { case 'Increment': if (typeof object[key].amount !== 'number') { throw new _node.Parse.Error(_node.Parse.Error.INVALID_JSON, 'objects to add must be an array'); } object[key] = object[key].amount; break; case 'SetOnInsert': object[key] = object[key].amount; break; case 'Add': if (!(object[key].objects instanceof Array)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_JSON, 'objects to add must be an array'); } object[key] = object[key].objects; break; case 'AddUnique': if (!(object[key].objects instanceof Array)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_JSON, 'objects to add must be an array'); } object[key] = object[key].objects; break; case 'Remove': if (!(object[key].objects instanceof Array)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_JSON, 'objects to add must be an array'); } object[key] = []; break; case 'Delete': delete object[key]; break; default: throw new _node.Parse.Error(_node.Parse.Error.COMMAND_UNAVAILABLE, `The ${object[key].__op} operator is not supported yet.`); } } } }; const transformAuthData = (className, object, schema) => { if (object.authData && className === '_User') { Object.keys(object.authData).forEach(provider => { const providerData = object.authData[provider]; const fieldName = `_auth_data_${provider}`; if (providerData == null) { object[fieldName] = { __op: 'Delete' }; } else { object[fieldName] = providerData; schema.fields[fieldName] = { type: 'Object' }; } }); delete object.authData; } }; // Transforms a Database format ACL to a REST API format ACL const untransformObjectACL = _ref2 => { let { _rperm, _wperm } = _ref2, output = _objectWithoutProperties(_ref2, _excluded2); if (_rperm || _wperm) { output.ACL = {}; (_rperm || []).forEach(entry => { if (!output.ACL[entry]) { output.ACL[entry] = { read: true }; } else { output.ACL[entry]['read'] = true; } }); (_wperm || []).forEach(entry => { if (!output.ACL[entry]) { output.ACL[entry] = { write: true }; } else { output.ACL[entry]['write'] = true; } }); } return output; }; /** * When querying, the fieldName may be compound, extract the root fieldName * `temperature.celsius` becomes `temperature` * @param {string} fieldName that may be a compound field name * @returns {string} the root name of the field */ const getRootFieldName = fieldName => { return fieldName.split('.')[0]; }; const relationSchema = { fields: { relatedId: { type: 'String' }, owningId: { type: 'String' } } }; const convertEmailToLowercase = (object, className, options) => { if (className === '_User' && options.convertEmailToLowercase) { if (typeof object['email'] === 'string') { object['email'] = object['email'].toLowerCase(); } } }; const convertUsernameToLowercase = (object, className, options) => { if (className === '_User' && options.convertUsernameToLowercase) { if (typeof object['username'] === 'string') { object['username'] = object['username'].toLowerCase(); } } }; class DatabaseController { constructor(adapter, options) { this.adapter = adapter; this.options = options || {}; this.idempotencyOptions = this.options.idempotencyOptions || {}; // Prevent mutable this.schema, otherwise one request could use // multiple schemas, so instead use loadSchema to get a schema. this.schemaPromise = null; this._transactionalSession = null; this.options = options; } collectionExists(className) { return this.adapter.classExists(className); } purgeCollection(className) { return this.loadSchema().then(schemaController => schemaController.getOneSchema(className)).then(schema => this.adapter.deleteObjectsByQuery(className, schema, {})); } validateClassName(className) { if (!SchemaController.classNameIsValid(className)) { return Promise.reject(new _node.Parse.Error(_node.Parse.Error.INVALID_CLASS_NAME, 'invalid className: ' + className)); } return Promise.resolve(); } // Returns a promise for a schemaController. loadSchema(options = { clearCache: false }) { if (this.schemaPromise != null) { return this.schemaPromise; } this.schemaPromise = SchemaController.load(this.adapter, options); this.schemaPromise.then(() => delete this.schemaPromise, () => delete this.schemaPromise); return this.loadSchema(options); } loadSchemaIfNeeded(schemaController, options = { clearCache: false }) { return schemaController ? Promise.resolve(schemaController) : this.loadSchema(options); } // Returns a promise for the classname that is related to the given // classname through the key. // TODO: make this not in the DatabaseController interface redirectClassNameForKey(className, key) { return this.loadSchema().then(schema => { var t = schema.getExpectedType(className, key); if (t != null && typeof t !== 'string' && t.type === 'Relation') { return t.targetClass; } return className; }); } // Uses the schema to validate the object (REST API format). // Returns a promise that resolves to the new schema. // This does not update this.schema, because in a situation like a // batch request, that could confuse other users of the schema. validateObject(className, object, query, runOptions, maintenance) { let schema; const acl = runOptions.acl; const isMaster = acl === undefined; var aclGroup = acl || []; return this.loadSchema().then(s => { schema = s; if (isMaster) { return Promise.resolve(); } return this.canAddField(schema, className, object, aclGroup, runOptions); }).then(() => { return schema.validateObject(className, object, query, maintenance); }); } update(className, query, update, { acl, many, upsert, addsField } = {}, skipSanitization = false, validateOnly = false, validSchemaController) { try { _Utils.default.checkProhibitedKeywords(this.options, update); } catch (error) { return Promise.reject(new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, error)); } const originalQuery = query; const originalUpdate = update; // Make a copy of the object, so we don't mutate the incoming data. update = (0, _deepcopy.default)(update); var relationUpdates = []; var isMaster = acl === undefined; var aclGroup = acl || []; return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => { return (isMaster ? Promise.resolve() : schemaController.validatePermission(className, aclGroup, 'update')).then(() => { relationUpdates = this.collectRelationUpdates(className, originalQuery.objectId, update); if (!isMaster) { query = this.addPointerPermissions(schemaController, className, 'update', query, aclGroup); if (addsField) { query = { $and: [query, this.addPointerPermissions(schemaController, className, 'addField', query, aclGroup)] }; } } if (!query) { return Promise.resolve(); } if (acl) { query = addWriteACL(query, acl); } validateQuery(query, isMaster, false, true); return schemaController.getOneSchema(className, true).catch(error => { // If the schema doesn't exist, pretend it exists with no fields. This behavior // will likely need revisiting. if (error === undefined) { return { fields: {} }; } throw error; }).then(schema => { Object.keys(update).forEach(fieldName => { if (fieldName.match(/^authData\.([a-zA-Z0-9_]+)\.id$/)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Invalid field name for update: ${fieldName}`); } const rootFieldName = getRootFieldName(fieldName); if (!SchemaController.fieldNameIsValid(rootFieldName, className) && !isSpecialUpdateKey(rootFieldName)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Invalid field name for update: ${fieldName}`); } }); for (const updateOperation in update) { if (update[updateOperation] && typeof update[updateOperation] === 'object' && Object.keys(update[updateOperation]).some(innerKey => innerKey.includes('$') || innerKey.includes('.'))) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_NESTED_KEY, "Nested keys should not contain the '$' or '.' characters"); } } update = transformObjectACL(update); convertEmailToLowercase(update, className, this.options); convertUsernameToLowercase(update, className, this.options); transformAuthData(className, update, schema); if (validateOnly) { return this.adapter.find(className, schema, query, {}).then(result => { if (!result || !result.length) { throw new _node.Parse.Error(_node.Parse.Error.OBJECT_NOT_FOUND, 'Object not found.'); } return {}; }); } if (many) { return this.adapter.updateObjectsByQuery(className, schema, query, update, this._transactionalSession); } else if (upsert) { return this.adapter.upsertOneObject(className, schema, query, update, this._transactionalSession); } else { return this.adapter.findOneAndUpdate(className, schema, query, update, this._transactionalSession); } }); }).then(result => { if (!result) { throw new _node.Parse.Error(_node.Parse.Error.OBJECT_NOT_FOUND, 'Object not found.'); } if (validateOnly) { return result; } return this.handleRelationUpdates(className, originalQuery.objectId, update, relationUpdates).then(() => { return result; }); }).then(result => { if (skipSanitization) { return Promise.resolve(result); } return this._sanitizeDatabaseResult(originalUpdate, result); }); }); } // Collect all relation-updating operations from a REST-format update. // Returns a list of all relation updates to perform // This mutates update. collectRelationUpdates(className, objectId, update) { var ops = []; var deleteMe = []; objectId = update.objectId || objectId; var process = (op, key) => { if (!op) { return; } if (op.__op == 'AddRelation') { ops.push({ key, op }); deleteMe.push(key); } if (op.__op == 'RemoveRelation') { ops.push({ key, op }); deleteMe.push(key); } if (op.__op == 'Batch') { for (var x of op.ops) { process(x, key); } } }; for (const key in update) { process(update[key], key); } for (const key of deleteMe) { delete update[key]; } return ops; } // Processes relation-updating operations from a REST-format update. // Returns a promise that resolves when all updates have been performed handleRelationUpdates(className, objectId, update, ops) { var pending = []; objectId = update.objectId || objectId; ops.forEach(({ key, op }) => { if (!op) { return; } if (op.__op == 'AddRelation') { for (const object of op.objects) { pending.push(this.addRelation(key, className, objectId, object.objectId)); } } if (op.__op == 'RemoveRelation') { for (const object of op.objects) { pending.push(this.removeRelation(key, className, objectId, object.objectId)); } } }); return Promise.all(pending); } // Adds a relation. // Returns a promise that resolves successfully iff the add was successful. addRelation(key, fromClassName, fromId, toId) { const doc = { relatedId: toId, owningId: fromId }; return this.adapter.upsertOneObject(`_Join:${key}:${fromClassName}`, relationSchema, doc, doc, this._transactionalSession); } // Removes a relation. // Returns a promise that resolves successfully iff the remove was // successful. removeRelation(key, fromClassName, fromId, toId) { var doc = { relatedId: toId, owningId: fromId }; return this.adapter.deleteObjectsByQuery(`_Join:${key}:${fromClassName}`, relationSchema, doc, this._transactionalSession).catch(error => { // We don't care if they try to delete a non-existent relation. if (error.code == _node.Parse.Error.OBJECT_NOT_FOUND) { return; } throw error; }); } // Removes objects matches this query from the database. // Returns a promise that resolves successfully iff the object was // deleted. // Options: // acl: a list of strings. If the object to be updated has an ACL, // one of the provided strings must provide the caller with // write permissions. destroy(className, query, { acl } = {}, validSchemaController) { const isMaster = acl === undefined; const aclGroup = acl || []; return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => { return (isMaster ? Promise.resolve() : schemaController.validatePermission(className, aclGroup, 'delete')).then(() => { if (!isMaster) { query = this.addPointerPermissions(schemaController, className, 'delete', query, aclGroup); if (!query) { throw new _node.Parse.Error(_node.Parse.Error.OBJECT_NOT_FOUND, 'Object not found.'); } } // delete by query if (acl) { query = addWriteACL(query, acl); } validateQuery(query, isMaster, false, false); return schemaController.getOneSchema(className).catch(error => { // If the schema doesn't exist, pretend it exists with no fields. This behavior // will likely need revisiting. if (error === undefined) { return { fields: {} }; } throw error; }).then(parseFormatSchema => this.adapter.deleteObjectsByQuery(className, parseFormatSchema, query, this._transactionalSession)).catch(error => { // When deleting sessions while changing passwords, don't throw an error if they don't have any sessions. if (className === '_Session' && error.code === _node.Parse.Error.OBJECT_NOT_FOUND) { return Promise.resolve({}); } throw error; }); }); }); } // Inserts an object into the database. // Returns a promise that resolves successfully iff the object saved. create(className, object, { acl } = {}, validateOnly = false, validSchemaController) { try { _Utils.default.checkProhibitedKeywords(this.options, object); } catch (error) { return Promise.reject(new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, error)); } // Make a copy of the object, so we don't mutate the incoming data. const originalObject = object; object = transformObjectACL(object); convertEmailToLowercase(object, className, this.options); convertUsernameToLowercase(object, className, this.options); object.createdAt = { iso: object.createdAt, __type: 'Date' }; object.updatedAt = { iso: object.updatedAt, __type: 'Date' }; var isMaster = acl === undefined; var aclGroup = acl || []; const relationUpdates = this.collectRelationUpdates(className, null, object); return this.validateClassName(className).then(() => this.loadSchemaIfNeeded(validSchemaController)).then(schemaController => { return (isMaster ? Promise.resolve() : schemaController.validatePermission(className, aclGroup, 'create')).then(() => schemaController.enforceClassExists(className)).then(() => schemaController.getOneSchema(className, true)).then(schema => { transformAuthData(className, object, schema); flattenUpdateOperatorsForCreate(object); if (validateOnly) { return {}; } return this.adapter.createObject(className, SchemaController.convertSchemaToAdapterSchema(schema), object, this._transactionalSession); }).then(result => { if (validateOnly) { return originalObject; } return this.handleRelationUpdates(className, object.objectId, object, relationUpdates).then(() => { return this._sanitizeDatabaseResult(originalObject, result.ops[0]); }); }); }); } canAddField(schema, className, object, aclGroup, runOptions) { const classSchema = schema.schemaData[className]; if (!classSchema) { return Promise.resolve(); } const fields = Object.keys(object); const schemaFields = Object.keys(classSchema.fields); const newKeys = fields.filter(field => { // Skip fields that are unset if (object[field] && object[field].__op && object[field].__op === 'Delete') { return false; } return schemaFields.indexOf(getRootFieldName(field)) < 0; }); if (newKeys.length > 0) { // adds a marker that new field is being adding during update runOptions.addsField = true; const action = runOptions.action; return schema.validatePermission(className, aclGroup, 'addField', action); } return Promise.resolve(); } // Won't delete collections in the system namespace /** * Delete all classes and clears the schema cache * * @param {boolean} fast set to true if it's ok to just delete rows and not indexes * @returns {Promise<void>} when the deletions completes */ deleteEverything(fast = false) { this.schemaPromise = null; _SchemaCache.default.clear(); return this.adapter.deleteAllClasses(fast); } // Returns a promise for a list of related ids given an owning id. // className here is the owning className. relatedIds(className, key, owningId, queryOptions) { const { skip, limit, sort } = queryOptions; const findOptions = {}; if (sort && sort.createdAt && this.adapter.canSortOnJoinTables) { findOptions.sort = { _id: sort.createdAt }; findOptions.limit = limit; findOptions.skip = skip; queryOptions.skip = 0; } return this.adapter.find(joinTableName(className, key), relationSchema, { owningId }, findOptions).then(results => results.map(result => result.relatedId)); } // Returns a promise for a list of owning ids given some related ids. // className here is the owning className. owningIds(className, key, relatedIds) { return this.adapter.find(joinTableName(className, key), relationSchema, { relatedId: { $in: relatedIds } }, { keys: ['owningId'] }).then(results => results.map(result => result.owningId)); } // Modifies query so that it no longer has $in on relation fields, or // equal-to-pointer constraints on relation fields. // Returns a promise that resolves when query is mutated reduceInRelation(className, query, schema) { // Search for an in-relation or equal-to-relation // Make it sequential for now, not sure of paralleization side effects const promises = []; if (query['$or']) { const ors = query['$or']; promises.push(...ors.map((aQuery, index) => { return this.reduceInRelation(className, aQuery, schema).then(aQuery => { query['$or'][index] = aQuery; }); })); } if (query['$and']) { const ands = query['$and']; promises.push(...ands.map((aQuery, index) => { return this.reduceInRelation(className, aQuery, schema).then(aQuery => { query['$and'][index] = aQuery; }); })); } const otherKeys = Object.keys(query).map(key => { if (key === '$and' || key === '$or') { return; } const t = schema.getExpectedType(className, key); if (!t || t.type !== 'Relation') { return Promise.resolve(query); } let queries = null; if (query[key] && (query[key]['$in'] || query[key]['$ne'] || query[key]['$nin'] || query[key].__type == 'Pointer')) { // Build the list of queries queries = Object.keys(query[key]).map(constraintKey => { let relatedIds; let isNegation = false; if (constraintKey === 'objectId') { relatedIds = [query[key].objectId]; } else if (constraintKey == '$in') { relatedIds = query[key]['$in'].map(r => r.objectId); } else if (constraintKey == '$nin') { isNegation = true; relatedIds = query[key]['$nin'].map(r => r.objectId); } else if (constraintKey == '$ne') { isNegation = true; relatedIds = [query[key]['$ne'].objectId]; } else { return; } return { isNegation, relatedIds }; }); } else { queries = [{ isNegation: false, relatedIds: [] }]; } // remove the current queryKey as we don,t need it anymore delete query[key]; // execute each query independently to build the list of // $in / $nin const promises = queries.map(q => { if (!q) { return Promise.resolve(); } return this.owningIds(className, key, q.relatedIds).then(ids => { if (q.isNegation) { this.addNotInObjectIdsIds(ids, query); } else { this.addInObjectIdsIds(ids, query); } return Promise.resolve(); }); }); return Promise.all(promises).then(() => { return Promise.resolve(); }); }); return Promise.all([...promises, ...otherKeys]).then(() => { return Promise.resolve(query); }); } // Modifies query so that it no longer has $relatedTo // Returns a promise that resolves when query is mutated reduceRelationKeys(className, query, queryOptions) { if (query['$or']) { return Promise.all(query['$or'].map(aQuery => { return this.reduceRelationKeys(className, aQuery, queryOptions); })); } if (query['$and']) { return Promise.all(query['$and'].map(aQuery => { return this.reduceRelationKeys(className, aQuery, queryOptions); })); } var relatedTo = query['$relatedTo']; if (relatedTo) { return this.relatedIds(relatedTo.object.className, relatedTo.key, relatedTo.object.objectId, queryOptions).then(ids => { delete query['$relatedTo']; this.addInObjectIdsIds(ids, query); return this.reduceRelationKeys(className, query, queryOptions); }).then(() => {}); } } addInObjectIdsIds(ids = null, query) { const idsFromString = typeof query.objectId === 'string' ? [query.objectId] : null; const idsFromEq = query.objectId && query.objectId['$eq'] ? [query.objectId['$eq']] : null; const idsFromIn = query.objectId && query.objectId['$in'] ? query.objectId['$in'] : null; // -disable-next const allIds = [idsFromString, idsFromEq, idsFromIn, ids].filter(list => list !== null); const totalLength = allIds.reduce((memo, list) => memo + list.length, 0); let idsIntersection = []; if (totalLength > 125) { idsIntersection = _intersect.default.big(allIds); } else { idsIntersection = (0, _intersect.default)(allIds); } // Need to make sure we don't clobber existing shorthand $eq constraints on objectId. if (!('objectId' in query)) { query.objectId = { $in: undefined }; } else if (typeof query.objectId === 'string') { query.objectId = { $in: undefined, $eq: query.objectId }; } query.objectId['$in'] = idsIntersection; return query; } addNotInObjectIdsIds(ids = [], query) { const idsFromNin = query.objectId && query.objectId['$nin'] ? query.objectId['$nin'] : []; let allIds = [...idsFromNin, ...ids].filter(list => list !== null); // make a set and spread to remove duplicates allIds = [...new Set(allIds)]; // Need to make sure we don't clobber existing shorthand $eq constraints on objectId. if (!('objectId' in query)) { query.objectId = { $nin: undefined }; } else if (typeof query.objectId === 'string') { query.objectId = { $nin: undefined, $eq: query.objectId }; } query.objectId['$nin'] = allIds; return query; } // Runs a query on the database. // Returns a promise that resolves to a list of items. // Options: // skip number of results to skip. // limit limit to this number of results. // sort an object where keys are the fields to sort by. // the value is +1 for ascending, -1 for descending. // count run a count instead of returning results. // acl restrict this operation with an ACL for the provided array // of user objectIds and roles. acl: null means no user. // when this field is not present, don't do anything regarding ACLs. // caseInsensitive make string comparisons case insensitive // TODO: make userIds not needed here. The db adapter shouldn't know // anything about users, ideally. Then, improve the format of the ACL // arg to work like the others. find(className, query, { skip, limit, acl, sort = {}, count, keys, op, distinct, pipeline, readPreference, hint, caseInsensitive = false, explain, comment } = {}, auth = {}, validSchemaController) { const isMaintenance = auth.isMaintenance; const isMaster = acl === undefined || isMaintenance; const aclGroup = acl || []; op = op || (typeof query.objectId == 'string' && Object.keys(query).length === 1 ? 'get' : 'find'); // Count operation if counting op = count === true ? 'count' : op; let classExists = true; return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => { //Allow volatile classes if querying with Master (for _PushStatus) //TODO: Move volatile classes concept into mongo adapter, postgres adapter shouldn't care //that api.parse.com breaks when _PushStatus exists in mongo. return schemaController.getOneSchema(className, isMaster).catch(error => { // Behavior for non-existent classes is kinda weird on Parse.com. Probably doesn't matter too much. // For now, pretend the class exists but has no objects, if (error === undefined) { classExists = false; return { fields: {} }; } throw error; }).then(schema => { // Parse.com treats queries on _created_at and _updated_at as if they were queries on createdAt and updatedAt, // so duplicate that behavior here. If both are specified, the correct behavior to match Parse.com is to // use the one that appears first in the sort list. if (sort._created_at) { sort.createdAt = sort._created_at; delete sort._created_at; } if (sort._updated_at) { sort.updatedAt = sort._updated_at; delete sort._updated_at; } const queryOptions = { skip, limit, sort, keys, readPreference, hint, caseInsensitive: this.options.enableCollationCaseComparison ? false : caseInsensitive, explain, comment }; Object.keys(sort).forEach(fieldName => { if (fieldName.match(/^authData\.([a-zA-Z0-9_]+)\.id$/)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Cannot sort by ${fieldName}`); } const rootFieldName = getRootFieldName(fieldName); if (!SchemaController.fieldNameIsValid(rootFieldName, className)) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`); } if (!schema.fields[fieldName.split('.')[0]] && fieldName !== 'score') { delete sort[fieldName]; } }); return (isMaster ? Promise.resolve() : schemaController.validatePermission(className, aclGroup, op)).then(() => this.reduceRelationKeys(className, query, queryOptions)).then(() => this.reduceInRelation(className, query, schemaController)).then(() => { let protectedFields; if (!isMaster) { query = this.addPointerPermissions(schemaController, className, op, query, aclGroup); /* Don't use projections to optimize the protectedFields since the protectedFields based on pointer-permissions are determined after querying. The filtering can overwrite the protected fields. */ protectedFields = this.addProtectedFields(schemaController, className, query, aclGroup, auth, queryOptions); } if (!query) { if (op === 'get') { throw new _node.Parse.Error(_node.Parse.Error.OBJECT_NOT_FOUND, 'Object not found.'); } else { return []; } } if (!isMaster) { if (op === 'update' || op === 'delete') { query = addWriteACL(query, aclGroup); } else { query = addReadACL(query, aclGroup); } } validateQuery(query, isMaster, isMaintenance, false); if (count) { if (!classExists) { return 0; } else { return this.adapter.count(className, schema, query, readPreference, undefined, hint, comment); } } else if (distinct) { if (!classExists) { return []; } else { return this.adapter.distinct(className, schema, query, distinct); } } else if (pipeline) { if (!classExists) { return []; } else { return this.adapter.aggregate(className, schema, pipeline, readPreference, hint, explain, comment); } } else if (explain) { return this.adapter.find(className, schema, query, queryOptions); } else { return this.adapter.find(className, schema, query, queryOptions).then(objects => objects.map(object => { object = untransformObjectACL(object); return filterSensitiveData(isMaster, isMaintenance, aclGroup, auth, op, schemaController, className, protectedFields, object); })).catch(error => { throw new _node.Parse.Error(_node.Parse.Error.INTERNAL_SERVER_ERROR, error); }); } }); }); }); } deleteSchema(className) { let schemaController; return this.loadSchema({ clearCache: true }).then(s => { schemaController = s; return schemaController.getOneSchema(className, true); }).catch(error => { if (error === undefined) { return { fields: {} }; } else { throw error; } }).then(schema => { return this.collectionExists(className).then(() => this.adapter.count(className, { fields: {} }, null, '', false)).then(count => { if (count > 0) { throw new _node.Parse.Error(255, `Class ${className} is not empty, contains ${count} objects, cannot drop schema.`); } return this.adapter.deleteClass(className); }).then(wasParseCollection => { if (wasParseCollection) { const relationFieldNames = Object.keys(schema.fields).filter(fieldName => schema.fields[fieldName].type === 'Relation'); return Promise.all(relationFieldNames.map(name => this.adapter.deleteClass(joinTableName(className, name)))).then(() => { _SchemaCache.default.del(className); return schemaController.reloadData(); }); } else { return Promise.resolve(); } }); }); } // This helps to create intermediate objects for simpler comparison of // key value pairs used in query objects. Each key value pair will represented // in a similar way to json objectToEntriesStrings(query) { return Object.entries(query).map(a => a.map(s => JSON.stringify(s)).join(':')); } // Naive logic reducer for OR operations meant to be used only for pointer permissions. reduceOrOperation(query) { if (!query.$or) { return query; } const queries = query.$or.map(q => this.objectToEntriesStrings(q)); let repeat = false; do { repeat = false; for (let i = 0; i < queries.length - 1; i++) { for (let j = i + 1; j < queries.length; j++) { const [shorter, longer] = queries[i].length > queries[j].length ? [j, i] : [i, j]; const foundEntries = queries[shorter].reduce((acc, entry) => acc + (queries[longer].includes(entry) ? 1 : 0), 0); const shorterEntries = queries[shorter].length; if (foundEntries === shorterEntries) { // If the shorter query is completely contained in the longer one, we can strike // out the longer query. query.$or.splice(longer, 1); queries.splice(longer, 1); repeat = true; break; } } } } while (repeat); if (query.$or.length === 1) { query = _objectSpread(_objectSpread({}, query), query.$or[0]); delete query.$or; } return query; } // Naive logic reducer for AND operations meant to be used only for pointer permissions. reduceAndOperation(query) { if (!query.$and) { return query; } const queries = query.$and.map(q => this.objectToEntriesStrings(q)); let repeat = false; do { repeat = false; for (let i = 0; i < queries.length - 1; i++) { for (let j = i + 1; j < queries.length; j++) { const [shorter, longer] = queries[i].length > queries[j].length ? [j, i] : [i, j]; const foundEntries = queries[shorter].reduce((acc, entry) => acc + (queries[longer].includes(entry) ? 1 : 0), 0); const shorterEntries = queries[shorter].length; if (foundEntries === shorterEntries) { // If the shorter query is completely contained in the longer one, we can strike // out the shorter query. query.$and.splice(shorter, 1); queries.splice(shorter, 1); repeat = true; break; } } } } while (repeat); if (query.$and.length === 1) { query = _objectSpread(_objectSpread({}, query), query.$and[0]); delete query.$and; } return query; } // Constraints query using CLP's pointer permissions (PP) if any. // 1. Etract the user id from caller's ACLgroup; // 2. Exctract a list of field names that are PP for target collection and operation; // 3. Constraint the original query so that each PP field must // point to caller's id (or contain it in case of PP field being an array) addPointerPermissions(schema, className, operation, query, aclGroup = []) { // Check if class has public permission for operation // If the BaseCLP pass, let go through if (schema.testPermissionsForClassName(className, aclGroup, operation)) { return query; } const perms = schema.getClassLevelPermissions(className); const userACL = aclGroup.filter(acl => { return acl.indexOf('role:') != 0 && acl != '*'; }); const groupKey = ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields'; const permFields = []; if (perms[operation] && perms[operation].pointerFields) { permFields.push(...perms[operation].pointerFields); } if (perms[groupKey]) { for (const field of perms[groupKey]) { if (!permFields.includes(field)) { permFields.push(field); } } } // the ACL should have exactly 1 user if (permFields.length > 0) { // the ACL should have exactly 1 user // No user set return undefined // If the length is > 1, that means we didn't de-dupe users correctly if (userACL.length != 1) { return; } const userId = userACL[0]; const userPointer = { __type: 'Pointer', className: '_User', objectId: userId }; const queries = permFields.map(key => { const fieldDescriptor = schema.getExpectedType(className, key); const fieldType = fieldDescriptor && typeof fieldDescriptor === 'object' && Object.prototype.hasOwnProperty.call(fieldDescriptor, 'type') ? fieldDescriptor.type : null; let queryClause; if (fieldType === 'Pointer') { // constraint for single pointer setup queryClause = { [key]: userPointer }; } else if (fieldType === 'Array') { // constraint for users-array setup queryClause = { [key]: { $all: [userPointer] } }; } else if (fieldType === 'Object') { // constraint for object setup queryClause = { [key]: userPointer }; } else { // This means that there is a CLP field of an unexpected type. This condition should not happen, which is // why is being treated as an error. throw Error(`An unexpected condition occurred when resolving pointer permissions: ${className} ${key}`); } // if we already have a constraint on the key, use the $and if (Object.prototype.hasOwnProperty.call(query, key)) { return this.reduceAndOperation({ $and: [queryClause, query] }); } // otherwise just add the constaint return Object.assign({}, query, queryClause); }); return queries.length === 1 ? queries[0] : this.reduceOrOperation({ $or: queries }); } else { return query; } } addProtectedFields(schema, className, query = {}, aclGroup = [], auth = {}, queryOptions = {}) { const perms = schema && schema.getClassLevelPermissions ? schema.getClassLevelPermissions(className) : schema; if (!perms) return null; const protectedFields = perms.protectedFields; if (!protectedFields) return null; if (aclGroup.indexOf(query.objectId) > -1) return null; // for queries where "keys" are set and do not include all 'userField':{field}, // we have to transparently include it, and then remove before returning to client // Because if such key not projected the permission won't be enforced properly // PS this is called when 'excludeKeys' already reduced to 'keys' const preserveKeys = queryOptions.keys; // these are keys that need to be included only // to be able to apply protectedFields by pointer // and then unset before returning to client (later in filterSensitiveFields) const serverOnlyKeys = []; const authenticated = auth.user; // map to allow check without array search const roles = (auth.userRoles || []).reduce((acc, r) => { acc[r] = protectedFields[r]; return acc; }, {}); // array of sets of protected fields. separate item for each applicable criteria const protectedKeysSets = []; for (const key in protectedFields) { // skip userFields if (key.startsWith('userField:')) { if (preserveKeys) { const fieldName = key.substring(10); if (!preserveKeys.includes(fieldName)) { // 1. put it there temporarily queryOptions.keys && queryOptions.keys.push(fieldName); // 2. preserve it delete later serverOnlyKeys.push(fieldName); } } continue; } // add public tier if (key === '*') { protectedKeysSets.push(protectedFields[key]); continue; } if (authenticated) { if (key === 'authenticated') { // for logged in users protectedKeysSets.push(protectedFields[key]); continue; } if (roles[key] && key.startsWith('role:')) { // add applicable roles protectedKeysSets.push(roles[key]); } } } // check if there's a rule for current user's id if (authenticated) { const userId = auth.user.id; if (perms.protectedFields[userId]) { protectedKeysSets.push(perms.protectedFields[userId]); } } // preserve fields to be removed before sending response to client if (serverOnlyKeys.length > 0) { perms.protectedFields.temporaryKeys = serverOnlyKeys; } let protectedKeys = protectedKeysSets.reduce((acc, next) => { if (next) { acc.push(...next); } return acc; }, []); // intersect all sets of protectedFields protectedKeysSets.forEach(fields => { if (fields) { protectedKeys = protectedKeys.filter(v => fields.includes(v)); } }); return protectedKeys; } createTransactionalSession() { return this.adapter.createTransactionalSession().then(transactionalSession => { this._transactionalSession = transactionalSession; }); } commitTransactionalSession() { if (!this._transactionalSession) { throw new Error('There is no transactional session to commit'); } return this.adapter.commitTransactionalSession(this._transactionalSession).then(() => { this._transactionalSession = null; }); } abortTransactionalSession() { if (!this._transactionalSession) { throw new Error('There is no transactional session to abort'); } return this.adapter.abortTransactionalSession(this._transactionalSession).then(() => { this._transactionalSession = null; }); } // TODO: create indexes on first creation of a _User object. Otherwise it's impossible to // have a Parse app without it having a _User collection. async performInitialization() { await this.adapter.performInitialization({ VolatileClassesSchemas: SchemaController.VolatileClassesSchemas }); const requiredUserFields = { fields: _objectSpread(_objectSpread({}, SchemaController.defaultColumns._Default), SchemaController.defaultColumns._User) }; const requiredRoleFields = { fields: _objectSpread(_objectSpread({}, SchemaController.defaultColumns._Default), SchemaController.defaultColumns._Role) }; const requiredIdempotencyFields = { fields: _objectSpread(_objectSpread({}, SchemaController.defaultColumns._Default), SchemaController.defaultColumns._Idempotency) }; await this.loadSchema().then(schema => schema.enforceClassExists('_User')); await this.loadSchema().then(schema => schema.enforceClassExists('_Role')); await this.loadSchema().then(schema => schema.enforceClassExists('_Idempotency')); await this.adapter.ensureUniqueness('_User', requiredUserFields, ['username']).catch(error => { _logger.default.warn('Unable to ensure uniqueness for usernames: ', error); throw error; }); if (!this.options.enableCollationCaseComparison) { await this.adapter.ensureIndex('_User', requiredUserFields, ['username'], 'case_insensitive_username', true).catch(error => { _logger.default.warn('Unable to create case insensitive username index: ', error); throw error; }); await this.adapter.ensureIndex('_User', requiredUserFields, ['email'], 'case_insensitive_email', true).catch(error => { _logger.default.warn('Unable to create case insensitive email index: ', error); throw error; }); } await this.adapter.ensureUniqueness('_User', requiredUserFields, ['email']).catch(error => { _logger.default.warn('Unable to ensure uniqueness for user email addresses: ', error); throw error; }); await this.adapter.ensureUniqueness('_Role', requiredRoleFields, ['name']).catch(error => { _logger.default.warn('Unable to ensure uniqueness for role name: ', error); throw error; }); await this.adapter.ensureUniqueness('_Idempotency', requiredIdempotencyFields, ['reqId']).catch(error => { _logger.default.warn('Unable to ensure uniqueness for idempotency request ID: ', error); throw error; }); const isMongoAdapter = this.adapter instanceof _MongoStorageAdapter.default; const isPostgresAdapter = this.adapter instanceof _PostgresStorageAdapter.default; if (isMongoAdapter || isPostgresAdapter) { let options = {}; if (isMongoAdapter) { options = { ttl: 0 }; } else if (isPostgresAdapter) { options = this.idempotencyOptions; options.setIdempotencyFunction = true; } await this.adapter.ensureIndex('_Idempotency', requiredIdempotencyFields, ['expire'], 'ttl', false, options).catch(error => { _logger.default.warn('Unable to create TTL index for idempotency expire date: ', error); throw error; }); } await this.adapter.updateSchemaWithIndexes(); } _expandResultOnKeyPath(object, key, value) { if (key.indexOf('.') < 0) { object[key] = value[key]; return object; } const path = key.split('.'); const firstKey = path[0]; const nextPath = path.slice(1).join('.'); // Scan request data for denied keywords if (this.options && this.options.requestKeywordDenylist) { // Scan request data for denied keywords for (const keyword of this.options.requestKeywordDenylist) { const match = _Utils.default.objectContainsKeyValue({ [firstKey]: true, [nextPath]: true }, keyword.key, true); if (match) { throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`); } } } object[firstKey] = this._expandResultOnKeyPath(object[firstKey] || {}, nextPath, value[firstKey]); delete object[key]; return object; } _sanitizeDatabaseResult(originalObject, result) { const response = {}; if (!result) { return Promise.resolve(response); } Object.keys(originalObject).forEach(key => { const keyUpdate = originalObject[key]; // determine if that was an op if (keyUpdate && typeof keyUpdate === 'object' && keyUpdate.__op && ['Add', 'AddUnique', 'Remove', 'Increment', 'SetOnInsert'].indexOf(keyUpdate.__op) > -1) { // only valid ops that produce an actionable result // the op may have happened on a keypath this._expandResultOnKeyPath(response, key, result); // Revert array to object conversion on dot notation for arrays (e.g. "field.0.key") if (key.includes('.')) { const [field, index] = key.split('.'); const isArrayIndex = Array.from(index).every(c => c >= '0' && c <= '9'); if (isArrayIndex && Array.isArray(result[field]) && !Array.isArray(response[field])) { response[field] = result[field]; } } } }); return Promise.resolve(response); } } module.exports = DatabaseController; // Expose validateQuery for tests module.exports._validateQuery = validateQuery; module.exports.filterSensitiveData = filterSensitiveData; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","require","_lodash","_interopRequireDefault","_intersect","_deepcopy","_logger","_Utils","SchemaController","_interopRequireWildcard","_StorageAdapter","_MongoStorageAdapter","_PostgresStorageAdapter","_SchemaCache","_excluded","_excluded2","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty","getOwnPropertyDescriptors","defineProperties","_toPropertyKey","value","configurable","writable","_toPrimitive","Symbol","toPrimitive","TypeError","String","Number","_objectWithoutProperties","_objectWithoutPropertiesLoose","indexOf","propertyIsEnumerable","addWriteACL","query","acl","newQuery","_","cloneDeep","_wperm","$in","addReadACL","_rperm","transformObjectACL","_ref","ACL","result","entry","read","write","specialQueryKeys","specialMasterQueryKeys","validateQuery","isMaster","isMaintenance","update","Parse","Error","INVALID_QUERY","$or","Array","$and","$nor","key","$regex","$options","match","includes","INVALID_KEY_NAME","filterSensitiveData","aclGroup","auth","operation","schema","className","protectedFields","object","userId","user","id","perms","getClassLevelPermissions","isReadOperation","protectedFieldsPointerPerm","startsWith","map","substring","newProtectedFields","overrideProtectedFields","pointerPerm","pointerPermIncludesUser","readUserFieldValue","isArray","some","objectId","fields","v","isUserClass","password","_hashed_password","sessionToken","_perms$protectedField","k","temporaryKeys","charAt","authData","specialKeysForUpdate","isSpecialUpdateKey","joinTableName","flattenUpdateOperatorsForCreate","__op","amount","INVALID_JSON","objects","COMMAND_UNAVAILABLE","transformAuthData","provider","providerData","fieldName","type","untransformObjectACL","_ref2","output","getRootFieldName","split","relationSchema","relatedId","owningId","convertEmailToLowercase","options","toLowerCase","convertUsernameToLowercase","DatabaseController","constructor","adapter","idempotencyOptions","schemaPromise","_transactionalSession","collectionExists","classExists","purgeCollection","loadSchema","then","schemaController","getOneSchema","deleteObjectsByQuery","validateClassName","classNameIsValid","Promise","reject","INVALID_CLASS_NAME","resolve","clearCache","load","loadSchemaIfNeeded","redirectClassNameForKey","getExpectedType","targetClass","validateObject","runOptions","maintenance","undefined","s","canAddField","many","upsert","addsField","skipSanitization","validateOnly","validSchemaController","Utils","checkProhibitedKeywords","error","originalQuery","originalUpdate","deepcopy","relationUpdates","validatePermission","collectRelationUpdates","addPointerPermissions","catch","rootFieldName","fieldNameIsValid","updateOperation","innerKey","INVALID_NESTED_KEY","find","OBJECT_NOT_FOUND","updateObjectsByQuery","upsertOneObject","findOneAndUpdate","handleRelationUpdates","_sanitizeDatabaseResult","ops","deleteMe","process","op","x","pending","addRelation","removeRelation","all","fromClassName","fromId","toId","doc","code","destroy","parseFormatSchema","create","originalObject","createdAt","iso","__type","updatedAt","enforceClassExists","createObject","convertSchemaToAdapterSchema","classSchema","schemaData","schemaFields","newKeys","field","action","deleteEverything","fast","SchemaCache","clear","deleteAllClasses","relatedIds","queryOptions","skip","limit","sort","findOptions","canSortOnJoinTables","_id","results","owningIds","reduceInRelation","promises","ors","aQuery","index","ands","otherKeys","queries","constraintKey","isNegation","q","ids","addNotInObjectIdsIds","addInObjectIdsIds","reduceRelationKeys","relatedTo","idsFromString","idsFromEq","idsFromIn","allIds","list","totalLength","reduce","memo","idsIntersection","intersect","big","$eq","idsFromNin","Set","$nin","count","distinct","pipeline","readPreference","hint","caseInsensitive","explain","comment","_created_at","_updated_at","enableCollationCaseComparison","addProtectedFields","aggregate","INTERNAL_SERVER_ERROR","deleteSchema","deleteClass","wasParseCollection","relationFieldNames","name","del","reloadData","objectToEntriesStrings","entries","JSON","stringify","join","reduceOrOperation","repeat","j","shorter","longer","foundEntries","acc","shorterEntries","splice","reduceAndOperation","testPermissionsForClassName","userACL","groupKey","permFields","pointerFields","userPointer","fieldDescriptor","fieldType","prototype","queryClause","$all","assign","preserveKeys","serverOnlyKeys","authenticated","roles","userRoles","protectedKeysSets","protectedKeys","next","createTransactionalSession","transactionalSession","commitTransactionalSession","abortTransactionalSession","performInitialization","VolatileClassesSchemas","requiredUserFields","defaultColumns","_Default","_User","requiredRoleFields","_Role","requiredIdempotencyFields","_Idempotency","ensureUniqueness","logger","warn","ensureIndex","isMongoAdapter","MongoStorageAdapter","isPostgresAdapter","PostgresStorageAdapter","ttl","setIdempotencyFunction","updateSchemaWithIndexes","_expandResultOnKeyPath","path","firstKey","nextPath","slice","requestKeywordDenylist","keyword","objectContainsKeyValue","response","keyUpdate","isArrayIndex","from","every","c","module","exports","_validateQuery"],"sources":["../../src/Controllers/DatabaseController.js"],"sourcesContent":["﻿// @flow\n// A database adapter that works with data exported from the hosted\n// Parse database.\n\n// @flow-disable-next\nimport { Parse } from 'parse/node';\n// @flow-disable-next\nimport _ from 'lodash';\n// @flow-disable-next\nimport intersect from 'intersect';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport logger from '../logger';\nimport Utils from '../Utils';\nimport * as SchemaController from './SchemaController';\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport MongoStorageAdapter from '../Adapters/Storage/Mongo/MongoStorageAdapter';\nimport PostgresStorageAdapter from '../Adapters/Storage/Postgres/PostgresStorageAdapter';\nimport SchemaCache from '../Adapters/Cache/SchemaCache';\nimport type { LoadSchemaOptions } from './types';\nimport type { ParseServerOptions } from '../Options';\nimport type { QueryOptions, FullQueryOptions } from '../Adapters/Storage/StorageAdapter';\n\nfunction addWriteACL(query, acl) {\n  const newQuery = _.cloneDeep(query);\n  //Can't be any existing '_wperm' query, we don't allow client queries on that, no need to $and\n  newQuery._wperm = { $in: [null, ...acl] };\n  return newQuery;\n}\n\nfunction addReadACL(query, acl) {\n  const newQuery = _.cloneDeep(query);\n  //Can't be any existing '_rperm' query, we don't allow client queries on that, no need to $and\n  newQuery._rperm = { $in: [null, '*', ...acl] };\n  return newQuery;\n}\n\n// Transforms a REST API formatted ACL object to our two-field mongo format.\nconst transformObjectACL = ({ ACL, ...result }) => {\n  if (!ACL) {\n    return result;\n  }\n\n  result._wperm = [];\n  result._rperm = [];\n\n  for (const entry in ACL) {\n    if (ACL[entry].read) {\n      result._rperm.push(entry);\n    }\n    if (ACL[entry].write) {\n      result._wperm.push(entry);\n    }\n  }\n  return result;\n};\n\nconst specialQueryKeys = ['$and', '$or', '$nor', '_rperm', '_wperm'];\nconst specialMasterQueryKeys = [\n  ...specialQueryKeys,\n  '_email_verify_token',\n  '_perishable_token',\n  '_tombstone',\n  '_email_verify_token_expires_at',\n  '_failed_login_count',\n  '_account_lockout_expires_at',\n  '_password_changed_at',\n  '_password_history',\n];\n\nconst validateQuery = (\n  query: any,\n  isMaster: boolean,\n  isMaintenance: boolean,\n  update: boolean\n): void => {\n  if (isMaintenance) {\n    isMaster = true;\n  }\n  if (query.ACL) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Cannot query on ACL.');\n  }\n\n  if (query.$or) {\n    if (query.$or instanceof Array) {\n      query.$or.forEach(value => validateQuery(value, isMaster, isMaintenance, update));\n    } else {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Bad $or format - use an array value.');\n    }\n  }\n\n  if (query.$and) {\n    if (query.$and instanceof Array) {\n      query.$and.forEach(value => validateQuery(value, isMaster, isMaintenance, update));\n    } else {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Bad $and format - use an array value.');\n    }\n  }\n\n  if (query.$nor) {\n    if (query.$nor instanceof Array && query.$nor.length > 0) {\n      query.$nor.forEach(value => validateQuery(value, isMaster, isMaintenance, update));\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_QUERY,\n        'Bad $nor format - use an array of at least 1 value.'\n      );\n    }\n  }\n\n  Object.keys(query).forEach(key => {\n    if (query && query[key] && query[key].$regex) {\n      if (typeof query[key].$options === 'string') {\n        if (!query[key].$options.match(/^[imxs]+$/)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_QUERY,\n            `Bad $options value for query: ${query[key].$options}`\n          );\n        }\n      }\n    }\n    if (\n      !key.match(/^[a-zA-Z][a-zA-Z0-9_\\.]*$/) &&\n      ((!specialQueryKeys.includes(key) && !isMaster && !update) ||\n        (update && isMaster && !specialMasterQueryKeys.includes(key)))\n    ) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid key name: ${key}`);\n    }\n  });\n};\n\n// Filters out any data that shouldn't be on this REST-formatted object.\nconst filterSensitiveData = (\n  isMaster: boolean,\n  isMaintenance: boolean,\n  aclGroup: any[],\n  auth: any,\n  operation: any,\n  schema: SchemaController.SchemaController | any,\n  className: string,\n  protectedFields: null | Array<any>,\n  object: any\n) => {\n  let userId = null;\n  if (auth && auth.user) userId = auth.user.id;\n\n  // replace protectedFields when using pointer-permissions\n  const perms =\n    schema && schema.getClassLevelPermissions ? schema.getClassLevelPermissions(className) : {};\n  if (perms) {\n    const isReadOperation = ['get', 'find'].indexOf(operation) > -1;\n\n    if (isReadOperation && perms.protectedFields) {\n      // extract protectedFields added with the pointer-permission prefix\n      const protectedFieldsPointerPerm = Object.keys(perms.protectedFields)\n        .filter(key => key.startsWith('userField:'))\n        .map(key => {\n          return { key: key.substring(10), value: perms.protectedFields[key] };\n        });\n\n      const newProtectedFields: Array<string>[] = [];\n      let overrideProtectedFields = false;\n\n      // check if the object grants the current user access based on the extracted fields\n      protectedFieldsPointerPerm.forEach(pointerPerm => {\n        let pointerPermIncludesUser = false;\n        const readUserFieldValue = object[pointerPerm.key];\n        if (readUserFieldValue) {\n          if (Array.isArray(readUserFieldValue)) {\n            pointerPermIncludesUser = readUserFieldValue.some(\n              user => user.objectId && user.objectId === userId\n            );\n          } else {\n            pointerPermIncludesUser =\n              readUserFieldValue.objectId && readUserFieldValue.objectId === userId;\n          }\n        }\n\n        if (pointerPermIncludesUser) {\n          overrideProtectedFields = true;\n          newProtectedFields.push(pointerPerm.value);\n        }\n      });\n\n      // if at least one pointer-permission affected the current user\n      // intersect vs protectedFields from previous stage (@see addProtectedFields)\n      // Sets theory (intersections): A x (B x C) == (A x B) x C\n      if (overrideProtectedFields && protectedFields) {\n        newProtectedFields.push(protectedFields);\n      }\n      // intersect all sets of protectedFields\n      newProtectedFields.forEach(fields => {\n        if (fields) {\n          // if there're no protctedFields by other criteria ( id / role / auth)\n          // then we must intersect each set (per userField)\n          if (!protectedFields) {\n            protectedFields = fields;\n          } else {\n            protectedFields = protectedFields.filter(v => fields.includes(v));\n          }\n        }\n      });\n    }\n  }\n\n  const isUserClass = className === '_User';\n  if (isUserClass) {\n    object.password = object._hashed_password;\n    delete object._hashed_password;\n    delete object.sessionToken;\n  }\n\n  if (isMaintenance) {\n    return object;\n  }\n\n  /* special treat for the user class: don't filter protectedFields if currently loggedin user is\n  the retrieved user */\n  if (!(isUserClass && userId && object.objectId === userId)) {\n    protectedFields && protectedFields.forEach(k => delete object[k]);\n\n    // fields not requested by client (excluded),\n    // but were needed to apply protectedFields\n    perms?.protectedFields?.temporaryKeys?.forEach(k => delete object[k]);\n  }\n\n  for (const key in object) {\n    if (key.charAt(0) === '_') {\n      delete object[key];\n    }\n  }\n\n  if (!isUserClass || isMaster) {\n    return object;\n  }\n\n  if (aclGroup.indexOf(object.objectId) > -1) {\n    return object;\n  }\n  delete object.authData;\n  return object;\n};\n\n// Runs an update on the database.\n// Returns a promise for an object with the new values for field\n// modifications that don't know their results ahead of time, like\n// 'increment'.\n// Options:\n//   acl:  a list of strings. If the object to be updated has an ACL,\n//         one of the provided strings must provide the caller with\n//         write permissions.\nconst specialKeysForUpdate = [\n  '_hashed_password',\n  '_perishable_token',\n  '_email_verify_token',\n  '_email_verify_token_expires_at',\n  '_account_lockout_expires_at',\n  '_failed_login_count',\n  '_perishable_token_expires_at',\n  '_password_changed_at',\n  '_password_history',\n];\n\nconst isSpecialUpdateKey = key => {\n  return specialKeysForUpdate.indexOf(key) >= 0;\n};\n\nfunction joinTableName(className, key) {\n  return `_Join:${key}:${className}`;\n}\n\nconst flattenUpdateOperatorsForCreate = object => {\n  for (const key in object) {\n    if (object[key] && object[key].__op) {\n      switch (object[key].__op) {\n        case 'Increment':\n          if (typeof object[key].amount !== 'number') {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].amount;\n          break;\n        case 'SetOnInsert':\n          object[key] = object[key].amount;\n          break;\n        case 'Add':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].objects;\n          break;\n        case 'AddUnique':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].objects;\n          break;\n        case 'Remove':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = [];\n          break;\n        case 'Delete':\n          delete object[key];\n          break;\n        default:\n          throw new Parse.Error(\n            Parse.Error.COMMAND_UNAVAILABLE,\n            `The ${object[key].__op} operator is not supported yet.`\n          );\n      }\n    }\n  }\n};\n\nconst transformAuthData = (className, object, schema) => {\n  if (object.authData && className === '_User') {\n    Object.keys(object.authData).forEach(provider => {\n      const providerData = object.authData[provider];\n      const fieldName = `_auth_data_${provider}`;\n      if (providerData == null) {\n        object[fieldName] = {\n          __op: 'Delete',\n        };\n      } else {\n        object[fieldName] = providerData;\n        schema.fields[fieldName] = { type: 'Object' };\n      }\n    });\n    delete object.authData;\n  }\n};\n// Transforms a Database format ACL to a REST API format ACL\nconst untransformObjectACL = ({ _rperm, _wperm, ...output }) => {\n  if (_rperm || _wperm) {\n    output.ACL = {};\n\n    (_rperm || []).forEach(entry => {\n      if (!output.ACL[entry]) {\n        output.ACL[entry] = { read: true };\n      } else {\n        output.ACL[entry]['read'] = true;\n      }\n    });\n\n    (_wperm || []).forEach(entry => {\n      if (!output.ACL[entry]) {\n        output.ACL[entry] = { write: true };\n      } else {\n        output.ACL[entry]['write'] = true;\n      }\n    });\n  }\n  return output;\n};\n\n/**\n * When querying, the fieldName may be compound, extract the root fieldName\n *     `temperature.celsius` becomes `temperature`\n * @param {string} fieldName that may be a compound field name\n * @returns {string} the root name of the field\n */\nconst getRootFieldName = (fieldName: string): string => {\n  return fieldName.split('.')[0];\n};\n\nconst relationSchema = {\n  fields: { relatedId: { type: 'String' }, owningId: { type: 'String' } },\n};\n\nconst convertEmailToLowercase = (object, className, options) => {\n  if (className === '_User' && options.convertEmailToLowercase) {\n    if (typeof object['email'] === 'string') {\n      object['email'] = object['email'].toLowerCase();\n    }\n  }\n};\n\nconst convertUsernameToLowercase = (object, className, options) => {\n  if (className === '_User' && options.convertUsernameToLowercase) {\n    if (typeof object['username'] === 'string') {\n      object['username'] = object['username'].toLowerCase();\n    }\n  }\n};\n\nclass DatabaseController {\n  adapter: StorageAdapter;\n  schemaCache: any;\n  schemaPromise: ?Promise<SchemaController.SchemaController>;\n  _transactionalSession: ?any;\n  options: ParseServerOptions;\n  idempotencyOptions: any;\n\n  constructor(adapter: StorageAdapter, options: ParseServerOptions) {\n    this.adapter = adapter;\n    this.options = options || {};\n    this.idempotencyOptions = this.options.idempotencyOptions || {};\n    // Prevent mutable this.schema, otherwise one request could use\n    // multiple schemas, so instead use loadSchema to get a schema.\n    this.schemaPromise = null;\n    this._transactionalSession = null;\n    this.options = options;\n  }\n\n  collectionExists(className: string): Promise<boolean> {\n    return this.adapter.classExists(className);\n  }\n\n  purgeCollection(className: string): Promise<void> {\n    return this.loadSchema()\n      .then(schemaController => schemaController.getOneSchema(className))\n      .then(schema => this.adapter.deleteObjectsByQuery(className, schema, {}));\n  }\n\n  validateClassName(className: string): Promise<void> {\n    if (!SchemaController.classNameIsValid(className)) {\n      return Promise.reject(\n        new Parse.Error(Parse.Error.INVALID_CLASS_NAME, 'invalid className: ' + className)\n      );\n    }\n    return Promise.resolve();\n  }\n\n  // Returns a promise for a schemaController.\n  loadSchema(\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<SchemaController.SchemaController> {\n    if (this.schemaPromise != null) {\n      return this.schemaPromise;\n    }\n    this.schemaPromise = SchemaController.load(this.adapter, options);\n    this.schemaPromise.then(\n      () => delete this.schemaPromise,\n      () => delete this.schemaPromise\n    );\n    return this.loadSchema(options);\n  }\n\n  loadSchemaIfNeeded(\n    schemaController: SchemaController.SchemaController,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<SchemaController.SchemaController> {\n    return schemaController ? Promise.resolve(schemaController) : this.loadSchema(options);\n  }\n\n  // Returns a promise for the classname that is related to the given\n  // classname through the key.\n  // TODO: make this not in the DatabaseController interface\n  redirectClassNameForKey(className: string, key: string): Promise<?string> {\n    return this.loadSchema().then(schema => {\n      var t = schema.getExpectedType(className, key);\n      if (t != null && typeof t !== 'string' && t.type === 'Relation') {\n        return t.targetClass;\n      }\n      return className;\n    });\n  }\n\n  // Uses the schema to validate the object (REST API format).\n  // Returns a promise that resolves to the new schema.\n  // This does not update this.schema, because in a situation like a\n  // batch request, that could confuse other users of the schema.\n  validateObject(\n    className: string,\n    object: any,\n    query: any,\n    runOptions: QueryOptions,\n    maintenance: boolean\n  ): Promise<boolean> {\n    let schema;\n    const acl = runOptions.acl;\n    const isMaster = acl === undefined;\n    var aclGroup: string[] = acl || [];\n    return this.loadSchema()\n      .then(s => {\n        schema = s;\n        if (isMaster) {\n          return Promise.resolve();\n        }\n        return this.canAddField(schema, className, object, aclGroup, runOptions);\n      })\n      .then(() => {\n        return schema.validateObject(className, object, query, maintenance);\n      });\n  }\n\n  update(\n    className: string,\n    query: any,\n    update: any,\n    { acl, many, upsert, addsField }: FullQueryOptions = {},\n    skipSanitization: boolean = false,\n    validateOnly: boolean = false,\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    try {\n      Utils.checkProhibitedKeywords(this.options, update);\n    } catch (error) {\n      return Promise.reject(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n    }\n    const originalQuery = query;\n    const originalUpdate = update;\n    // Make a copy of the object, so we don't mutate the incoming data.\n    update = deepcopy(update);\n    var relationUpdates = [];\n    var isMaster = acl === undefined;\n    var aclGroup = acl || [];\n\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      return (isMaster\n        ? Promise.resolve()\n        : schemaController.validatePermission(className, aclGroup, 'update')\n      )\n        .then(() => {\n          relationUpdates = this.collectRelationUpdates(className, originalQuery.objectId, update);\n          if (!isMaster) {\n            query = this.addPointerPermissions(\n              schemaController,\n              className,\n              'update',\n              query,\n              aclGroup\n            );\n\n            if (addsField) {\n              query = {\n                $and: [\n                  query,\n                  this.addPointerPermissions(\n                    schemaController,\n                    className,\n                    'addField',\n                    query,\n                    aclGroup\n                  ),\n                ],\n              };\n            }\n          }\n          if (!query) {\n            return Promise.resolve();\n          }\n          if (acl) {\n            query = addWriteACL(query, acl);\n          }\n          validateQuery(query, isMaster, false, true);\n          return schemaController\n            .getOneSchema(className, true)\n            .catch(error => {\n              // If the schema doesn't exist, pretend it exists with no fields. This behavior\n              // will likely need revisiting.\n              if (error === undefined) {\n                return { fields: {} };\n              }\n              throw error;\n            })\n            .then(schema => {\n              Object.keys(update).forEach(fieldName => {\n                if (fieldName.match(/^authData\\.([a-zA-Z0-9_]+)\\.id$/)) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_KEY_NAME,\n                    `Invalid field name for update: ${fieldName}`\n                  );\n                }\n                const rootFieldName = getRootFieldName(fieldName);\n                if (\n                  !SchemaController.fieldNameIsValid(rootFieldName, className) &&\n                  !isSpecialUpdateKey(rootFieldName)\n                ) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_KEY_NAME,\n                    `Invalid field name for update: ${fieldName}`\n                  );\n                }\n              });\n              for (const updateOperation in update) {\n                if (\n                  update[updateOperation] &&\n                  typeof update[updateOperation] === 'object' &&\n                  Object.keys(update[updateOperation]).some(\n                    innerKey => innerKey.includes('$') || innerKey.includes('.')\n                  )\n                ) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_NESTED_KEY,\n                    \"Nested keys should not contain the '$' or '.' characters\"\n                  );\n                }\n              }\n              update = transformObjectACL(update);\n              convertEmailToLowercase(update, className, this.options);\n              convertUsernameToLowercase(update, className, this.options);\n              transformAuthData(className, update, schema);\n              if (validateOnly) {\n                return this.adapter.find(className, schema, query, {}).then(result => {\n                  if (!result || !result.length) {\n                    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n                  }\n                  return {};\n                });\n              }\n              if (many) {\n                return this.adapter.updateObjectsByQuery(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              } else if (upsert) {\n                return this.adapter.upsertOneObject(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              } else {\n                return this.adapter.findOneAndUpdate(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              }\n            });\n        })\n        .then((result: any) => {\n          if (!result) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n          if (validateOnly) {\n            return result;\n          }\n          return this.handleRelationUpdates(\n            className,\n            originalQuery.objectId,\n            update,\n            relationUpdates\n          ).then(() => {\n            return result;\n          });\n        })\n        .then(result => {\n          if (skipSanitization) {\n            return Promise.resolve(result);\n          }\n          return this._sanitizeDatabaseResult(originalUpdate, result);\n        });\n    });\n  }\n\n  // Collect all relation-updating operations from a REST-format update.\n  // Returns a list of all relation updates to perform\n  // This mutates update.\n  collectRelationUpdates(className: string, objectId: ?string, update: any) {\n    var ops = [];\n    var deleteMe = [];\n    objectId = update.objectId || objectId;\n\n    var process = (op, key) => {\n      if (!op) {\n        return;\n      }\n      if (op.__op == 'AddRelation') {\n        ops.push({ key, op });\n        deleteMe.push(key);\n      }\n\n      if (op.__op == 'RemoveRelation') {\n        ops.push({ key, op });\n        deleteMe.push(key);\n      }\n\n      if (op.__op == 'Batch') {\n        for (var x of op.ops) {\n          process(x, key);\n        }\n      }\n    };\n\n    for (const key in update) {\n      process(update[key], key);\n    }\n    for (const key of deleteMe) {\n      delete update[key];\n    }\n    return ops;\n  }\n\n  // Processes relation-updating operations from a REST-format update.\n  // Returns a promise that resolves when all updates have been performed\n  handleRelationUpdates(className: string, objectId: string, update: any, ops: any) {\n    var pending = [];\n    objectId = update.objectId || objectId;\n    ops.forEach(({ key, op }) => {\n      if (!op) {\n        return;\n      }\n      if (op.__op == 'AddRelation') {\n        for (const object of op.objects) {\n          pending.push(this.addRelation(key, className, objectId, object.objectId));\n        }\n      }\n\n      if (op.__op == 'RemoveRelation') {\n        for (const object of op.objects) {\n          pending.push(this.removeRelation(key, className, objectId, object.objectId));\n        }\n      }\n    });\n\n    return Promise.all(pending);\n  }\n\n  // Adds a relation.\n  // Returns a promise that resolves successfully iff the add was successful.\n  addRelation(key: string, fromClassName: string, fromId: string, toId: string) {\n    const doc = {\n      relatedId: toId,\n      owningId: fromId,\n    };\n    return this.adapter.upsertOneObject(\n      `_Join:${key}:${fromClassName}`,\n      relationSchema,\n      doc,\n      doc,\n      this._transactionalSession\n    );\n  }\n\n  // Removes a relation.\n  // Returns a promise that resolves successfully iff the remove was\n  // successful.\n  removeRelation(key: string, fromClassName: string, fromId: string, toId: string) {\n    var doc = {\n      relatedId: toId,\n      owningId: fromId,\n    };\n    return this.adapter\n      .deleteObjectsByQuery(\n        `_Join:${key}:${fromClassName}`,\n        relationSchema,\n        doc,\n        this._transactionalSession\n      )\n      .catch(error => {\n        // We don't care if they try to delete a non-existent relation.\n        if (error.code == Parse.Error.OBJECT_NOT_FOUND) {\n          return;\n        }\n        throw error;\n      });\n  }\n\n  // Removes objects matches this query from the database.\n  // Returns a promise that resolves successfully iff the object was\n  // deleted.\n  // Options:\n  //   acl:  a list of strings. If the object to be updated has an ACL,\n  //         one of the provided strings must provide the caller with\n  //         write permissions.\n  destroy(\n    className: string,\n    query: any,\n    { acl }: QueryOptions = {},\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const isMaster = acl === undefined;\n    const aclGroup = acl || [];\n\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      return (isMaster\n        ? Promise.resolve()\n        : schemaController.validatePermission(className, aclGroup, 'delete')\n      ).then(() => {\n        if (!isMaster) {\n          query = this.addPointerPermissions(\n            schemaController,\n            className,\n            'delete',\n            query,\n            aclGroup\n          );\n          if (!query) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n        }\n        // delete by query\n        if (acl) {\n          query = addWriteACL(query, acl);\n        }\n        validateQuery(query, isMaster, false, false);\n        return schemaController\n          .getOneSchema(className)\n          .catch(error => {\n            // If the schema doesn't exist, pretend it exists with no fields. This behavior\n            // will likely need revisiting.\n            if (error === undefined) {\n              return { fields: {} };\n            }\n            throw error;\n          })\n          .then(parseFormatSchema =>\n            this.adapter.deleteObjectsByQuery(\n              className,\n              parseFormatSchema,\n              query,\n              this._transactionalSession\n            )\n          )\n          .catch(error => {\n            // When deleting sessions while changing passwords, don't throw an error if they don't have any sessions.\n            if (className === '_Session' && error.code === Parse.Error.OBJECT_NOT_FOUND) {\n              return Promise.resolve({});\n            }\n            throw error;\n          });\n      });\n    });\n  }\n\n  // Inserts an object into the database.\n  // Returns a promise that resolves successfully iff the object saved.\n  create(\n    className: string,\n    object: any,\n    { acl }: QueryOptions = {},\n    validateOnly: boolean = false,\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    try {\n      Utils.checkProhibitedKeywords(this.options, object);\n    } catch (error) {\n      return Promise.reject(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n    }\n    // Make a copy of the object, so we don't mutate the incoming data.\n    const originalObject = object;\n    object = transformObjectACL(object);\n\n    convertEmailToLowercase(object, className, this.options);\n    convertUsernameToLowercase(object, className, this.options);\n    object.createdAt = { iso: object.createdAt, __type: 'Date' };\n    object.updatedAt = { iso: object.updatedAt, __type: 'Date' };\n\n    var isMaster = acl === undefined;\n    var aclGroup = acl || [];\n    const relationUpdates = this.collectRelationUpdates(className, null, object);\n\n    return this.validateClassName(className)\n      .then(() => this.loadSchemaIfNeeded(validSchemaController))\n      .then(schemaController => {\n        return (isMaster\n          ? Promise.resolve()\n          : schemaController.validatePermission(className, aclGroup, 'create')\n        )\n          .then(() => schemaController.enforceClassExists(className))\n          .then(() => schemaController.getOneSchema(className, true))\n          .then(schema => {\n            transformAuthData(className, object, schema);\n            flattenUpdateOperatorsForCreate(object);\n            if (validateOnly) {\n              return {};\n            }\n            return this.adapter.createObject(\n              className,\n              SchemaController.convertSchemaToAdapterSchema(schema),\n              object,\n              this._transactionalSession\n            );\n          })\n          .then(result => {\n            if (validateOnly) {\n              return originalObject;\n            }\n            return this.handleRelationUpdates(\n              className,\n              object.objectId,\n              object,\n              relationUpdates\n            ).then(() => {\n              return this._sanitizeDatabaseResult(originalObject, result.ops[0]);\n            });\n          });\n      });\n  }\n\n  canAddField(\n    schema: SchemaController.SchemaController,\n    className: string,\n    object: any,\n    aclGroup: string[],\n    runOptions: QueryOptions\n  ): Promise<void> {\n    const classSchema = schema.schemaData[className];\n    if (!classSchema) {\n      return Promise.resolve();\n    }\n    const fields = Object.keys(object);\n    const schemaFields = Object.keys(classSchema.fields);\n    const newKeys = fields.filter(field => {\n      // Skip fields that are unset\n      if (object[field] && object[field].__op && object[field].__op === 'Delete') {\n        return false;\n      }\n      return schemaFields.indexOf(getRootFieldName(field)) < 0;\n    });\n    if (newKeys.length > 0) {\n      // adds a marker that new field is being adding during update\n      runOptions.addsField = true;\n\n      const action = runOptions.action;\n      return schema.validatePermission(className, aclGroup, 'addField', action);\n    }\n    return Promise.resolve();\n  }\n\n  // Won't delete collections in the system namespace\n  /**\n   * Delete all classes and clears the schema cache\n   *\n   * @param {boolean} fast set to true if it's ok to just delete rows and not indexes\n   * @returns {Promise<void>} when the deletions completes\n   */\n  deleteEverything(fast: boolean = false): Promise<any> {\n    this.schemaPromise = null;\n    SchemaCache.clear();\n    return this.adapter.deleteAllClasses(fast);\n  }\n\n  // Returns a promise for a list of related ids given an owning id.\n  // className here is the owning className.\n  relatedIds(\n    className: string,\n    key: string,\n    owningId: string,\n    queryOptions: QueryOptions\n  ): Promise<Array<string>> {\n    const { skip, limit, sort } = queryOptions;\n    const findOptions = {};\n    if (sort && sort.createdAt && this.adapter.canSortOnJoinTables) {\n      findOptions.sort = { _id: sort.createdAt };\n      findOptions.limit = limit;\n      findOptions.skip = skip;\n      queryOptions.skip = 0;\n    }\n    return this.adapter\n      .find(joinTableName(className, key), relationSchema, { owningId }, findOptions)\n      .then(results => results.map(result => result.relatedId));\n  }\n\n  // Returns a promise for a list of owning ids given some related ids.\n  // className here is the owning className.\n  owningIds(className: string, key: string, relatedIds: string[]): Promise<string[]> {\n    return this.adapter\n      .find(\n        joinTableName(className, key),\n        relationSchema,\n        { relatedId: { $in: relatedIds } },\n        { keys: ['owningId'] }\n      )\n      .then(results => results.map(result => result.owningId));\n  }\n\n  // Modifies query so that it no longer has $in on relation fields, or\n  // equal-to-pointer constraints on relation fields.\n  // Returns a promise that resolves when query is mutated\n  reduceInRelation(className: string, query: any, schema: any): Promise<any> {\n    // Search for an in-relation or equal-to-relation\n    // Make it sequential for now, not sure of paralleization side effects\n    const promises = [];\n    if (query['$or']) {\n      const ors = query['$or'];\n      promises.push(\n        ...ors.map((aQuery, index) => {\n          return this.reduceInRelation(className, aQuery, schema).then(aQuery => {\n            query['$or'][index] = aQuery;\n          });\n        })\n      );\n    }\n    if (query['$and']) {\n      const ands = query['$and'];\n      promises.push(\n        ...ands.map((aQuery, index) => {\n          return this.reduceInRelation(className, aQuery, schema).then(aQuery => {\n            query['$and'][index] = aQuery;\n          });\n        })\n      );\n    }\n\n    const otherKeys = Object.keys(query).map(key => {\n      if (key === '$and' || key === '$or') {\n        return;\n      }\n      const t = schema.getExpectedType(className, key);\n      if (!t || t.type !== 'Relation') {\n        return Promise.resolve(query);\n      }\n      let queries: ?(any[]) = null;\n      if (\n        query[key] &&\n        (query[key]['$in'] ||\n          query[key]['$ne'] ||\n          query[key]['$nin'] ||\n          query[key].__type == 'Pointer')\n      ) {\n        // Build the list of queries\n        queries = Object.keys(query[key]).map(constraintKey => {\n          let relatedIds;\n          let isNegation = false;\n          if (constraintKey === 'objectId') {\n            relatedIds = [query[key].objectId];\n          } else if (constraintKey == '$in') {\n            relatedIds = query[key]['$in'].map(r => r.objectId);\n          } else if (constraintKey == '$nin') {\n            isNegation = true;\n            relatedIds = query[key]['$nin'].map(r => r.objectId);\n          } else if (constraintKey == '$ne') {\n            isNegation = true;\n            relatedIds = [query[key]['$ne'].objectId];\n          } else {\n            return;\n          }\n          return {\n            isNegation,\n            relatedIds,\n          };\n        });\n      } else {\n        queries = [{ isNegation: false, relatedIds: [] }];\n      }\n\n      // remove the current queryKey as we don,t need it anymore\n      delete query[key];\n      // execute each query independently to build the list of\n      // $in / $nin\n      const promises = queries.map(q => {\n        if (!q) {\n          return Promise.resolve();\n        }\n        return this.owningIds(className, key, q.relatedIds).then(ids => {\n          if (q.isNegation) {\n            this.addNotInObjectIdsIds(ids, query);\n          } else {\n            this.addInObjectIdsIds(ids, query);\n          }\n          return Promise.resolve();\n        });\n      });\n\n      return Promise.all(promises).then(() => {\n        return Promise.resolve();\n      });\n    });\n\n    return Promise.all([...promises, ...otherKeys]).then(() => {\n      return Promise.resolve(query);\n    });\n  }\n\n  // Modifies query so that it no longer has $relatedTo\n  // Returns a promise that resolves when query is mutated\n  reduceRelationKeys(className: string, query: any, queryOptions: any): ?Promise<void> {\n    if (query['$or']) {\n      return Promise.all(\n        query['$or'].map(aQuery => {\n          return this.reduceRelationKeys(className, aQuery, queryOptions);\n        })\n      );\n    }\n    if (query['$and']) {\n      return Promise.all(\n        query['$and'].map(aQuery => {\n          return this.reduceRelationKeys(className, aQuery, queryOptions);\n        })\n      );\n    }\n    var relatedTo = query['$relatedTo'];\n    if (relatedTo) {\n      return this.relatedIds(\n        relatedTo.object.className,\n        relatedTo.key,\n        relatedTo.object.objectId,\n        queryOptions\n      )\n        .then(ids => {\n          delete query['$relatedTo'];\n          this.addInObjectIdsIds(ids, query);\n          return this.reduceRelationKeys(className, query, queryOptions);\n        })\n        .then(() => {});\n    }\n  }\n\n  addInObjectIdsIds(ids: ?Array<string> = null, query: any) {\n    const idsFromString: ?Array<string> =\n      typeof query.objectId === 'string' ? [query.objectId] : null;\n    const idsFromEq: ?Array<string> =\n      query.objectId && query.objectId['$eq'] ? [query.objectId['$eq']] : null;\n    const idsFromIn: ?Array<string> =\n      query.objectId && query.objectId['$in'] ? query.objectId['$in'] : null;\n\n    // @flow-disable-next\n    const allIds: Array<Array<string>> = [idsFromString, idsFromEq, idsFromIn, ids].filter(\n      list => list !== null\n    );\n    const totalLength = allIds.reduce((memo, list) => memo + list.length, 0);\n\n    let idsIntersection = [];\n    if (totalLength > 125) {\n      idsIntersection = intersect.big(allIds);\n    } else {\n      idsIntersection = intersect(allIds);\n    }\n\n    // Need to make sure we don't clobber existing shorthand $eq constraints on objectId.\n    if (!('objectId' in query)) {\n      query.objectId = {\n        $in: undefined,\n      };\n    } else if (typeof query.objectId === 'string') {\n      query.objectId = {\n        $in: undefined,\n        $eq: query.objectId,\n      };\n    }\n    query.objectId['$in'] = idsIntersection;\n\n    return query;\n  }\n\n  addNotInObjectIdsIds(ids: string[] = [], query: any) {\n    const idsFromNin = query.objectId && query.objectId['$nin'] ? query.objectId['$nin'] : [];\n    let allIds = [...idsFromNin, ...ids].filter(list => list !== null);\n\n    // make a set and spread to remove duplicates\n    allIds = [...new Set(allIds)];\n\n    // Need to make sure we don't clobber existing shorthand $eq constraints on objectId.\n    if (!('objectId' in query)) {\n      query.objectId = {\n        $nin: undefined,\n      };\n    } else if (typeof query.objectId === 'string') {\n      query.objectId = {\n        $nin: undefined,\n        $eq: query.objectId,\n      };\n    }\n\n    query.objectId['$nin'] = allIds;\n    return query;\n  }\n\n  // Runs a query on the database.\n  // Returns a promise that resolves to a list of items.\n  // Options:\n  //   skip    number of results to skip.\n  //   limit   limit to this number of results.\n  //   sort    an object where keys are the fields to sort by.\n  //           the value is +1 for ascending, -1 for descending.\n  //   count   run a count instead of returning results.\n  //   acl     restrict this operation with an ACL for the provided array\n  //           of user objectIds and roles. acl: null means no user.\n  //           when this field is not present, don't do anything regarding ACLs.\n  //  caseInsensitive make string comparisons case insensitive\n  // TODO: make userIds not needed here. The db adapter shouldn't know\n  // anything about users, ideally. Then, improve the format of the ACL\n  // arg to work like the others.\n  find(\n    className: string,\n    query: any,\n    {\n      skip,\n      limit,\n      acl,\n      sort = {},\n      count,\n      keys,\n      op,\n      distinct,\n      pipeline,\n      readPreference,\n      hint,\n      caseInsensitive = false,\n      explain,\n      comment,\n    }: any = {},\n    auth: any = {},\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const isMaintenance = auth.isMaintenance;\n    const isMaster = acl === undefined || isMaintenance;\n    const aclGroup = acl || [];\n    op =\n      op || (typeof query.objectId == 'string' && Object.keys(query).length === 1 ? 'get' : 'find');\n    // Count operation if counting\n    op = count === true ? 'count' : op;\n\n    let classExists = true;\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      //Allow volatile classes if querying with Master (for _PushStatus)\n      //TODO: Move volatile classes concept into mongo adapter, postgres adapter shouldn't care\n      //that api.parse.com breaks when _PushStatus exists in mongo.\n      return schemaController\n        .getOneSchema(className, isMaster)\n        .catch(error => {\n          // Behavior for non-existent classes is kinda weird on Parse.com. Probably doesn't matter too much.\n          // For now, pretend the class exists but has no objects,\n          if (error === undefined) {\n            classExists = false;\n            return { fields: {} };\n          }\n          throw error;\n        })\n        .then(schema => {\n          // Parse.com treats queries on _created_at and _updated_at as if they were queries on createdAt and updatedAt,\n          // so duplicate that behavior here. If both are specified, the correct behavior to match Parse.com is to\n          // use the one that appears first in the sort list.\n          if (sort._created_at) {\n            sort.createdAt = sort._created_at;\n            delete sort._created_at;\n          }\n          if (sort._updated_at) {\n            sort.updatedAt = sort._updated_at;\n            delete sort._updated_at;\n          }\n          const queryOptions = {\n            skip,\n            limit,\n            sort,\n            keys,\n            readPreference,\n            hint,\n            caseInsensitive: this.options.enableCollationCaseComparison ? false : caseInsensitive,\n            explain,\n            comment,\n          };\n          Object.keys(sort).forEach(fieldName => {\n            if (fieldName.match(/^authData\\.([a-zA-Z0-9_]+)\\.id$/)) {\n              throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Cannot sort by ${fieldName}`);\n            }\n            const rootFieldName = getRootFieldName(fieldName);\n            if (!SchemaController.fieldNameIsValid(rootFieldName, className)) {\n              throw new Parse.Error(\n                Parse.Error.INVALID_KEY_NAME,\n                `Invalid field name: ${fieldName}.`\n              );\n            }\n            if (!schema.fields[fieldName.split('.')[0]] && fieldName !== 'score') {\n              delete sort[fieldName];\n            }\n          });\n          return (isMaster\n            ? Promise.resolve()\n            : schemaController.validatePermission(className, aclGroup, op)\n          )\n            .then(() => this.reduceRelationKeys(className, query, queryOptions))\n            .then(() => this.reduceInRelation(className, query, schemaController))\n            .then(() => {\n              let protectedFields;\n              if (!isMaster) {\n                query = this.addPointerPermissions(\n                  schemaController,\n                  className,\n                  op,\n                  query,\n                  aclGroup\n                );\n                /* Don't use projections to optimize the protectedFields since the protectedFields\n                  based on pointer-permissions are determined after querying. The filtering can\n                  overwrite the protected fields. */\n                protectedFields = this.addProtectedFields(\n                  schemaController,\n                  className,\n                  query,\n                  aclGroup,\n                  auth,\n                  queryOptions\n                );\n              }\n              if (!query) {\n                if (op === 'get') {\n                  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n                } else {\n                  return [];\n                }\n              }\n              if (!isMaster) {\n                if (op === 'update' || op === 'delete') {\n                  query = addWriteACL(query, aclGroup);\n                } else {\n                  query = addReadACL(query, aclGroup);\n                }\n              }\n              validateQuery(query, isMaster, isMaintenance, false);\n              if (count) {\n                if (!classExists) {\n                  return 0;\n                } else {\n                  return this.adapter.count(\n                    className,\n                    schema,\n                    query,\n                    readPreference,\n                    undefined,\n                    hint,\n                    comment\n                  );\n                }\n              } else if (distinct) {\n                if (!classExists) {\n                  return [];\n                } else {\n                  return this.adapter.distinct(className, schema, query, distinct);\n                }\n              } else if (pipeline) {\n                if (!classExists) {\n                  return [];\n                } else {\n                  return this.adapter.aggregate(\n                    className,\n                    schema,\n                    pipeline,\n                    readPreference,\n                    hint,\n                    explain,\n                    comment\n                  );\n                }\n              } else if (explain) {\n                return this.adapter.find(className, schema, query, queryOptions);\n              } else {\n                return this.adapter\n                  .find(className, schema, query, queryOptions)\n                  .then(objects =>\n                    objects.map(object => {\n                      object = untransformObjectACL(object);\n                      return filterSensitiveData(\n                        isMaster,\n                        isMaintenance,\n                        aclGroup,\n                        auth,\n                        op,\n                        schemaController,\n                        className,\n                        protectedFields,\n                        object\n                      );\n                    })\n                  )\n                  .catch(error => {\n                    throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, error);\n                  });\n              }\n            });\n        });\n    });\n  }\n\n  deleteSchema(className: string): Promise<void> {\n    let schemaController;\n    return this.loadSchema({ clearCache: true })\n      .then(s => {\n        schemaController = s;\n        return schemaController.getOneSchema(className, true);\n      })\n      .catch(error => {\n        if (error === undefined) {\n          return { fields: {} };\n        } else {\n          throw error;\n        }\n      })\n      .then((schema: any) => {\n        return this.collectionExists(className)\n          .then(() => this.adapter.count(className, { fields: {} }, null, '', false))\n          .then(count => {\n            if (count > 0) {\n              throw new Parse.Error(\n                255,\n                `Class ${className} is not empty, contains ${count} objects, cannot drop schema.`\n              );\n            }\n            return this.adapter.deleteClass(className);\n          })\n          .then(wasParseCollection => {\n            if (wasParseCollection) {\n              const relationFieldNames = Object.keys(schema.fields).filter(\n                fieldName => schema.fields[fieldName].type === 'Relation'\n              );\n              return Promise.all(\n                relationFieldNames.map(name =>\n                  this.adapter.deleteClass(joinTableName(className, name))\n                )\n              ).then(() => {\n                SchemaCache.del(className);\n                return schemaController.reloadData();\n              });\n            } else {\n              return Promise.resolve();\n            }\n          });\n      });\n  }\n\n  // This helps to create intermediate objects for simpler comparison of\n  // key value pairs used in query objects. Each key value pair will represented\n  // in a similar way to json\n  objectToEntriesStrings(query: any): Array<string> {\n    return Object.entries(query).map(a => a.map(s => JSON.stringify(s)).join(':'));\n  }\n\n  // Naive logic reducer for OR operations meant to be used only for pointer permissions.\n  reduceOrOperation(query: { $or: Array<any> }): any {\n    if (!query.$or) {\n      return query;\n    }\n    const queries = query.$or.map(q => this.objectToEntriesStrings(q));\n    let repeat = false;\n    do {\n      repeat = false;\n      for (let i = 0; i < queries.length - 1; i++) {\n        for (let j = i + 1; j < queries.length; j++) {\n          const [shorter, longer] = queries[i].length > queries[j].length ? [j, i] : [i, j];\n          const foundEntries = queries[shorter].reduce(\n            (acc, entry) => acc + (queries[longer].includes(entry) ? 1 : 0),\n            0\n          );\n          const shorterEntries = queries[shorter].length;\n          if (foundEntries === shorterEntries) {\n            // If the shorter query is completely contained in the longer one, we can strike\n            // out the longer query.\n            query.$or.splice(longer, 1);\n            queries.splice(longer, 1);\n            repeat = true;\n            break;\n          }\n        }\n      }\n    } while (repeat);\n    if (query.$or.length === 1) {\n      query = { ...query, ...query.$or[0] };\n      delete query.$or;\n    }\n    return query;\n  }\n\n  // Naive logic reducer for AND operations meant to be used only for pointer permissions.\n  reduceAndOperation(query: { $and: Array<any> }): any {\n    if (!query.$and) {\n      return query;\n    }\n    const queries = query.$and.map(q => this.objectToEntriesStrings(q));\n    let repeat = false;\n    do {\n      repeat = false;\n      for (let i = 0; i < queries.length - 1; i++) {\n        for (let j = i + 1; j < queries.length; j++) {\n          const [shorter, longer] = queries[i].length > queries[j].length ? [j, i] : [i, j];\n          const foundEntries = queries[shorter].reduce(\n            (acc, entry) => acc + (queries[longer].includes(entry) ? 1 : 0),\n            0\n          );\n          const shorterEntries = queries[shorter].length;\n          if (foundEntries === shorterEntries) {\n            // If the shorter query is completely contained in the longer one, we can strike\n            // out the shorter query.\n            query.$and.splice(shorter, 1);\n            queries.splice(shorter, 1);\n            repeat = true;\n            break;\n          }\n        }\n      }\n    } while (repeat);\n    if (query.$and.length === 1) {\n      query = { ...query, ...query.$and[0] };\n      delete query.$and;\n    }\n    return query;\n  }\n\n  // Constraints query using CLP's pointer permissions (PP) if any.\n  // 1. Etract the user id from caller's ACLgroup;\n  // 2. Exctract a list of field names that are PP for target collection and operation;\n  // 3. Constraint the original query so that each PP field must\n  // point to caller's id (or contain it in case of PP field being an array)\n  addPointerPermissions(\n    schema: SchemaController.SchemaController,\n    className: string,\n    operation: string,\n    query: any,\n    aclGroup: any[] = []\n  ): any {\n    // Check if class has public permission for operation\n    // If the BaseCLP pass, let go through\n    if (schema.testPermissionsForClassName(className, aclGroup, operation)) {\n      return query;\n    }\n    const perms = schema.getClassLevelPermissions(className);\n\n    const userACL = aclGroup.filter(acl => {\n      return acl.indexOf('role:') != 0 && acl != '*';\n    });\n\n    const groupKey =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    const permFields = [];\n\n    if (perms[operation] && perms[operation].pointerFields) {\n      permFields.push(...perms[operation].pointerFields);\n    }\n\n    if (perms[groupKey]) {\n      for (const field of perms[groupKey]) {\n        if (!permFields.includes(field)) {\n          permFields.push(field);\n        }\n      }\n    }\n    // the ACL should have exactly 1 user\n    if (permFields.length > 0) {\n      // the ACL should have exactly 1 user\n      // No user set return undefined\n      // If the length is > 1, that means we didn't de-dupe users correctly\n      if (userACL.length != 1) {\n        return;\n      }\n      const userId = userACL[0];\n      const userPointer = {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: userId,\n      };\n\n      const queries = permFields.map(key => {\n        const fieldDescriptor = schema.getExpectedType(className, key);\n        const fieldType =\n          fieldDescriptor &&\n          typeof fieldDescriptor === 'object' &&\n          Object.prototype.hasOwnProperty.call(fieldDescriptor, 'type')\n            ? fieldDescriptor.type\n            : null;\n\n        let queryClause;\n\n        if (fieldType === 'Pointer') {\n          // constraint for single pointer setup\n          queryClause = { [key]: userPointer };\n        } else if (fieldType === 'Array') {\n          // constraint for users-array setup\n          queryClause = { [key]: { $all: [userPointer] } };\n        } else if (fieldType === 'Object') {\n          // constraint for object setup\n          queryClause = { [key]: userPointer };\n        } else {\n          // This means that there is a CLP field of an unexpected type. This condition should not happen, which is\n          // why is being treated as an error.\n          throw Error(\n            `An unexpected condition occurred when resolving pointer permissions: ${className} ${key}`\n          );\n        }\n        // if we already have a constraint on the key, use the $and\n        if (Object.prototype.hasOwnProperty.call(query, key)) {\n          return this.reduceAndOperation({ $and: [queryClause, query] });\n        }\n        // otherwise just add the constaint\n        return Object.assign({}, query, queryClause);\n      });\n\n      return queries.length === 1 ? queries[0] : this.reduceOrOperation({ $or: queries });\n    } else {\n      return query;\n    }\n  }\n\n  addProtectedFields(\n    schema: SchemaController.SchemaController | any,\n    className: string,\n    query: any = {},\n    aclGroup: any[] = [],\n    auth: any = {},\n    queryOptions: FullQueryOptions = {}\n  ): null | string[] {\n    const perms =\n      schema && schema.getClassLevelPermissions\n        ? schema.getClassLevelPermissions(className)\n        : schema;\n    if (!perms) return null;\n\n    const protectedFields = perms.protectedFields;\n    if (!protectedFields) return null;\n\n    if (aclGroup.indexOf(query.objectId) > -1) return null;\n\n    // for queries where \"keys\" are set and do not include all 'userField':{field},\n    // we have to transparently include it, and then remove before returning to client\n    // Because if such key not projected the permission won't be enforced properly\n    // PS this is called when 'excludeKeys' already reduced to 'keys'\n    const preserveKeys = queryOptions.keys;\n\n    // these are keys that need to be included only\n    // to be able to apply protectedFields by pointer\n    // and then unset before returning to client (later in  filterSensitiveFields)\n    const serverOnlyKeys = [];\n\n    const authenticated = auth.user;\n\n    // map to allow check without array search\n    const roles = (auth.userRoles || []).reduce((acc, r) => {\n      acc[r] = protectedFields[r];\n      return acc;\n    }, {});\n\n    // array of sets of protected fields. separate item for each applicable criteria\n    const protectedKeysSets = [];\n\n    for (const key in protectedFields) {\n      // skip userFields\n      if (key.startsWith('userField:')) {\n        if (preserveKeys) {\n          const fieldName = key.substring(10);\n          if (!preserveKeys.includes(fieldName)) {\n            // 1. put it there temporarily\n            queryOptions.keys && queryOptions.keys.push(fieldName);\n            // 2. preserve it delete later\n            serverOnlyKeys.push(fieldName);\n          }\n        }\n        continue;\n      }\n\n      // add public tier\n      if (key === '*') {\n        protectedKeysSets.push(protectedFields[key]);\n        continue;\n      }\n\n      if (authenticated) {\n        if (key === 'authenticated') {\n          // for logged in users\n          protectedKeysSets.push(protectedFields[key]);\n          continue;\n        }\n\n        if (roles[key] && key.startsWith('role:')) {\n          // add applicable roles\n          protectedKeysSets.push(roles[key]);\n        }\n      }\n    }\n\n    // check if there's a rule for current user's id\n    if (authenticated) {\n      const userId = auth.user.id;\n      if (perms.protectedFields[userId]) {\n        protectedKeysSets.push(perms.protectedFields[userId]);\n      }\n    }\n\n    // preserve fields to be removed before sending response to client\n    if (serverOnlyKeys.length > 0) {\n      perms.protectedFields.temporaryKeys = serverOnlyKeys;\n    }\n\n    let protectedKeys = protectedKeysSets.reduce((acc, next) => {\n      if (next) {\n        acc.push(...next);\n      }\n      return acc;\n    }, []);\n\n    // intersect all sets of protectedFields\n    protectedKeysSets.forEach(fields => {\n      if (fields) {\n        protectedKeys = protectedKeys.filter(v => fields.includes(v));\n      }\n    });\n\n    return protectedKeys;\n  }\n\n  createTransactionalSession() {\n    return this.adapter.createTransactionalSession().then(transactionalSession => {\n      this._transactionalSession = transactionalSession;\n    });\n  }\n\n  commitTransactionalSession() {\n    if (!this._transactionalSession) {\n      throw new Error('There is no transactional session to commit');\n    }\n    return this.adapter.commitTransactionalSession(this._transactionalSession).then(() => {\n      this._transactionalSession = null;\n    });\n  }\n\n  abortTransactionalSession() {\n    if (!this._transactionalSession) {\n      throw new Error('There is no transactional session to abort');\n    }\n    return this.adapter.abortTransactionalSession(this._transactionalSession).then(() => {\n      this._transactionalSession = null;\n    });\n  }\n\n  // TODO: create indexes on first creation of a _User object. Otherwise it's impossible to\n  // have a Parse app without it having a _User collection.\n  async performInitialization() {\n    await this.adapter.performInitialization({\n      VolatileClassesSchemas: SchemaController.VolatileClassesSchemas,\n    });\n    const requiredUserFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._User,\n      },\n    };\n    const requiredRoleFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._Role,\n      },\n    };\n    const requiredIdempotencyFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._Idempotency,\n      },\n    };\n    await this.loadSchema().then(schema => schema.enforceClassExists('_User'));\n    await this.loadSchema().then(schema => schema.enforceClassExists('_Role'));\n    await this.loadSchema().then(schema => schema.enforceClassExists('_Idempotency'));\n\n    await this.adapter.ensureUniqueness('_User', requiredUserFields, ['username']).catch(error => {\n      logger.warn('Unable to ensure uniqueness for usernames: ', error);\n      throw error;\n    });\n\n    if (!this.options.enableCollationCaseComparison) {\n      await this.adapter\n        .ensureIndex('_User', requiredUserFields, ['username'], 'case_insensitive_username', true)\n        .catch(error => {\n          logger.warn('Unable to create case insensitive username index: ', error);\n          throw error;\n        });\n\n      await this.adapter\n        .ensureIndex('_User', requiredUserFields, ['email'], 'case_insensitive_email', true)\n        .catch(error => {\n          logger.warn('Unable to create case insensitive email index: ', error);\n          throw error;\n        });\n    }\n\n    await this.adapter.ensureUniqueness('_User', requiredUserFields, ['email']).catch(error => {\n      logger.warn('Unable to ensure uniqueness for user email addresses: ', error);\n      throw error;\n    });\n\n    await this.adapter.ensureUniqueness('_Role', requiredRoleFields, ['name']).catch(error => {\n      logger.warn('Unable to ensure uniqueness for role name: ', error);\n      throw error;\n    });\n\n    await this.adapter\n      .ensureUniqueness('_Idempotency', requiredIdempotencyFields, ['reqId'])\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for idempotency request ID: ', error);\n        throw error;\n      });\n\n    const isMongoAdapter = this.adapter instanceof MongoStorageAdapter;\n    const isPostgresAdapter = this.adapter instanceof PostgresStorageAdapter;\n    if (isMongoAdapter || isPostgresAdapter) {\n      let options = {};\n      if (isMongoAdapter) {\n        options = {\n          ttl: 0,\n        };\n      } else if (isPostgresAdapter) {\n        options = this.idempotencyOptions;\n        options.setIdempotencyFunction = true;\n      }\n      await this.adapter\n        .ensureIndex('_Idempotency', requiredIdempotencyFields, ['expire'], 'ttl', false, options)\n        .catch(error => {\n          logger.warn('Unable to create TTL index for idempotency expire date: ', error);\n          throw error;\n        });\n    }\n    await this.adapter.updateSchemaWithIndexes();\n  }\n\n  _expandResultOnKeyPath(object: any, key: string, value: any): any {\n    if (key.indexOf('.') < 0) {\n      object[key] = value[key];\n      return object;\n    }\n    const path = key.split('.');\n    const firstKey = path[0];\n    const nextPath = path.slice(1).join('.');\n\n    // Scan request data for denied keywords\n    if (this.options && this.options.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of this.options.requestKeywordDenylist) {\n        const match = Utils.objectContainsKeyValue(\n          { [firstKey]: true, [nextPath]: true },\n          keyword.key,\n          true\n        );\n        if (match) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_KEY_NAME,\n            `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`\n          );\n        }\n      }\n    }\n\n    object[firstKey] = this._expandResultOnKeyPath(\n      object[firstKey] || {},\n      nextPath,\n      value[firstKey]\n    );\n    delete object[key];\n    return object;\n  }\n\n  _sanitizeDatabaseResult(originalObject: any, result: any): Promise<any> {\n    const response = {};\n    if (!result) {\n      return Promise.resolve(response);\n    }\n    Object.keys(originalObject).forEach(key => {\n      const keyUpdate = originalObject[key];\n      // determine if that was an op\n      if (\n        keyUpdate &&\n        typeof keyUpdate === 'object' &&\n        keyUpdate.__op &&\n        ['Add', 'AddUnique', 'Remove', 'Increment', 'SetOnInsert'].indexOf(keyUpdate.__op) > -1\n      ) {\n        // only valid ops that produce an actionable result\n        // the op may have happened on a keypath\n        this._expandResultOnKeyPath(response, key, result);\n        // Revert array to object conversion on dot notation for arrays (e.g. \"field.0.key\")\n        if (key.includes('.')) {\n          const [field, index] = key.split('.');\n          const isArrayIndex = Array.from(index).every(c => c >= '0' && c <= '9');\n          if (isArrayIndex && Array.isArray(result[field]) && !Array.isArray(response[field])) {\n            response[field] = result[field];\n          }\n        }\n      }\n    });\n    return Promise.resolve(response);\n  }\n\n  static _validateQuery: (any, boolean, boolean, boolean) => void;\n  static filterSensitiveData: (boolean, boolean, any[], any, any, any, string, any[], any) => void;\n}\n\nmodule.exports = DatabaseController;\n// Expose validateQuery for tests\nmodule.exports._validateQuery = validateQuery;\nmodule.exports.filterSensitiveData = filterSensitiveData;\n"],"mappings":";;AAKA,IAAAA,KAAA,GAAAC,OAAA;AAEA,IAAAC,OAAA,GAAAC,sBAAA,CAAAF,OAAA;AAEA,IAAAG,UAAA,GAAAD,sBAAA,CAAAF,OAAA;AAEA,IAAAI,SAAA,GAAAF,sBAAA,CAAAF,OAAA;AACA,IAAAK,OAAA,GAAAH,sBAAA,CAAAF,OAAA;AACA,IAAAM,MAAA,GAAAJ,sBAAA,CAAAF,OAAA;AACA,IAAAO,gBAAA,GAAAC,uBAAA,CAAAR,OAAA;AACA,IAAAS,eAAA,GAAAT,OAAA;AACA,IAAAU,oBAAA,GAAAR,sBAAA,CAAAF,OAAA;AACA,IAAAW,uBAAA,GAAAT,sBAAA,CAAAF,OAAA;AACA,IAAAY,YAAA,GAAAV,sBAAA,CAAAF,OAAA;AAAwD,MAAAa,SAAA;EAAAC,UAAA,yBAjBxD;AACA;AAEA;AAEA;AAEA;AAEA;AAAA,SAAAC,yBAAAC,CAAA,6BAAAC,OAAA,mBAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAF,wBAAA,YAAAA,CAAAC,CAAA,WAAAA,CAAA,GAAAG,CAAA,GAAAD,CAAA,KAAAF,CAAA;AAAA,SAAAR,wBAAAQ,CAAA,EAAAE,CAAA,SAAAA,CAAA,IAAAF,CAAA,IAAAA,CAAA,CAAAI,UAAA,SAAAJ,CAAA,eAAAA,CAAA,uBAAAA,CAAA,yBAAAA,CAAA,WAAAK,OAAA,EAAAL,CAAA,QAAAG,CAAA,GAAAJ,wBAAA,CAAAG,CAAA,OAAAC,CAAA,IAAAA,CAAA,CAAAG,GAAA,CAAAN,CAAA,UAAAG,CAAA,CAAAI,GAAA,CAAAP,CAAA,OAAAQ,CAAA,KAAAC,SAAA,UAAAC,CAAA,GAAAC,MAAA,CAAAC,cAAA,IAAAD,MAAA,CAAAE,wBAAA,WAAAC,CAAA,IAAAd,CAAA,oBAAAc,CAAA,OAAAC,cAAA,CAAAC,IAAA,CAAAhB,CAAA,EAAAc,CAAA,SAAAG,CAAA,GAAAP,CAAA,GAAAC,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAc,CAAA,UAAAG,CAAA,KAAAA,CAAA,CAAAV,GAAA,IAAAU,CAAA,CAAAC,GAAA,IAAAP,MAAA,CAAAC,cAAA,CAAAJ,CAAA,EAAAM,CAAA,EAAAG,CAAA,IAAAT,CAAA,CAAAM,CAAA,IAAAd,CAAA,CAAAc,CAAA,YAAAN,CAAA,CAAAH,OAAA,GAAAL,CAAA,EAAAG,CAAA,IAAAA,CAAA,CAAAe,GAAA,CAAAlB,CAAA,EAAAQ,CAAA,GAAAA,CAAA;AAAA,SAAAtB,uBAAAc,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAI,UAAA,GAAAJ,CAAA,KAAAK,OAAA,EAAAL,CAAA;AAAA,SAAAmB,QAAAnB,CAAA,EAAAE,CAAA,QAAAC,CAAA,GAAAQ,MAAA,CAAAS,IAAA,CAAApB,CAAA,OAAAW,MAAA,CAAAU,qBAAA,QAAAC,CAAA,GAAAX,MAAA,CAAAU,qBAAA,CAAArB,CAAA,GAAAE,CAAA,KAAAoB,CAAA,GAAAA,CAAA,CAAAC,MAAA,WAAArB,CAAA,WAAAS,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAE,CAAA,EAAAsB,UAAA,OAAArB,CAAA,CAAAsB,IAAA,CAAAC,KAAA,CAAAvB,CAAA,EAAAmB,CAAA,YAAAnB,CAAA;AAAA,SAAAwB,cAAA3B,CAAA,aAAAE,CAAA,MAAAA,CAAA,GAAA0B,SAAA,CAAAC,MAAA,EAAA3B,CAAA,UAAAC,CAAA,WAAAyB,SAAA,CAAA1B,CAAA,IAAA0B,SAAA,CAAA1B,CAAA,QAAAA,CAAA,OAAAiB,OAAA,CAAAR,MAAA,CAAAR,CAAA,OAAA2B,OAAA,WAAA5B,CAAA,IAAA6B,eAAA,CAAA/B,CAAA,EAAAE,CAAA,EAAAC,CAAA,CAAAD,CAAA,SAAAS,MAAA,CAAAqB,yBAAA,GAAArB,MAAA,CAAAsB,gBAAA,CAAAjC,CAAA,EAAAW,MAAA,CAAAqB,yBAAA,CAAA7B,CAAA,KAAAgB,OAAA,CAAAR,MAAA,CAAAR,CAAA,GAAA2B,OAAA,WAAA5B,CAAA,IAAAS,MAAA,CAAAC,cAAA,CAAAZ,CAAA,EAAAE,CAAA,EAAAS,MAAA,CAAAE,wBAAA,CAAAV,CAAA,EAAAD,CAAA,iBAAAF,CAAA;AAAA,SAAA+B,gBAAA/B,CAAA,EAAAE,CAAA,EAAAC,CAAA,YAAAD,CAAA,GAAAgC,cAAA,CAAAhC,CAAA,MAAAF,CAAA,GAAAW,MAAA,CAAAC,cAAA,CAAAZ,CAAA,EAAAE,CAAA,IAAAiC,KAAA,EAAAhC,CAAA,EAAAqB,UAAA,MAAAY,YAAA,MAAAC,QAAA,UAAArC,CAAA,CAAAE,CAAA,IAAAC,CAAA,EAAAH,CAAA;AAAA,SAAAkC,eAAA/B,CAAA,QAAAc,CAAA,GAAAqB,YAAA,CAAAnC,CAAA,uCAAAc,CAAA,GAAAA,CAAA,GAAAA,CAAA;AAAA,SAAAqB,aAAAnC,CAAA,EAAAD,CAAA,2BAAAC,CAAA,KAAAA,CAAA,SAAAA,CAAA,MAAAH,CAAA,GAAAG,CAAA,CAAAoC,MAAA,CAAAC,WAAA,kBAAAxC,CAAA,QAAAiB,CAAA,GAAAjB,CAAA,CAAAgB,IAAA,CAAAb,CAAA,EAAAD,CAAA,uCAAAe,CAAA,SAAAA,CAAA,YAAAwB,SAAA,yEAAAvC,CAAA,GAAAwC,MAAA,GAAAC,MAAA,EAAAxC,CAAA;AAAA,SAAAyC,yBAAA5C,CAAA,EAAAG,CAAA,gBAAAH,CAAA,iBAAAsB,CAAA,EAAApB,CAAA,EAAAe,CAAA,GAAA4B,6BAAA,CAAA7C,CAAA,EAAAG,CAAA,OAAAQ,MAAA,CAAAU,qBAAA,QAAAb,CAAA,GAAAG,MAAA,CAAAU,qBAAA,CAAArB,CAAA,QAAAE,CAAA,MAAAA,CAAA,GAAAM,CAAA,CAAAqB,MAAA,EAAA3B,CAAA,IAAAoB,CAAA,GAAAd,CAAA,CAAAN,CAAA,GAAAC,CAAA,CAAA2C,OAAA,CAAAxB,CAAA,aAAAyB,oBAAA,CAAA/B,IAAA,CAAAhB,CAAA,EAAAsB,CAAA,MAAAL,CAAA,CAAAK,CAAA,IAAAtB,CAAA,CAAAsB,CAAA,aAAAL,CAAA;AAAA,SAAA4B,8BAAA3C,CAAA,EAAAF,CAAA,gBAAAE,CAAA,iBAAAC,CAAA,gBAAAK,CAAA,IAAAN,CAAA,SAAAa,cAAA,CAAAC,IAAA,CAAAd,CAAA,EAAAM,CAAA,SAAAR,CAAA,CAAA8C,OAAA,CAAAtC,CAAA,kBAAAL,CAAA,CAAAK,CAAA,IAAAN,CAAA,CAAAM,CAAA,YAAAL,CAAA;AAaA,SAAS6C,WAAWA,CAACC,KAAK,EAAEC,GAAG,EAAE;EAC/B,MAAMC,QAAQ,GAAGC,eAAC,CAACC,SAAS,CAACJ,KAAK,CAAC;EACnC;EACAE,QAAQ,CAACG,MAAM,GAAG;IAAEC,GAAG,EAAE,CAAC,IAAI,EAAE,GAAGL,GAAG;EAAE,CAAC;EACzC,OAAOC,QAAQ;AACjB;AAEA,SAASK,UAAUA,CAACP,KAAK,EAAEC,GAAG,EAAE;EAC9B,MAAMC,QAAQ,GAAGC,eAAC,CAACC,SAAS,CAACJ,KAAK,CAAC;EACnC;EACAE,QAAQ,CAACM,MAAM,GAAG;IAAEF,GAAG,EAAE,CAAC,IAAI,EAAE,GAAG,EAAE,GAAGL,GAAG;EAAE,CAAC;EAC9C,OAAOC,QAAQ;AACjB;;AAEA;AACA,MAAMO,kBAAkB,GAAGC,IAAA,IAAwB;EAAA,IAAvB;MAAEC;IAAe,CAAC,GAAAD,IAAA;IAARE,MAAM,GAAAjB,wBAAA,CAAAe,IAAA,EAAA9D,SAAA;EAC1C,IAAI,CAAC+D,GAAG,EAAE;IACR,OAAOC,MAAM;EACf;EAEAA,MAAM,CAACP,MAAM,GAAG,EAAE;EAClBO,MAAM,CAACJ,MAAM,GAAG,EAAE;EAElB,KAAK,MAAMK,KAAK,IAAIF,GAAG,EAAE;IACvB,IAAIA,GAAG,CAACE,KAAK,CAAC,CAACC,IAAI,EAAE;MACnBF,MAAM,CAACJ,MAAM,CAAChC,IAAI,CAACqC,KAAK,CAAC;IAC3B;IACA,IAAIF,GAAG,CAACE,KAAK,CAAC,CAACE,KAAK,EAAE;MACpBH,MAAM,CAACP,MAAM,CAAC7B,IAAI,CAACqC,KAAK,CAAC;IAC3B;EACF;EACA,OAAOD,MAAM;AACf,CAAC;AAED,MAAMI,gBAAgB,GAAG,CAAC,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,CAAC;AACpE,MAAMC,sBAAsB,GAAG,CAC7B,GAAGD,gBAAgB,EACnB,qBAAqB,EACrB,mBAAmB,EACnB,YAAY,EACZ,gCAAgC,EAChC,qBAAqB,EACrB,6BAA6B,EAC7B,sBAAsB,EACtB,mBAAmB,CACpB;AAED,MAAME,aAAa,GAAGA,CACpBlB,KAAU,EACVmB,QAAiB,EACjBC,aAAsB,EACtBC,MAAe,KACN;EACT,IAAID,aAAa,EAAE;IACjBD,QAAQ,GAAG,IAAI;EACjB;EACA,IAAInB,KAAK,CAACW,GAAG,EAAE;IACb,MAAM,IAAIW,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACC,aAAa,EAAE,sBAAsB,CAAC;EAC1E;EAEA,IAAIxB,KAAK,CAACyB,GAAG,EAAE;IACb,IAAIzB,KAAK,CAACyB,GAAG,YAAYC,KAAK,EAAE;MAC9B1B,KAAK,CAACyB,GAAG,CAAC5C,OAAO,CAACK,KAAK,IAAIgC,aAAa,CAAChC,KAAK,EAAEiC,QAAQ,EAAEC,aAAa,EAAEC,MAAM,CAAC,CAAC;IACnF,CAAC,MAAM;MACL,MAAM,IAAIC,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACC,aAAa,EAAE,sCAAsC,CAAC;IAC1F;EACF;EAEA,IAAIxB,KAAK,CAAC2B,IAAI,EAAE;IACd,IAAI3B,KAAK,CAAC2B,IAAI,YAAYD,KAAK,EAAE;MAC/B1B,KAAK,CAAC2B,IAAI,CAAC9C,OAAO,CAACK,KAAK,IAAIgC,aAAa,CAAChC,KAAK,EAAEiC,QAAQ,EAAEC,aAAa,EAAEC,MAAM,CAAC,CAAC;IACpF,CAAC,MAAM;MACL,MAAM,IAAIC,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACC,aAAa,EAAE,uCAAuC,CAAC;IAC3F;EACF;EAEA,IAAIxB,KAAK,CAAC4B,IAAI,EAAE;IACd,IAAI5B,KAAK,CAAC4B,IAAI,YAAYF,KAAK,IAAI1B,KAAK,CAAC4B,IAAI,CAAChD,MAAM,GAAG,CAAC,EAAE;MACxDoB,KAAK,CAAC4B,IAAI,CAAC/C,OAAO,CAACK,KAAK,IAAIgC,aAAa,CAAChC,KAAK,EAAEiC,QAAQ,EAAEC,aAAa,EAAEC,MAAM,CAAC,CAAC;IACpF,CAAC,MAAM;MACL,MAAM,IAAIC,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACC,aAAa,EACzB,qDACF,CAAC;IACH;EACF;EAEA9D,MAAM,CAACS,IAAI,CAAC6B,KAAK,CAAC,CAACnB,OAAO,CAACgD,GAAG,IAAI;IAChC,IAAI7B,KAAK,IAAIA,KAAK,CAAC6B,GAAG,CAAC,IAAI7B,KAAK,CAAC6B,GAAG,CAAC,CAACC,MAAM,EAAE;MAC5C,IAAI,OAAO9B,KAAK,CAAC6B,GAAG,CAAC,CAACE,QAAQ,KAAK,QAAQ,EAAE;QAC3C,IAAI,CAAC/B,KAAK,CAAC6B,GAAG,CAAC,CAACE,QAAQ,CAACC,KAAK,CAAC,WAAW,CAAC,EAAE;UAC3C,MAAM,IAAIV,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACC,aAAa,EACzB,iCAAiCxB,KAAK,CAAC6B,GAAG,CAAC,CAACE,QAAQ,EACtD,CAAC;QACH;MACF;IACF;IACA,IACE,CAACF,GAAG,CAACG,KAAK,CAAC,2BAA2B,CAAC,KACrC,CAAChB,gBAAgB,CAACiB,QAAQ,CAACJ,GAAG,CAAC,IAAI,CAACV,QAAQ,IAAI,CAACE,MAAM,IACtDA,MAAM,IAAIF,QAAQ,IAAI,CAACF,sBAAsB,CAACgB,QAAQ,CAACJ,GAAG,CAAE,CAAC,EAChE;MACA,MAAM,IAAIP,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAAE,qBAAqBL,GAAG,EAAE,CAAC;IACjF;EACF,CAAC,CAAC;AACJ,CAAC;;AAED;AACA,MAAMM,mBAAmB,GAAGA,CAC1BhB,QAAiB,EACjBC,aAAsB,EACtBgB,QAAe,EACfC,IAAS,EACTC,SAAc,EACdC,MAA+C,EAC/CC,SAAiB,EACjBC,eAAkC,EAClCC,MAAW,KACR;EACH,IAAIC,MAAM,GAAG,IAAI;EACjB,IAAIN,IAAI,IAAIA,IAAI,CAACO,IAAI,EAAED,MAAM,GAAGN,IAAI,CAACO,IAAI,CAACC,EAAE;;EAE5C;EACA,MAAMC,KAAK,GACTP,MAAM,IAAIA,MAAM,CAACQ,wBAAwB,GAAGR,MAAM,CAACQ,wBAAwB,CAACP,SAAS,CAAC,GAAG,CAAC,CAAC;EAC7F,IAAIM,KAAK,EAAE;IACT,MAAME,eAAe,GAAG,CAAC,KAAK,EAAE,MAAM,CAAC,CAACnD,OAAO,CAACyC,SAAS,CAAC,GAAG,CAAC,CAAC;IAE/D,IAAIU,eAAe,IAAIF,KAAK,CAACL,eAAe,EAAE;MAC5C;MACA,MAAMQ,0BAA0B,GAAGvF,MAAM,CAACS,IAAI,CAAC2E,KAAK,CAACL,eAAe,CAAC,CAClEnE,MAAM,CAACuD,GAAG,IAAIA,GAAG,CAACqB,UAAU,CAAC,YAAY,CAAC,CAAC,CAC3CC,GAAG,CAACtB,GAAG,IAAI;QACV,OAAO;UAAEA,GAAG,EAAEA,GAAG,CAACuB,SAAS,CAAC,EAAE,CAAC;UAAElE,KAAK,EAAE4D,KAAK,CAACL,eAAe,CAACZ,GAAG;QAAE,CAAC;MACtE,CAAC,CAAC;MAEJ,MAAMwB,kBAAmC,GAAG,EAAE;MAC9C,IAAIC,uBAAuB,GAAG,KAAK;;MAEnC;MACAL,0BAA0B,CAACpE,OAAO,CAAC0E,WAAW,IAAI;QAChD,IAAIC,uBAAuB,GAAG,KAAK;QACnC,MAAMC,kBAAkB,GAAGf,MAAM,CAACa,WAAW,CAAC1B,GAAG,CAAC;QAClD,IAAI4B,kBAAkB,EAAE;UACtB,IAAI/B,KAAK,CAACgC,OAAO,CAACD,kBAAkB,CAAC,EAAE;YACrCD,uBAAuB,GAAGC,kBAAkB,CAACE,IAAI,CAC/Cf,IAAI,IAAIA,IAAI,CAACgB,QAAQ,IAAIhB,IAAI,CAACgB,QAAQ,KAAKjB,MAC7C,CAAC;UACH,CAAC,MAAM;YACLa,uBAAuB,GACrBC,kBAAkB,CAACG,QAAQ,IAAIH,kBAAkB,CAACG,QAAQ,KAAKjB,MAAM;UACzE;QACF;QAEA,IAAIa,uBAAuB,EAAE;UAC3BF,uBAAuB,GAAG,IAAI;UAC9BD,kBAAkB,CAAC7E,IAAI,CAAC+E,WAAW,CAACrE,KAAK,CAAC;QAC5C;MACF,CAAC,CAAC;;MAEF;MACA;MACA;MACA,IAAIoE,uBAAuB,IAAIb,eAAe,EAAE;QAC9CY,kBAAkB,CAAC7E,IAAI,CAACiE,eAAe,CAAC;MAC1C;MACA;MACAY,kBAAkB,CAACxE,OAAO,CAACgF,MAAM,IAAI;QACnC,IAAIA,MAAM,EAAE;UACV;UACA;UACA,IAAI,CAACpB,eAAe,EAAE;YACpBA,eAAe,GAAGoB,MAAM;UAC1B,CAAC,MAAM;YACLpB,eAAe,GAAGA,eAAe,CAACnE,MAAM,CAACwF,CAAC,IAAID,MAAM,CAAC5B,QAAQ,CAAC6B,CAAC,CAAC,CAAC;UACnE;QACF;MACF,CAAC,CAAC;IACJ;EACF;EAEA,MAAMC,WAAW,GAAGvB,SAAS,KAAK,OAAO;EACzC,IAAIuB,WAAW,EAAE;IACfrB,MAAM,CAACsB,QAAQ,GAAGtB,MAAM,CAACuB,gBAAgB;IACzC,OAAOvB,MAAM,CAACuB,gBAAgB;IAC9B,OAAOvB,MAAM,CAACwB,YAAY;EAC5B;EAEA,IAAI9C,aAAa,EAAE;IACjB,OAAOsB,MAAM;EACf;;EAEA;AACF;EACE,IAAI,EAAEqB,WAAW,IAAIpB,MAAM,IAAID,MAAM,CAACkB,QAAQ,KAAKjB,MAAM,CAAC,EAAE;IAAA,IAAAwB,qBAAA;IAC1D1B,eAAe,IAAIA,eAAe,CAAC5D,OAAO,CAACuF,CAAC,IAAI,OAAO1B,MAAM,CAAC0B,CAAC,CAAC,CAAC;;IAEjE;IACA;IACAtB,KAAK,aAALA,KAAK,gBAAAqB,qBAAA,GAALrB,KAAK,CAAEL,eAAe,cAAA0B,qBAAA,gBAAAA,qBAAA,GAAtBA,qBAAA,CAAwBE,aAAa,cAAAF,qBAAA,eAArCA,qBAAA,CAAuCtF,OAAO,CAACuF,CAAC,IAAI,OAAO1B,MAAM,CAAC0B,CAAC,CAAC,CAAC;EACvE;EAEA,KAAK,MAAMvC,GAAG,IAAIa,MAAM,EAAE;IACxB,IAAIb,GAAG,CAACyC,MAAM,CAAC,CAAC,CAAC,KAAK,GAAG,EAAE;MACzB,OAAO5B,MAAM,CAACb,GAAG,CAAC;IACpB;EACF;EAEA,IAAI,CAACkC,WAAW,IAAI5C,QAAQ,EAAE;IAC5B,OAAOuB,MAAM;EACf;EAEA,IAAIN,QAAQ,CAACvC,OAAO,CAAC6C,MAAM,CAACkB,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE;IAC1C,OAAOlB,MAAM;EACf;EACA,OAAOA,MAAM,CAAC6B,QAAQ;EACtB,OAAO7B,MAAM;AACf,CAAC;;AAED;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM8B,oBAAoB,GAAG,CAC3B,kBAAkB,EAClB,mBAAmB,EACnB,qBAAqB,EACrB,gCAAgC,EAChC,6BAA6B,EAC7B,qBAAqB,EACrB,8BAA8B,EAC9B,sBAAsB,EACtB,mBAAmB,CACpB;AAED,MAAMC,kBAAkB,GAAG5C,GAAG,IAAI;EAChC,OAAO2C,oBAAoB,CAAC3E,OAAO,CAACgC,GAAG,CAAC,IAAI,CAAC;AAC/C,CAAC;AAED,SAAS6C,aAAaA,CAAClC,SAAS,EAAEX,GAAG,EAAE;EACrC,OAAO,SAASA,GAAG,IAAIW,SAAS,EAAE;AACpC;AAEA,MAAMmC,+BAA+B,GAAGjC,MAAM,IAAI;EAChD,KAAK,MAAMb,GAAG,IAAIa,MAAM,EAAE;IACxB,IAAIA,MAAM,CAACb,GAAG,CAAC,IAAIa,MAAM,CAACb,GAAG,CAAC,CAAC+C,IAAI,EAAE;MACnC,QAAQlC,MAAM,CAACb,GAAG,CAAC,CAAC+C,IAAI;QACtB,KAAK,WAAW;UACd,IAAI,OAAOlC,MAAM,CAACb,GAAG,CAAC,CAACgD,MAAM,KAAK,QAAQ,EAAE;YAC1C,MAAM,IAAIvD,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACuD,YAAY,EAAE,iCAAiC,CAAC;UACpF;UACApC,MAAM,CAACb,GAAG,CAAC,GAAGa,MAAM,CAACb,GAAG,CAAC,CAACgD,MAAM;UAChC;QACF,KAAK,aAAa;UAChBnC,MAAM,CAACb,GAAG,CAAC,GAAGa,MAAM,CAACb,GAAG,CAAC,CAACgD,MAAM;UAChC;QACF,KAAK,KAAK;UACR,IAAI,EAAEnC,MAAM,CAACb,GAAG,CAAC,CAACkD,OAAO,YAAYrD,KAAK,CAAC,EAAE;YAC3C,MAAM,IAAIJ,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACuD,YAAY,EAAE,iCAAiC,CAAC;UACpF;UACApC,MAAM,CAACb,GAAG,CAAC,GAAGa,MAAM,CAACb,GAAG,CAAC,CAACkD,OAAO;UACjC;QACF,KAAK,WAAW;UACd,IAAI,EAAErC,MAAM,CAACb,GAAG,CAAC,CAACkD,OAAO,YAAYrD,KAAK,CAAC,EAAE;YAC3C,MAAM,IAAIJ,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACuD,YAAY,EAAE,iCAAiC,CAAC;UACpF;UACApC,MAAM,CAACb,GAAG,CAAC,GAAGa,MAAM,CAACb,GAAG,CAAC,CAACkD,OAAO;UACjC;QACF,KAAK,QAAQ;UACX,IAAI,EAAErC,MAAM,CAACb,GAAG,CAAC,CAACkD,OAAO,YAAYrD,KAAK,CAAC,EAAE;YAC3C,MAAM,IAAIJ,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACuD,YAAY,EAAE,iCAAiC,CAAC;UACpF;UACApC,MAAM,CAACb,GAAG,CAAC,GAAG,EAAE;UAChB;QACF,KAAK,QAAQ;UACX,OAAOa,MAAM,CAACb,GAAG,CAAC;UAClB;QACF;UACE,MAAM,IAAIP,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACyD,mBAAmB,EAC/B,OAAOtC,MAAM,CAACb,GAAG,CAAC,CAAC+C,IAAI,iCACzB,CAAC;MACL;IACF;EACF;AACF,CAAC;AAED,MAAMK,iBAAiB,GAAGA,CAACzC,SAAS,EAAEE,MAAM,EAAEH,MAAM,KAAK;EACvD,IAAIG,MAAM,CAAC6B,QAAQ,IAAI/B,SAAS,KAAK,OAAO,EAAE;IAC5C9E,MAAM,CAACS,IAAI,CAACuE,MAAM,CAAC6B,QAAQ,CAAC,CAAC1F,OAAO,CAACqG,QAAQ,IAAI;MAC/C,MAAMC,YAAY,GAAGzC,MAAM,CAAC6B,QAAQ,CAACW,QAAQ,CAAC;MAC9C,MAAME,SAAS,GAAG,cAAcF,QAAQ,EAAE;MAC1C,IAAIC,YAAY,IAAI,IAAI,EAAE;QACxBzC,MAAM,CAAC0C,SAAS,CAAC,GAAG;UAClBR,IAAI,EAAE;QACR,CAAC;MACH,CAAC,MAAM;QACLlC,MAAM,CAAC0C,SAAS,CAAC,GAAGD,YAAY;QAChC5C,MAAM,CAACsB,MAAM,CAACuB,SAAS,CAAC,GAAG;UAAEC,IAAI,EAAE;QAAS,CAAC;MAC/C;IACF,CAAC,CAAC;IACF,OAAO3C,MAAM,CAAC6B,QAAQ;EACxB;AACF,CAAC;AACD;AACA,MAAMe,oBAAoB,GAAGC,KAAA,IAAmC;EAAA,IAAlC;MAAE/E,MAAM;MAAEH;IAAkB,CAAC,GAAAkF,KAAA;IAARC,MAAM,GAAA7F,wBAAA,CAAA4F,KAAA,EAAA1I,UAAA;EACvD,IAAI2D,MAAM,IAAIH,MAAM,EAAE;IACpBmF,MAAM,CAAC7E,GAAG,GAAG,CAAC,CAAC;IAEf,CAACH,MAAM,IAAI,EAAE,EAAE3B,OAAO,CAACgC,KAAK,IAAI;MAC9B,IAAI,CAAC2E,MAAM,CAAC7E,GAAG,CAACE,KAAK,CAAC,EAAE;QACtB2E,MAAM,CAAC7E,GAAG,CAACE,KAAK,CAAC,GAAG;UAAEC,IAAI,EAAE;QAAK,CAAC;MACpC,CAAC,MAAM;QACL0E,MAAM,CAAC7E,GAAG,CAACE,KAAK,CAAC,CAAC,MAAM,CAAC,GAAG,IAAI;MAClC;IACF,CAAC,CAAC;IAEF,CAACR,MAAM,IAAI,EAAE,EAAExB,OAAO,CAACgC,KAAK,IAAI;MAC9B,IAAI,CAAC2E,MAAM,CAAC7E,GAAG,CAACE,KAAK,CAAC,EAAE;QACtB2E,MAAM,CAAC7E,GAAG,CAACE,KAAK,CAAC,GAAG;UAAEE,KAAK,EAAE;QAAK,CAAC;MACrC,CAAC,MAAM;QACLyE,MAAM,CAAC7E,GAAG,CAACE,KAAK,CAAC,CAAC,OAAO,CAAC,GAAG,IAAI;MACnC;IACF,CAAC,CAAC;EACJ;EACA,OAAO2E,MAAM;AACf,CAAC;;AAED;AACA;AACA;AACA;AACA;AACA;AACA,MAAMC,gBAAgB,GAAIL,SAAiB,IAAa;EACtD,OAAOA,SAAS,CAACM,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;AAChC,CAAC;AAED,MAAMC,cAAc,GAAG;EACrB9B,MAAM,EAAE;IAAE+B,SAAS,EAAE;MAAEP,IAAI,EAAE;IAAS,CAAC;IAAEQ,QAAQ,EAAE;MAAER,IAAI,EAAE;IAAS;EAAE;AACxE,CAAC;AAED,MAAMS,uBAAuB,GAAGA,CAACpD,MAAM,EAAEF,SAAS,EAAEuD,OAAO,KAAK;EAC9D,IAAIvD,SAAS,KAAK,OAAO,IAAIuD,OAAO,CAACD,uBAAuB,EAAE;IAC5D,IAAI,OAAOpD,MAAM,CAAC,OAAO,CAAC,KAAK,QAAQ,EAAE;MACvCA,MAAM,CAAC,OAAO,CAAC,GAAGA,MAAM,CAAC,OAAO,CAAC,CAACsD,WAAW,CAAC,CAAC;IACjD;EACF;AACF,CAAC;AAED,MAAMC,0BAA0B,GAAGA,CAACvD,MAAM,EAAEF,SAAS,EAAEuD,OAAO,KAAK;EACjE,IAAIvD,SAAS,KAAK,OAAO,IAAIuD,OAAO,CAACE,0BAA0B,EAAE;IAC/D,IAAI,OAAOvD,MAAM,CAAC,UAAU,CAAC,KAAK,QAAQ,EAAE;MAC1CA,MAAM,CAAC,UAAU,CAAC,GAAGA,MAAM,CAAC,UAAU,CAAC,CAACsD,WAAW,CAAC,CAAC;IACvD;EACF;AACF,CAAC;AAED,MAAME,kBAAkB,CAAC;EAQvBC,WAAWA,CAACC,OAAuB,EAAEL,OAA2B,EAAE;IAChE,IAAI,CAACK,OAAO,GAAGA,OAAO;IACtB,IAAI,CAACL,OAAO,GAAGA,OAAO,IAAI,CAAC,CAAC;IAC5B,IAAI,CAACM,kBAAkB,GAAG,IAAI,CAACN,OAAO,CAACM,kBAAkB,IAAI,CAAC,CAAC;IAC/D;IACA;IACA,IAAI,CAACC,aAAa,GAAG,IAAI;IACzB,IAAI,CAACC,qBAAqB,GAAG,IAAI;IACjC,IAAI,CAACR,OAAO,GAAGA,OAAO;EACxB;EAEAS,gBAAgBA,CAAChE,SAAiB,EAAoB;IACpD,OAAO,IAAI,CAAC4D,OAAO,CAACK,WAAW,CAACjE,SAAS,CAAC;EAC5C;EAEAkE,eAAeA,CAAClE,SAAiB,EAAiB;IAChD,OAAO,IAAI,CAACmE,UAAU,CAAC,CAAC,CACrBC,IAAI,CAACC,gBAAgB,IAAIA,gBAAgB,CAACC,YAAY,CAACtE,SAAS,CAAC,CAAC,CAClEoE,IAAI,CAACrE,MAAM,IAAI,IAAI,CAAC6D,OAAO,CAACW,oBAAoB,CAACvE,SAAS,EAAED,MAAM,EAAE,CAAC,CAAC,CAAC,CAAC;EAC7E;EAEAyE,iBAAiBA,CAACxE,SAAiB,EAAiB;IAClD,IAAI,CAAClG,gBAAgB,CAAC2K,gBAAgB,CAACzE,SAAS,CAAC,EAAE;MACjD,OAAO0E,OAAO,CAACC,MAAM,CACnB,IAAI7F,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAAC6F,kBAAkB,EAAE,qBAAqB,GAAG5E,SAAS,CACnF,CAAC;IACH;IACA,OAAO0E,OAAO,CAACG,OAAO,CAAC,CAAC;EAC1B;;EAEA;EACAV,UAAUA,CACRZ,OAA0B,GAAG;IAAEuB,UAAU,EAAE;EAAM,CAAC,EACN;IAC5C,IAAI,IAAI,CAAChB,aAAa,IAAI,IAAI,EAAE;MAC9B,OAAO,IAAI,CAACA,aAAa;IAC3B;IACA,IAAI,CAACA,aAAa,GAAGhK,gBAAgB,CAACiL,IAAI,CAAC,IAAI,CAACnB,OAAO,EAAEL,OAAO,CAAC;IACjE,IAAI,CAACO,aAAa,CAACM,IAAI,CACrB,MAAM,OAAO,IAAI,CAACN,aAAa,EAC/B,MAAM,OAAO,IAAI,CAACA,aACpB,CAAC;IACD,OAAO,IAAI,CAACK,UAAU,CAACZ,OAAO,CAAC;EACjC;EAEAyB,kBAAkBA,CAChBX,gBAAmD,EACnDd,OAA0B,GAAG;IAAEuB,UAAU,EAAE;EAAM,CAAC,EACN;IAC5C,OAAOT,gBAAgB,GAAGK,OAAO,CAACG,OAAO,CAACR,gBAAgB,CAAC,GAAG,IAAI,CAACF,UAAU,CAACZ,OAAO,CAAC;EACxF;;EAEA;EACA;EACA;EACA0B,uBAAuBA,CAACjF,SAAiB,EAAEX,GAAW,EAAoB;IACxE,OAAO,IAAI,CAAC8E,UAAU,CAAC,CAAC,CAACC,IAAI,CAACrE,MAAM,IAAI;MACtC,IAAIrF,CAAC,GAAGqF,MAAM,CAACmF,eAAe,CAAClF,SAAS,EAAEX,GAAG,CAAC;MAC9C,IAAI3E,CAAC,IAAI,IAAI,IAAI,OAAOA,CAAC,KAAK,QAAQ,IAAIA,CAAC,CAACmI,IAAI,KAAK,UAAU,EAAE;QAC/D,OAAOnI,CAAC,CAACyK,WAAW;MACtB;MACA,OAAOnF,SAAS;IAClB,CAAC,CAAC;EACJ;;EAEA;EACA;EACA;EACA;EACAoF,cAAcA,CACZpF,SAAiB,EACjBE,MAAW,EACX1C,KAAU,EACV6H,UAAwB,EACxBC,WAAoB,EACF;IAClB,IAAIvF,MAAM;IACV,MAAMtC,GAAG,GAAG4H,UAAU,CAAC5H,GAAG;IAC1B,MAAMkB,QAAQ,GAAGlB,GAAG,KAAK8H,SAAS;IAClC,IAAI3F,QAAkB,GAAGnC,GAAG,IAAI,EAAE;IAClC,OAAO,IAAI,CAAC0G,UAAU,CAAC,CAAC,CACrBC,IAAI,CAACoB,CAAC,IAAI;MACTzF,MAAM,GAAGyF,CAAC;MACV,IAAI7G,QAAQ,EAAE;QACZ,OAAO+F,OAAO,CAACG,OAAO,CAAC,CAAC;MAC1B;MACA,OAAO,IAAI,CAACY,WAAW,CAAC1F,MAAM,EAAEC,SAAS,EAAEE,MAAM,EAAEN,QAAQ,EAAEyF,UAAU,CAAC;IAC1E,CAAC,CAAC,CACDjB,IAAI,CAAC,MAAM;MACV,OAAOrE,MAAM,CAACqF,cAAc,CAACpF,SAAS,EAAEE,MAAM,EAAE1C,KAAK,EAAE8H,WAAW,CAAC;IACrE,CAAC,CAAC;EACN;EAEAzG,MAAMA,CACJmB,SAAiB,EACjBxC,KAAU,EACVqB,MAAW,EACX;IAAEpB,GAAG;IAAEiI,IAAI;IAAEC,MAAM;IAAEC;EAA4B,CAAC,GAAG,CAAC,CAAC,EACvDC,gBAAyB,GAAG,KAAK,EACjCC,YAAqB,GAAG,KAAK,EAC7BC,qBAAwD,EAC1C;IACd,IAAI;MACFC,cAAK,CAACC,uBAAuB,CAAC,IAAI,CAAC1C,OAAO,EAAE1E,MAAM,CAAC;IACrD,CAAC,CAAC,OAAOqH,KAAK,EAAE;MACd,OAAOxB,OAAO,CAACC,MAAM,CAAC,IAAI7F,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAAEwG,KAAK,CAAC,CAAC;IAC7E;IACA,MAAMC,aAAa,GAAG3I,KAAK;IAC3B,MAAM4I,cAAc,GAAGvH,MAAM;IAC7B;IACAA,MAAM,GAAG,IAAAwH,iBAAQ,EAACxH,MAAM,CAAC;IACzB,IAAIyH,eAAe,GAAG,EAAE;IACxB,IAAI3H,QAAQ,GAAGlB,GAAG,KAAK8H,SAAS;IAChC,IAAI3F,QAAQ,GAAGnC,GAAG,IAAI,EAAE;IAExB,OAAO,IAAI,CAACuH,kBAAkB,CAACe,qBAAqB,CAAC,CAAC3B,IAAI,CAACC,gBAAgB,IAAI;MAC7E,OAAO,CAAC1F,QAAQ,GACZ+F,OAAO,CAACG,OAAO,CAAC,CAAC,GACjBR,gBAAgB,CAACkC,kBAAkB,CAACvG,SAAS,EAAEJ,QAAQ,EAAE,QAAQ,CAAC,EAEnEwE,IAAI,CAAC,MAAM;QACVkC,eAAe,GAAG,IAAI,CAACE,sBAAsB,CAACxG,SAAS,EAAEmG,aAAa,CAAC/E,QAAQ,EAAEvC,MAAM,CAAC;QACxF,IAAI,CAACF,QAAQ,EAAE;UACbnB,KAAK,GAAG,IAAI,CAACiJ,qBAAqB,CAChCpC,gBAAgB,EAChBrE,SAAS,EACT,QAAQ,EACRxC,KAAK,EACLoC,QACF,CAAC;UAED,IAAIgG,SAAS,EAAE;YACbpI,KAAK,GAAG;cACN2B,IAAI,EAAE,CACJ3B,KAAK,EACL,IAAI,CAACiJ,qBAAqB,CACxBpC,gBAAgB,EAChBrE,SAAS,EACT,UAAU,EACVxC,KAAK,EACLoC,QACF,CAAC;YAEL,CAAC;UACH;QACF;QACA,IAAI,CAACpC,KAAK,EAAE;UACV,OAAOkH,OAAO,CAACG,OAAO,CAAC,CAAC;QAC1B;QACA,IAAIpH,GAAG,EAAE;UACPD,KAAK,GAAGD,WAAW,CAACC,KAAK,EAAEC,GAAG,CAAC;QACjC;QACAiB,aAAa,CAAClB,KAAK,EAAEmB,QAAQ,EAAE,KAAK,EAAE,IAAI,CAAC;QAC3C,OAAO0F,gBAAgB,CACpBC,YAAY,CAACtE,SAAS,EAAE,IAAI,CAAC,CAC7B0G,KAAK,CAACR,KAAK,IAAI;UACd;UACA;UACA,IAAIA,KAAK,KAAKX,SAAS,EAAE;YACvB,OAAO;cAAElE,MAAM,EAAE,CAAC;YAAE,CAAC;UACvB;UACA,MAAM6E,KAAK;QACb,CAAC,CAAC,CACD9B,IAAI,CAACrE,MAAM,IAAI;UACd7E,MAAM,CAACS,IAAI,CAACkD,MAAM,CAAC,CAACxC,OAAO,CAACuG,SAAS,IAAI;YACvC,IAAIA,SAAS,CAACpD,KAAK,CAAC,iCAAiC,CAAC,EAAE;cACtD,MAAM,IAAIV,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAC5B,kCAAkCkD,SAAS,EAC7C,CAAC;YACH;YACA,MAAM+D,aAAa,GAAG1D,gBAAgB,CAACL,SAAS,CAAC;YACjD,IACE,CAAC9I,gBAAgB,CAAC8M,gBAAgB,CAACD,aAAa,EAAE3G,SAAS,CAAC,IAC5D,CAACiC,kBAAkB,CAAC0E,aAAa,CAAC,EAClC;cACA,MAAM,IAAI7H,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAC5B,kCAAkCkD,SAAS,EAC7C,CAAC;YACH;UACF,CAAC,CAAC;UACF,KAAK,MAAMiE,eAAe,IAAIhI,MAAM,EAAE;YACpC,IACEA,MAAM,CAACgI,eAAe,CAAC,IACvB,OAAOhI,MAAM,CAACgI,eAAe,CAAC,KAAK,QAAQ,IAC3C3L,MAAM,CAACS,IAAI,CAACkD,MAAM,CAACgI,eAAe,CAAC,CAAC,CAAC1F,IAAI,CACvC2F,QAAQ,IAAIA,QAAQ,CAACrH,QAAQ,CAAC,GAAG,CAAC,IAAIqH,QAAQ,CAACrH,QAAQ,CAAC,GAAG,CAC7D,CAAC,EACD;cACA,MAAM,IAAIX,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACgI,kBAAkB,EAC9B,0DACF,CAAC;YACH;UACF;UACAlI,MAAM,GAAGZ,kBAAkB,CAACY,MAAM,CAAC;UACnCyE,uBAAuB,CAACzE,MAAM,EAAEmB,SAAS,EAAE,IAAI,CAACuD,OAAO,CAAC;UACxDE,0BAA0B,CAAC5E,MAAM,EAAEmB,SAAS,EAAE,IAAI,CAACuD,OAAO,CAAC;UAC3Dd,iBAAiB,CAACzC,SAAS,EAAEnB,MAAM,EAAEkB,MAAM,CAAC;UAC5C,IAAI+F,YAAY,EAAE;YAChB,OAAO,IAAI,CAAClC,OAAO,CAACoD,IAAI,CAAChH,SAAS,EAAED,MAAM,EAAEvC,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC4G,IAAI,CAAChG,MAAM,IAAI;cACpE,IAAI,CAACA,MAAM,IAAI,CAACA,MAAM,CAAChC,MAAM,EAAE;gBAC7B,MAAM,IAAI0C,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACkI,gBAAgB,EAAE,mBAAmB,CAAC;cAC1E;cACA,OAAO,CAAC,CAAC;YACX,CAAC,CAAC;UACJ;UACA,IAAIvB,IAAI,EAAE;YACR,OAAO,IAAI,CAAC9B,OAAO,CAACsD,oBAAoB,CACtClH,SAAS,EACTD,MAAM,EACNvC,KAAK,EACLqB,MAAM,EACN,IAAI,CAACkF,qBACP,CAAC;UACH,CAAC,MAAM,IAAI4B,MAAM,EAAE;YACjB,OAAO,IAAI,CAAC/B,OAAO,CAACuD,eAAe,CACjCnH,SAAS,EACTD,MAAM,EACNvC,KAAK,EACLqB,MAAM,EACN,IAAI,CAACkF,qBACP,CAAC;UACH,CAAC,MAAM;YACL,OAAO,IAAI,CAACH,OAAO,CAACwD,gBAAgB,CAClCpH,SAAS,EACTD,MAAM,EACNvC,KAAK,EACLqB,MAAM,EACN,IAAI,CAACkF,qBACP,CAAC;UACH;QACF,CAAC,CAAC;MACN,CAAC,CAAC,CACDK,IAAI,CAAEhG,MAAW,IAAK;QACrB,IAAI,CAACA,MAAM,EAAE;UACX,MAAM,IAAIU,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACkI,gBAAgB,EAAE,mBAAmB,CAAC;QAC1E;QACA,IAAInB,YAAY,EAAE;UAChB,OAAO1H,MAAM;QACf;QACA,OAAO,IAAI,CAACiJ,qBAAqB,CAC/BrH,SAAS,EACTmG,aAAa,CAAC/E,QAAQ,EACtBvC,MAAM,EACNyH,eACF,CAAC,CAAClC,IAAI,CAAC,MAAM;UACX,OAAOhG,MAAM;QACf,CAAC,CAAC;MACJ,CAAC,CAAC,CACDgG,IAAI,CAAChG,MAAM,IAAI;QACd,IAAIyH,gBAAgB,EAAE;UACpB,OAAOnB,OAAO,CAACG,OAAO,CAACzG,MAAM,CAAC;QAChC;QACA,OAAO,IAAI,CAACkJ,uBAAuB,CAAClB,cAAc,EAAEhI,MAAM,CAAC;MAC7D,CAAC,CAAC;IACN,CAAC,CAAC;EACJ;;EAEA;EACA;EACA;EACAoI,sBAAsBA,CAACxG,SAAiB,EAAEoB,QAAiB,EAAEvC,MAAW,EAAE;IACxE,IAAI0I,GAAG,GAAG,EAAE;IACZ,IAAIC,QAAQ,GAAG,EAAE;IACjBpG,QAAQ,GAAGvC,MAAM,CAACuC,QAAQ,IAAIA,QAAQ;IAEtC,IAAIqG,OAAO,GAAGA,CAACC,EAAE,EAAErI,GAAG,KAAK;MACzB,IAAI,CAACqI,EAAE,EAAE;QACP;MACF;MACA,IAAIA,EAAE,CAACtF,IAAI,IAAI,aAAa,EAAE;QAC5BmF,GAAG,CAACvL,IAAI,CAAC;UAAEqD,GAAG;UAAEqI;QAAG,CAAC,CAAC;QACrBF,QAAQ,CAACxL,IAAI,CAACqD,GAAG,CAAC;MACpB;MAEA,IAAIqI,EAAE,CAACtF,IAAI,IAAI,gBAAgB,EAAE;QAC/BmF,GAAG,CAACvL,IAAI,CAAC;UAAEqD,GAAG;UAAEqI;QAAG,CAAC,CAAC;QACrBF,QAAQ,CAACxL,IAAI,CAACqD,GAAG,CAAC;MACpB;MAEA,IAAIqI,EAAE,CAACtF,IAAI,IAAI,OAAO,EAAE;QACtB,KAAK,IAAIuF,CAAC,IAAID,EAAE,CAACH,GAAG,EAAE;UACpBE,OAAO,CAACE,CAAC,EAAEtI,GAAG,CAAC;QACjB;MACF;IACF,CAAC;IAED,KAAK,MAAMA,GAAG,IAAIR,MAAM,EAAE;MACxB4I,OAAO,CAAC5I,MAAM,CAACQ,GAAG,CAAC,EAAEA,GAAG,CAAC;IAC3B;IACA,KAAK,MAAMA,GAAG,IAAImI,QAAQ,EAAE;MAC1B,OAAO3I,MAAM,CAACQ,GAAG,CAAC;IACpB;IACA,OAAOkI,GAAG;EACZ;;EAEA;EACA;EACAF,qBAAqBA,CAACrH,SAAiB,EAAEoB,QAAgB,EAAEvC,MAAW,EAAE0I,GAAQ,EAAE;IAChF,IAAIK,OAAO,GAAG,EAAE;IAChBxG,QAAQ,GAAGvC,MAAM,CAACuC,QAAQ,IAAIA,QAAQ;IACtCmG,GAAG,CAAClL,OAAO,CAAC,CAAC;MAAEgD,GAAG;MAAEqI;IAAG,CAAC,KAAK;MAC3B,IAAI,CAACA,EAAE,EAAE;QACP;MACF;MACA,IAAIA,EAAE,CAACtF,IAAI,IAAI,aAAa,EAAE;QAC5B,KAAK,MAAMlC,MAAM,IAAIwH,EAAE,CAACnF,OAAO,EAAE;UAC/BqF,OAAO,CAAC5L,IAAI,CAAC,IAAI,CAAC6L,WAAW,CAACxI,GAAG,EAAEW,SAAS,EAAEoB,QAAQ,EAAElB,MAAM,CAACkB,QAAQ,CAAC,CAAC;QAC3E;MACF;MAEA,IAAIsG,EAAE,CAACtF,IAAI,IAAI,gBAAgB,EAAE;QAC/B,KAAK,MAAMlC,MAAM,IAAIwH,EAAE,CAACnF,OAAO,EAAE;UAC/BqF,OAAO,CAAC5L,IAAI,CAAC,IAAI,CAAC8L,cAAc,CAACzI,GAAG,EAAEW,SAAS,EAAEoB,QAAQ,EAAElB,MAAM,CAACkB,QAAQ,CAAC,CAAC;QAC9E;MACF;IACF,CAAC,CAAC;IAEF,OAAOsD,OAAO,CAACqD,GAAG,CAACH,OAAO,CAAC;EAC7B;;EAEA;EACA;EACAC,WAAWA,CAACxI,GAAW,EAAE2I,aAAqB,EAAEC,MAAc,EAAEC,IAAY,EAAE;IAC5E,MAAMC,GAAG,GAAG;MACV/E,SAAS,EAAE8E,IAAI;MACf7E,QAAQ,EAAE4E;IACZ,CAAC;IACD,OAAO,IAAI,CAACrE,OAAO,CAACuD,eAAe,CACjC,SAAS9H,GAAG,IAAI2I,aAAa,EAAE,EAC/B7E,cAAc,EACdgF,GAAG,EACHA,GAAG,EACH,IAAI,CAACpE,qBACP,CAAC;EACH;;EAEA;EACA;EACA;EACA+D,cAAcA,CAACzI,GAAW,EAAE2I,aAAqB,EAAEC,MAAc,EAAEC,IAAY,EAAE;IAC/E,IAAIC,GAAG,GAAG;MACR/E,SAAS,EAAE8E,IAAI;MACf7E,QAAQ,EAAE4E;IACZ,CAAC;IACD,OAAO,IAAI,CAACrE,OAAO,CAChBW,oBAAoB,CACnB,SAASlF,GAAG,IAAI2I,aAAa,EAAE,EAC/B7E,cAAc,EACdgF,GAAG,EACH,IAAI,CAACpE,qBACP,CAAC,CACA2C,KAAK,CAACR,KAAK,IAAI;MACd;MACA,IAAIA,KAAK,CAACkC,IAAI,IAAItJ,WAAK,CAACC,KAAK,CAACkI,gBAAgB,EAAE;QAC9C;MACF;MACA,MAAMf,KAAK;IACb,CAAC,CAAC;EACN;;EAEA;EACA;EACA;EACA;EACA;EACA;EACA;EACAmC,OAAOA,CACLrI,SAAiB,EACjBxC,KAAU,EACV;IAAEC;EAAkB,CAAC,GAAG,CAAC,CAAC,EAC1BsI,qBAAwD,EAC1C;IACd,MAAMpH,QAAQ,GAAGlB,GAAG,KAAK8H,SAAS;IAClC,MAAM3F,QAAQ,GAAGnC,GAAG,IAAI,EAAE;IAE1B,OAAO,IAAI,CAACuH,kBAAkB,CAACe,qBAAqB,CAAC,CAAC3B,IAAI,CAACC,gBAAgB,IAAI;MAC7E,OAAO,CAAC1F,QAAQ,GACZ+F,OAAO,CAACG,OAAO,CAAC,CAAC,GACjBR,gBAAgB,CAACkC,kBAAkB,CAACvG,SAAS,EAAEJ,QAAQ,EAAE,QAAQ,CAAC,EACpEwE,IAAI,CAAC,MAAM;QACX,IAAI,CAACzF,QAAQ,EAAE;UACbnB,KAAK,GAAG,IAAI,CAACiJ,qBAAqB,CAChCpC,gBAAgB,EAChBrE,SAAS,EACT,QAAQ,EACRxC,KAAK,EACLoC,QACF,CAAC;UACD,IAAI,CAACpC,KAAK,EAAE;YACV,MAAM,IAAIsB,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACkI,gBAAgB,EAAE,mBAAmB,CAAC;UAC1E;QACF;QACA;QACA,IAAIxJ,GAAG,EAAE;UACPD,KAAK,GAAGD,WAAW,CAACC,KAAK,EAAEC,GAAG,CAAC;QACjC;QACAiB,aAAa,CAAClB,KAAK,EAAEmB,QAAQ,EAAE,KAAK,EAAE,KAAK,CAAC;QAC5C,OAAO0F,gBAAgB,CACpBC,YAAY,CAACtE,SAAS,CAAC,CACvB0G,KAAK,CAACR,KAAK,IAAI;UACd;UACA;UACA,IAAIA,KAAK,KAAKX,SAAS,EAAE;YACvB,OAAO;cAAElE,MAAM,EAAE,CAAC;YAAE,CAAC;UACvB;UACA,MAAM6E,KAAK;QACb,CAAC,CAAC,CACD9B,IAAI,CAACkE,iBAAiB,IACrB,IAAI,CAAC1E,OAAO,CAACW,oBAAoB,CAC/BvE,SAAS,EACTsI,iBAAiB,EACjB9K,KAAK,EACL,IAAI,CAACuG,qBACP,CACF,CAAC,CACA2C,KAAK,CAACR,KAAK,IAAI;UACd;UACA,IAAIlG,SAAS,KAAK,UAAU,IAAIkG,KAAK,CAACkC,IAAI,KAAKtJ,WAAK,CAACC,KAAK,CAACkI,gBAAgB,EAAE;YAC3E,OAAOvC,OAAO,CAACG,OAAO,CAAC,CAAC,CAAC,CAAC;UAC5B;UACA,MAAMqB,KAAK;QACb,CAAC,CAAC;MACN,CAAC,CAAC;IACJ,CAAC,CAAC;EACJ;;EAEA;EACA;EACAqC,MAAMA,CACJvI,SAAiB,EACjBE,MAAW,EACX;IAAEzC;EAAkB,CAAC,GAAG,CAAC,CAAC,EAC1BqI,YAAqB,GAAG,KAAK,EAC7BC,qBAAwD,EAC1C;IACd,IAAI;MACFC,cAAK,CAACC,uBAAuB,CAAC,IAAI,CAAC1C,OAAO,EAAErD,MAAM,CAAC;IACrD,CAAC,CAAC,OAAOgG,KAAK,EAAE;MACd,OAAOxB,OAAO,CAACC,MAAM,CAAC,IAAI7F,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAAEwG,KAAK,CAAC,CAAC;IAC7E;IACA;IACA,MAAMsC,cAAc,GAAGtI,MAAM;IAC7BA,MAAM,GAAGjC,kBAAkB,CAACiC,MAAM,CAAC;IAEnCoD,uBAAuB,CAACpD,MAAM,EAAEF,SAAS,EAAE,IAAI,CAACuD,OAAO,CAAC;IACxDE,0BAA0B,CAACvD,MAAM,EAAEF,SAAS,EAAE,IAAI,CAACuD,OAAO,CAAC;IAC3DrD,MAAM,CAACuI,SAAS,GAAG;MAAEC,GAAG,EAAExI,MAAM,CAACuI,SAAS;MAAEE,MAAM,EAAE;IAAO,CAAC;IAC5DzI,MAAM,CAAC0I,SAAS,GAAG;MAAEF,GAAG,EAAExI,MAAM,CAAC0I,SAAS;MAAED,MAAM,EAAE;IAAO,CAAC;IAE5D,IAAIhK,QAAQ,GAAGlB,GAAG,KAAK8H,SAAS;IAChC,IAAI3F,QAAQ,GAAGnC,GAAG,IAAI,EAAE;IACxB,MAAM6I,eAAe,GAAG,IAAI,CAACE,sBAAsB,CAACxG,SAAS,EAAE,IAAI,EAAEE,MAAM,CAAC;IAE5E,OAAO,IAAI,CAACsE,iBAAiB,CAACxE,SAAS,CAAC,CACrCoE,IAAI,CAAC,MAAM,IAAI,CAACY,kBAAkB,CAACe,qBAAqB,CAAC,CAAC,CAC1D3B,IAAI,CAACC,gBAAgB,IAAI;MACxB,OAAO,CAAC1F,QAAQ,GACZ+F,OAAO,CAACG,OAAO,CAAC,CAAC,GACjBR,gBAAgB,CAACkC,kBAAkB,CAACvG,SAAS,EAAEJ,QAAQ,EAAE,QAAQ,CAAC,EAEnEwE,IAAI,CAAC,MAAMC,gBAAgB,CAACwE,kBAAkB,CAAC7I,SAAS,CAAC,CAAC,CAC1DoE,IAAI,CAAC,MAAMC,gBAAgB,CAACC,YAAY,CAACtE,SAAS,EAAE,IAAI,CAAC,CAAC,CAC1DoE,IAAI,CAACrE,MAAM,IAAI;QACd0C,iBAAiB,CAACzC,SAAS,EAAEE,MAAM,EAAEH,MAAM,CAAC;QAC5CoC,+BAA+B,CAACjC,MAAM,CAAC;QACvC,IAAI4F,YAAY,EAAE;UAChB,OAAO,CAAC,CAAC;QACX;QACA,OAAO,IAAI,CAAClC,OAAO,CAACkF,YAAY,CAC9B9I,SAAS,EACTlG,gBAAgB,CAACiP,4BAA4B,CAAChJ,MAAM,CAAC,EACrDG,MAAM,EACN,IAAI,CAAC6D,qBACP,CAAC;MACH,CAAC,CAAC,CACDK,IAAI,CAAChG,MAAM,IAAI;QACd,IAAI0H,YAAY,EAAE;UAChB,OAAO0C,cAAc;QACvB;QACA,OAAO,IAAI,CAACnB,qBAAqB,CAC/BrH,SAAS,EACTE,MAAM,CAACkB,QAAQ,EACflB,MAAM,EACNoG,eACF,CAAC,CAAClC,IAAI,CAAC,MAAM;UACX,OAAO,IAAI,CAACkD,uBAAuB,CAACkB,cAAc,EAAEpK,MAAM,CAACmJ,GAAG,CAAC,CAAC,CAAC,CAAC;QACpE,CAAC,CAAC;MACJ,CAAC,CAAC;IACN,CAAC,CAAC;EACN;EAEA9B,WAAWA,CACT1F,MAAyC,EACzCC,SAAiB,EACjBE,MAAW,EACXN,QAAkB,EAClByF,UAAwB,EACT;IACf,MAAM2D,WAAW,GAAGjJ,MAAM,CAACkJ,UAAU,CAACjJ,SAAS,CAAC;IAChD,IAAI,CAACgJ,WAAW,EAAE;MAChB,OAAOtE,OAAO,CAACG,OAAO,CAAC,CAAC;IAC1B;IACA,MAAMxD,MAAM,GAAGnG,MAAM,CAACS,IAAI,CAACuE,MAAM,CAAC;IAClC,MAAMgJ,YAAY,GAAGhO,MAAM,CAACS,IAAI,CAACqN,WAAW,CAAC3H,MAAM,CAAC;IACpD,MAAM8H,OAAO,GAAG9H,MAAM,CAACvF,MAAM,CAACsN,KAAK,IAAI;MACrC;MACA,IAAIlJ,MAAM,CAACkJ,KAAK,CAAC,IAAIlJ,MAAM,CAACkJ,KAAK,CAAC,CAAChH,IAAI,IAAIlC,MAAM,CAACkJ,KAAK,CAAC,CAAChH,IAAI,KAAK,QAAQ,EAAE;QAC1E,OAAO,KAAK;MACd;MACA,OAAO8G,YAAY,CAAC7L,OAAO,CAAC4F,gBAAgB,CAACmG,KAAK,CAAC,CAAC,GAAG,CAAC;IAC1D,CAAC,CAAC;IACF,IAAID,OAAO,CAAC/M,MAAM,GAAG,CAAC,EAAE;MACtB;MACAiJ,UAAU,CAACO,SAAS,GAAG,IAAI;MAE3B,MAAMyD,MAAM,GAAGhE,UAAU,CAACgE,MAAM;MAChC,OAAOtJ,MAAM,CAACwG,kBAAkB,CAACvG,SAAS,EAAEJ,QAAQ,EAAE,UAAU,EAAEyJ,MAAM,CAAC;IAC3E;IACA,OAAO3E,OAAO,CAACG,OAAO,CAAC,CAAC;EAC1B;;EAEA;EACA;AACF;AACA;AACA;AACA;AACA;EACEyE,gBAAgBA,CAACC,IAAa,GAAG,KAAK,EAAgB;IACpD,IAAI,CAACzF,aAAa,GAAG,IAAI;IACzB0F,oBAAW,CAACC,KAAK,CAAC,CAAC;IACnB,OAAO,IAAI,CAAC7F,OAAO,CAAC8F,gBAAgB,CAACH,IAAI,CAAC;EAC5C;;EAEA;EACA;EACAI,UAAUA,CACR3J,SAAiB,EACjBX,GAAW,EACXgE,QAAgB,EAChBuG,YAA0B,EACF;IACxB,MAAM;MAAEC,IAAI;MAAEC,KAAK;MAAEC;IAAK,CAAC,GAAGH,YAAY;IAC1C,MAAMI,WAAW,GAAG,CAAC,CAAC;IACtB,IAAID,IAAI,IAAIA,IAAI,CAACtB,SAAS,IAAI,IAAI,CAAC7E,OAAO,CAACqG,mBAAmB,EAAE;MAC9DD,WAAW,CAACD,IAAI,GAAG;QAAEG,GAAG,EAAEH,IAAI,CAACtB;MAAU,CAAC;MAC1CuB,WAAW,CAACF,KAAK,GAAGA,KAAK;MACzBE,WAAW,CAACH,IAAI,GAAGA,IAAI;MACvBD,YAAY,CAACC,IAAI,GAAG,CAAC;IACvB;IACA,OAAO,IAAI,CAACjG,OAAO,CAChBoD,IAAI,CAAC9E,aAAa,CAAClC,SAAS,EAAEX,GAAG,CAAC,EAAE8D,cAAc,EAAE;MAAEE;IAAS,CAAC,EAAE2G,WAAW,CAAC,CAC9E5F,IAAI,CAAC+F,OAAO,IAAIA,OAAO,CAACxJ,GAAG,CAACvC,MAAM,IAAIA,MAAM,CAACgF,SAAS,CAAC,CAAC;EAC7D;;EAEA;EACA;EACAgH,SAASA,CAACpK,SAAiB,EAAEX,GAAW,EAAEsK,UAAoB,EAAqB;IACjF,OAAO,IAAI,CAAC/F,OAAO,CAChBoD,IAAI,CACH9E,aAAa,CAAClC,SAAS,EAAEX,GAAG,CAAC,EAC7B8D,cAAc,EACd;MAAEC,SAAS,EAAE;QAAEtF,GAAG,EAAE6L;MAAW;IAAE,CAAC,EAClC;MAAEhO,IAAI,EAAE,CAAC,UAAU;IAAE,CACvB,CAAC,CACAyI,IAAI,CAAC+F,OAAO,IAAIA,OAAO,CAACxJ,GAAG,CAACvC,MAAM,IAAIA,MAAM,CAACiF,QAAQ,CAAC,CAAC;EAC5D;;EAEA;EACA;EACA;EACAgH,gBAAgBA,CAACrK,SAAiB,EAAExC,KAAU,EAAEuC,MAAW,EAAgB;IACzE;IACA;IACA,MAAMuK,QAAQ,GAAG,EAAE;IACnB,IAAI9M,KAAK,CAAC,KAAK,CAAC,EAAE;MAChB,MAAM+M,GAAG,GAAG/M,KAAK,CAAC,KAAK,CAAC;MACxB8M,QAAQ,CAACtO,IAAI,CACX,GAAGuO,GAAG,CAAC5J,GAAG,CAAC,CAAC6J,MAAM,EAAEC,KAAK,KAAK;QAC5B,OAAO,IAAI,CAACJ,gBAAgB,CAACrK,SAAS,EAAEwK,MAAM,EAAEzK,MAAM,CAAC,CAACqE,IAAI,CAACoG,MAAM,IAAI;UACrEhN,KAAK,CAAC,KAAK,CAAC,CAACiN,KAAK,CAAC,GAAGD,MAAM;QAC9B,CAAC,CAAC;MACJ,CAAC,CACH,CAAC;IACH;IACA,IAAIhN,KAAK,CAAC,MAAM,CAAC,EAAE;MACjB,MAAMkN,IAAI,GAAGlN,KAAK,CAAC,MAAM,CAAC;MAC1B8M,QAAQ,CAACtO,IAAI,CACX,GAAG0O,IAAI,CAAC/J,GAAG,CAAC,CAAC6J,MAAM,EAAEC,KAAK,KAAK;QAC7B,OAAO,IAAI,CAACJ,gBAAgB,CAACrK,SAAS,EAAEwK,MAAM,EAAEzK,MAAM,CAAC,CAACqE,IAAI,CAACoG,MAAM,IAAI;UACrEhN,KAAK,CAAC,MAAM,CAAC,CAACiN,KAAK,CAAC,GAAGD,MAAM;QAC/B,CAAC,CAAC;MACJ,CAAC,CACH,CAAC;IACH;IAEA,MAAMG,SAAS,GAAGzP,MAAM,CAACS,IAAI,CAAC6B,KAAK,CAAC,CAACmD,GAAG,CAACtB,GAAG,IAAI;MAC9C,IAAIA,GAAG,KAAK,MAAM,IAAIA,GAAG,KAAK,KAAK,EAAE;QACnC;MACF;MACA,MAAM3E,CAAC,GAAGqF,MAAM,CAACmF,eAAe,CAAClF,SAAS,EAAEX,GAAG,CAAC;MAChD,IAAI,CAAC3E,CAAC,IAAIA,CAAC,CAACmI,IAAI,KAAK,UAAU,EAAE;QAC/B,OAAO6B,OAAO,CAACG,OAAO,CAACrH,KAAK,CAAC;MAC/B;MACA,IAAIoN,OAAiB,GAAG,IAAI;MAC5B,IACEpN,KAAK,CAAC6B,GAAG,CAAC,KACT7B,KAAK,CAAC6B,GAAG,CAAC,CAAC,KAAK,CAAC,IAChB7B,KAAK,CAAC6B,GAAG,CAAC,CAAC,KAAK,CAAC,IACjB7B,KAAK,CAAC6B,GAAG,CAAC,CAAC,MAAM,CAAC,IAClB7B,KAAK,CAAC6B,GAAG,CAAC,CAACsJ,MAAM,IAAI,SAAS,CAAC,EACjC;QACA;QACAiC,OAAO,GAAG1P,MAAM,CAACS,IAAI,CAAC6B,KAAK,CAAC6B,GAAG,CAAC,CAAC,CAACsB,GAAG,CAACkK,aAAa,IAAI;UACrD,IAAIlB,UAAU;UACd,IAAImB,UAAU,GAAG,KAAK;UACtB,IAAID,aAAa,KAAK,UAAU,EAAE;YAChClB,UAAU,GAAG,CAACnM,KAAK,CAAC6B,GAAG,CAAC,CAAC+B,QAAQ,CAAC;UACpC,CAAC,MAAM,IAAIyJ,aAAa,IAAI,KAAK,EAAE;YACjClB,UAAU,GAAGnM,KAAK,CAAC6B,GAAG,CAAC,CAAC,KAAK,CAAC,CAACsB,GAAG,CAAClG,CAAC,IAAIA,CAAC,CAAC2G,QAAQ,CAAC;UACrD,CAAC,MAAM,IAAIyJ,aAAa,IAAI,MAAM,EAAE;YAClCC,UAAU,GAAG,IAAI;YACjBnB,UAAU,GAAGnM,KAAK,CAAC6B,GAAG,CAAC,CAAC,MAAM,CAAC,CAACsB,GAAG,CAAClG,CAAC,IAAIA,CAAC,CAAC2G,QAAQ,CAAC;UACtD,CAAC,MAAM,IAAIyJ,aAAa,IAAI,KAAK,EAAE;YACjCC,UAAU,GAAG,IAAI;YACjBnB,UAAU,GAAG,CAACnM,KAAK,CAAC6B,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC+B,QAAQ,CAAC;UAC3C,CAAC,MAAM;YACL;UACF;UACA,OAAO;YACL0J,UAAU;YACVnB;UACF,CAAC;QACH,CAAC,CAAC;MACJ,CAAC,MAAM;QACLiB,OAAO,GAAG,CAAC;UAAEE,UAAU,EAAE,KAAK;UAAEnB,UAAU,EAAE;QAAG,CAAC,CAAC;MACnD;;MAEA;MACA,OAAOnM,KAAK,CAAC6B,GAAG,CAAC;MACjB;MACA;MACA,MAAMiL,QAAQ,GAAGM,OAAO,CAACjK,GAAG,CAACoK,CAAC,IAAI;QAChC,IAAI,CAACA,CAAC,EAAE;UACN,OAAOrG,OAAO,CAACG,OAAO,CAAC,CAAC;QAC1B;QACA,OAAO,IAAI,CAACuF,SAAS,CAACpK,SAAS,EAAEX,GAAG,EAAE0L,CAAC,CAACpB,UAAU,CAAC,CAACvF,IAAI,CAAC4G,GAAG,IAAI;UAC9D,IAAID,CAAC,CAACD,UAAU,EAAE;YAChB,IAAI,CAACG,oBAAoB,CAACD,GAAG,EAAExN,KAAK,CAAC;UACvC,CAAC,MAAM;YACL,IAAI,CAAC0N,iBAAiB,CAACF,GAAG,EAAExN,KAAK,CAAC;UACpC;UACA,OAAOkH,OAAO,CAACG,OAAO,CAAC,CAAC;QAC1B,CAAC,CAAC;MACJ,CAAC,CAAC;MAEF,OAAOH,OAAO,CAACqD,GAAG,CAACuC,QAAQ,CAAC,CAAClG,IAAI,CAAC,MAAM;QACtC,OAAOM,OAAO,CAACG,OAAO,CAAC,CAAC;MAC1B,CAAC,CAAC;IACJ,CAAC,CAAC;IAEF,OAAOH,OAAO,CAACqD,GAAG,CAAC,CAAC,GAAGuC,QAAQ,EAAE,GAAGK,SAAS,CAAC,CAAC,CAACvG,IAAI,CAAC,MAAM;MACzD,OAAOM,OAAO,CAACG,OAAO,CAACrH,KAAK,CAAC;IAC/B,CAAC,CAAC;EACJ;;EAEA;EACA;EACA2N,kBAAkBA,CAACnL,SAAiB,EAAExC,KAAU,EAAEoM,YAAiB,EAAkB;IACnF,IAAIpM,KAAK,CAAC,KAAK,CAAC,EAAE;MAChB,OAAOkH,OAAO,CAACqD,GAAG,CAChBvK,KAAK,CAAC,KAAK,CAAC,CAACmD,GAAG,CAAC6J,MAAM,IAAI;QACzB,OAAO,IAAI,CAACW,kBAAkB,CAACnL,SAAS,EAAEwK,MAAM,EAAEZ,YAAY,CAAC;MACjE,CAAC,CACH,CAAC;IACH;IACA,IAAIpM,KAAK,CAAC,MAAM,CAAC,EAAE;MACjB,OAAOkH,OAAO,CAACqD,GAAG,CAChBvK,KAAK,CAAC,MAAM,CAAC,CAACmD,GAAG,CAAC6J,MAAM,IAAI;QAC1B,OAAO,IAAI,CAACW,kBAAkB,CAACnL,SAAS,EAAEwK,MAAM,EAAEZ,YAAY,CAAC;MACjE,CAAC,CACH,CAAC;IACH;IACA,IAAIwB,SAAS,GAAG5N,KAAK,CAAC,YAAY,CAAC;IACnC,IAAI4N,SAAS,EAAE;MACb,OAAO,IAAI,CAACzB,UAAU,CACpByB,SAAS,CAAClL,MAAM,CAACF,SAAS,EAC1BoL,SAAS,CAAC/L,GAAG,EACb+L,SAAS,CAAClL,MAAM,CAACkB,QAAQ,EACzBwI,YACF,CAAC,CACExF,IAAI,CAAC4G,GAAG,IAAI;QACX,OAAOxN,KAAK,CAAC,YAAY,CAAC;QAC1B,IAAI,CAAC0N,iBAAiB,CAACF,GAAG,EAAExN,KAAK,CAAC;QAClC,OAAO,IAAI,CAAC2N,kBAAkB,CAACnL,SAAS,EAAExC,KAAK,EAAEoM,YAAY,CAAC;MAChE,CAAC,CAAC,CACDxF,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;IACnB;EACF;EAEA8G,iBAAiBA,CAACF,GAAmB,GAAG,IAAI,EAAExN,KAAU,EAAE;IACxD,MAAM6N,aAA6B,GACjC,OAAO7N,KAAK,CAAC4D,QAAQ,KAAK,QAAQ,GAAG,CAAC5D,KAAK,CAAC4D,QAAQ,CAAC,GAAG,IAAI;IAC9D,MAAMkK,SAAyB,GAC7B9N,KAAK,CAAC4D,QAAQ,IAAI5D,KAAK,CAAC4D,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC5D,KAAK,CAAC4D,QAAQ,CAAC,KAAK,CAAC,CAAC,GAAG,IAAI;IAC1E,MAAMmK,SAAyB,GAC7B/N,KAAK,CAAC4D,QAAQ,IAAI5D,KAAK,CAAC4D,QAAQ,CAAC,KAAK,CAAC,GAAG5D,KAAK,CAAC4D,QAAQ,CAAC,KAAK,CAAC,GAAG,IAAI;;IAExE;IACA,MAAMoK,MAA4B,GAAG,CAACH,aAAa,EAAEC,SAAS,EAAEC,SAAS,EAAEP,GAAG,CAAC,CAAClP,MAAM,CACpF2P,IAAI,IAAIA,IAAI,KAAK,IACnB,CAAC;IACD,MAAMC,WAAW,GAAGF,MAAM,CAACG,MAAM,CAAC,CAACC,IAAI,EAAEH,IAAI,KAAKG,IAAI,GAAGH,IAAI,CAACrP,MAAM,EAAE,CAAC,CAAC;IAExE,IAAIyP,eAAe,GAAG,EAAE;IACxB,IAAIH,WAAW,GAAG,GAAG,EAAE;MACrBG,eAAe,GAAGC,kBAAS,CAACC,GAAG,CAACP,MAAM,CAAC;IACzC,CAAC,MAAM;MACLK,eAAe,GAAG,IAAAC,kBAAS,EAACN,MAAM,CAAC;IACrC;;IAEA;IACA,IAAI,EAAE,UAAU,IAAIhO,KAAK,CAAC,EAAE;MAC1BA,KAAK,CAAC4D,QAAQ,GAAG;QACftD,GAAG,EAAEyH;MACP,CAAC;IACH,CAAC,MAAM,IAAI,OAAO/H,KAAK,CAAC4D,QAAQ,KAAK,QAAQ,EAAE;MAC7C5D,KAAK,CAAC4D,QAAQ,GAAG;QACftD,GAAG,EAAEyH,SAAS;QACdyG,GAAG,EAAExO,KAAK,CAAC4D;MACb,CAAC;IACH;IACA5D,KAAK,CAAC4D,QAAQ,CAAC,KAAK,CAAC,GAAGyK,eAAe;IAEvC,OAAOrO,KAAK;EACd;EAEAyN,oBAAoBA,CAACD,GAAa,GAAG,EAAE,EAAExN,KAAU,EAAE;IACnD,MAAMyO,UAAU,GAAGzO,KAAK,CAAC4D,QAAQ,IAAI5D,KAAK,CAAC4D,QAAQ,CAAC,MAAM,CAAC,GAAG5D,KAAK,CAAC4D,QAAQ,CAAC,MAAM,CAAC,GAAG,EAAE;IACzF,IAAIoK,MAAM,GAAG,CAAC,GAAGS,UAAU,EAAE,GAAGjB,GAAG,CAAC,CAAClP,MAAM,CAAC2P,IAAI,IAAIA,IAAI,KAAK,IAAI,CAAC;;IAElE;IACAD,MAAM,GAAG,CAAC,GAAG,IAAIU,GAAG,CAACV,MAAM,CAAC,CAAC;;IAE7B;IACA,IAAI,EAAE,UAAU,IAAIhO,KAAK,CAAC,EAAE;MAC1BA,KAAK,CAAC4D,QAAQ,GAAG;QACf+K,IAAI,EAAE5G;MACR,CAAC;IACH,CAAC,MAAM,IAAI,OAAO/H,KAAK,CAAC4D,QAAQ,KAAK,QAAQ,EAAE;MAC7C5D,KAAK,CAAC4D,QAAQ,GAAG;QACf+K,IAAI,EAAE5G,SAAS;QACfyG,GAAG,EAAExO,KAAK,CAAC4D;MACb,CAAC;IACH;IAEA5D,KAAK,CAAC4D,QAAQ,CAAC,MAAM,CAAC,GAAGoK,MAAM;IAC/B,OAAOhO,KAAK;EACd;;EAEA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACAwJ,IAAIA,CACFhH,SAAiB,EACjBxC,KAAU,EACV;IACEqM,IAAI;IACJC,KAAK;IACLrM,GAAG;IACHsM,IAAI,GAAG,CAAC,CAAC;IACTqC,KAAK;IACLzQ,IAAI;IACJ+L,EAAE;IACF2E,QAAQ;IACRC,QAAQ;IACRC,cAAc;IACdC,IAAI;IACJC,eAAe,GAAG,KAAK;IACvBC,OAAO;IACPC;EACG,CAAC,GAAG,CAAC,CAAC,EACX9M,IAAS,GAAG,CAAC,CAAC,EACdkG,qBAAwD,EAC1C;IACd,MAAMnH,aAAa,GAAGiB,IAAI,CAACjB,aAAa;IACxC,MAAMD,QAAQ,GAAGlB,GAAG,KAAK8H,SAAS,IAAI3G,aAAa;IACnD,MAAMgB,QAAQ,GAAGnC,GAAG,IAAI,EAAE;IAC1BiK,EAAE,GACAA,EAAE,KAAK,OAAOlK,KAAK,CAAC4D,QAAQ,IAAI,QAAQ,IAAIlG,MAAM,CAACS,IAAI,CAAC6B,KAAK,CAAC,CAACpB,MAAM,KAAK,CAAC,GAAG,KAAK,GAAG,MAAM,CAAC;IAC/F;IACAsL,EAAE,GAAG0E,KAAK,KAAK,IAAI,GAAG,OAAO,GAAG1E,EAAE;IAElC,IAAIzD,WAAW,GAAG,IAAI;IACtB,OAAO,IAAI,CAACe,kBAAkB,CAACe,qBAAqB,CAAC,CAAC3B,IAAI,CAACC,gBAAgB,IAAI;MAC7E;MACA;MACA;MACA,OAAOA,gBAAgB,CACpBC,YAAY,CAACtE,SAAS,EAAErB,QAAQ,CAAC,CACjC+H,KAAK,CAACR,KAAK,IAAI;QACd;QACA;QACA,IAAIA,KAAK,KAAKX,SAAS,EAAE;UACvBtB,WAAW,GAAG,KAAK;UACnB,OAAO;YAAE5C,MAAM,EAAE,CAAC;UAAE,CAAC;QACvB;QACA,MAAM6E,KAAK;MACb,CAAC,CAAC,CACD9B,IAAI,CAACrE,MAAM,IAAI;QACd;QACA;QACA;QACA,IAAIgK,IAAI,CAAC6C,WAAW,EAAE;UACpB7C,IAAI,CAACtB,SAAS,GAAGsB,IAAI,CAAC6C,WAAW;UACjC,OAAO7C,IAAI,CAAC6C,WAAW;QACzB;QACA,IAAI7C,IAAI,CAAC8C,WAAW,EAAE;UACpB9C,IAAI,CAACnB,SAAS,GAAGmB,IAAI,CAAC8C,WAAW;UACjC,OAAO9C,IAAI,CAAC8C,WAAW;QACzB;QACA,MAAMjD,YAAY,GAAG;UACnBC,IAAI;UACJC,KAAK;UACLC,IAAI;UACJpO,IAAI;UACJ4Q,cAAc;UACdC,IAAI;UACJC,eAAe,EAAE,IAAI,CAAClJ,OAAO,CAACuJ,6BAA6B,GAAG,KAAK,GAAGL,eAAe;UACrFC,OAAO;UACPC;QACF,CAAC;QACDzR,MAAM,CAACS,IAAI,CAACoO,IAAI,CAAC,CAAC1N,OAAO,CAACuG,SAAS,IAAI;UACrC,IAAIA,SAAS,CAACpD,KAAK,CAAC,iCAAiC,CAAC,EAAE;YACtD,MAAM,IAAIV,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAAE,kBAAkBkD,SAAS,EAAE,CAAC;UACpF;UACA,MAAM+D,aAAa,GAAG1D,gBAAgB,CAACL,SAAS,CAAC;UACjD,IAAI,CAAC9I,gBAAgB,CAAC8M,gBAAgB,CAACD,aAAa,EAAE3G,SAAS,CAAC,EAAE;YAChE,MAAM,IAAIlB,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAC5B,uBAAuBkD,SAAS,GAClC,CAAC;UACH;UACA,IAAI,CAAC7C,MAAM,CAACsB,MAAM,CAACuB,SAAS,CAACM,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAIN,SAAS,KAAK,OAAO,EAAE;YACpE,OAAOmH,IAAI,CAACnH,SAAS,CAAC;UACxB;QACF,CAAC,CAAC;QACF,OAAO,CAACjE,QAAQ,GACZ+F,OAAO,CAACG,OAAO,CAAC,CAAC,GACjBR,gBAAgB,CAACkC,kBAAkB,CAACvG,SAAS,EAAEJ,QAAQ,EAAE8H,EAAE,CAAC,EAE7DtD,IAAI,CAAC,MAAM,IAAI,CAAC+G,kBAAkB,CAACnL,SAAS,EAAExC,KAAK,EAAEoM,YAAY,CAAC,CAAC,CACnExF,IAAI,CAAC,MAAM,IAAI,CAACiG,gBAAgB,CAACrK,SAAS,EAAExC,KAAK,EAAE6G,gBAAgB,CAAC,CAAC,CACrED,IAAI,CAAC,MAAM;UACV,IAAInE,eAAe;UACnB,IAAI,CAACtB,QAAQ,EAAE;YACbnB,KAAK,GAAG,IAAI,CAACiJ,qBAAqB,CAChCpC,gBAAgB,EAChBrE,SAAS,EACT0H,EAAE,EACFlK,KAAK,EACLoC,QACF,CAAC;YACD;AAChB;AACA;YACgBK,eAAe,GAAG,IAAI,CAAC8M,kBAAkB,CACvC1I,gBAAgB,EAChBrE,SAAS,EACTxC,KAAK,EACLoC,QAAQ,EACRC,IAAI,EACJ+J,YACF,CAAC;UACH;UACA,IAAI,CAACpM,KAAK,EAAE;YACV,IAAIkK,EAAE,KAAK,KAAK,EAAE;cAChB,MAAM,IAAI5I,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACkI,gBAAgB,EAAE,mBAAmB,CAAC;YAC1E,CAAC,MAAM;cACL,OAAO,EAAE;YACX;UACF;UACA,IAAI,CAACtI,QAAQ,EAAE;YACb,IAAI+I,EAAE,KAAK,QAAQ,IAAIA,EAAE,KAAK,QAAQ,EAAE;cACtClK,KAAK,GAAGD,WAAW,CAACC,KAAK,EAAEoC,QAAQ,CAAC;YACtC,CAAC,MAAM;cACLpC,KAAK,GAAGO,UAAU,CAACP,KAAK,EAAEoC,QAAQ,CAAC;YACrC;UACF;UACAlB,aAAa,CAAClB,KAAK,EAAEmB,QAAQ,EAAEC,aAAa,EAAE,KAAK,CAAC;UACpD,IAAIwN,KAAK,EAAE;YACT,IAAI,CAACnI,WAAW,EAAE;cAChB,OAAO,CAAC;YACV,CAAC,MAAM;cACL,OAAO,IAAI,CAACL,OAAO,CAACwI,KAAK,CACvBpM,SAAS,EACTD,MAAM,EACNvC,KAAK,EACL+O,cAAc,EACdhH,SAAS,EACTiH,IAAI,EACJG,OACF,CAAC;YACH;UACF,CAAC,MAAM,IAAIN,QAAQ,EAAE;YACnB,IAAI,CAACpI,WAAW,EAAE;cAChB,OAAO,EAAE;YACX,CAAC,MAAM;cACL,OAAO,IAAI,CAACL,OAAO,CAACyI,QAAQ,CAACrM,SAAS,EAAED,MAAM,EAAEvC,KAAK,EAAE6O,QAAQ,CAAC;YAClE;UACF,CAAC,MAAM,IAAIC,QAAQ,EAAE;YACnB,IAAI,CAACrI,WAAW,EAAE;cAChB,OAAO,EAAE;YACX,CAAC,MAAM;cACL,OAAO,IAAI,CAACL,OAAO,CAACoJ,SAAS,CAC3BhN,SAAS,EACTD,MAAM,EACNuM,QAAQ,EACRC,cAAc,EACdC,IAAI,EACJE,OAAO,EACPC,OACF,CAAC;YACH;UACF,CAAC,MAAM,IAAID,OAAO,EAAE;YAClB,OAAO,IAAI,CAAC9I,OAAO,CAACoD,IAAI,CAAChH,SAAS,EAAED,MAAM,EAAEvC,KAAK,EAAEoM,YAAY,CAAC;UAClE,CAAC,MAAM;YACL,OAAO,IAAI,CAAChG,OAAO,CAChBoD,IAAI,CAAChH,SAAS,EAAED,MAAM,EAAEvC,KAAK,EAAEoM,YAAY,CAAC,CAC5CxF,IAAI,CAAC7B,OAAO,IACXA,OAAO,CAAC5B,GAAG,CAACT,MAAM,IAAI;cACpBA,MAAM,GAAG4C,oBAAoB,CAAC5C,MAAM,CAAC;cACrC,OAAOP,mBAAmB,CACxBhB,QAAQ,EACRC,aAAa,EACbgB,QAAQ,EACRC,IAAI,EACJ6H,EAAE,EACFrD,gBAAgB,EAChBrE,SAAS,EACTC,eAAe,EACfC,MACF,CAAC;YACH,CAAC,CACH,CAAC,CACAwG,KAAK,CAACR,KAAK,IAAI;cACd,MAAM,IAAIpH,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACkO,qBAAqB,EAAE/G,KAAK,CAAC;YACjE,CAAC,CAAC;UACN;QACF,CAAC,CAAC;MACN,CAAC,CAAC;IACN,CAAC,CAAC;EACJ;EAEAgH,YAAYA,CAAClN,SAAiB,EAAiB;IAC7C,IAAIqE,gBAAgB;IACpB,OAAO,IAAI,CAACF,UAAU,CAAC;MAAEW,UAAU,EAAE;IAAK,CAAC,CAAC,CACzCV,IAAI,CAACoB,CAAC,IAAI;MACTnB,gBAAgB,GAAGmB,CAAC;MACpB,OAAOnB,gBAAgB,CAACC,YAAY,CAACtE,SAAS,EAAE,IAAI,CAAC;IACvD,CAAC,CAAC,CACD0G,KAAK,CAACR,KAAK,IAAI;MACd,IAAIA,KAAK,KAAKX,SAAS,EAAE;QACvB,OAAO;UAAElE,MAAM,EAAE,CAAC;QAAE,CAAC;MACvB,CAAC,MAAM;QACL,MAAM6E,KAAK;MACb;IACF,CAAC,CAAC,CACD9B,IAAI,CAAErE,MAAW,IAAK;MACrB,OAAO,IAAI,CAACiE,gBAAgB,CAAChE,SAAS,CAAC,CACpCoE,IAAI,CAAC,MAAM,IAAI,CAACR,OAAO,CAACwI,KAAK,CAACpM,SAAS,EAAE;QAAEqB,MAAM,EAAE,CAAC;MAAE,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC,CAC1E+C,IAAI,CAACgI,KAAK,IAAI;QACb,IAAIA,KAAK,GAAG,CAAC,EAAE;UACb,MAAM,IAAItN,WAAK,CAACC,KAAK,CACnB,GAAG,EACH,SAASiB,SAAS,2BAA2BoM,KAAK,+BACpD,CAAC;QACH;QACA,OAAO,IAAI,CAACxI,OAAO,CAACuJ,WAAW,CAACnN,SAAS,CAAC;MAC5C,CAAC,CAAC,CACDoE,IAAI,CAACgJ,kBAAkB,IAAI;QAC1B,IAAIA,kBAAkB,EAAE;UACtB,MAAMC,kBAAkB,GAAGnS,MAAM,CAACS,IAAI,CAACoE,MAAM,CAACsB,MAAM,CAAC,CAACvF,MAAM,CAC1D8G,SAAS,IAAI7C,MAAM,CAACsB,MAAM,CAACuB,SAAS,CAAC,CAACC,IAAI,KAAK,UACjD,CAAC;UACD,OAAO6B,OAAO,CAACqD,GAAG,CAChBsF,kBAAkB,CAAC1M,GAAG,CAAC2M,IAAI,IACzB,IAAI,CAAC1J,OAAO,CAACuJ,WAAW,CAACjL,aAAa,CAAClC,SAAS,EAAEsN,IAAI,CAAC,CACzD,CACF,CAAC,CAAClJ,IAAI,CAAC,MAAM;YACXoF,oBAAW,CAAC+D,GAAG,CAACvN,SAAS,CAAC;YAC1B,OAAOqE,gBAAgB,CAACmJ,UAAU,CAAC,CAAC;UACtC,CAAC,CAAC;QACJ,CAAC,MAAM;UACL,OAAO9I,OAAO,CAACG,OAAO,CAAC,CAAC;QAC1B;MACF,CAAC,CAAC;IACN,CAAC,CAAC;EACN;;EAEA;EACA;EACA;EACA4I,sBAAsBA,CAACjQ,KAAU,EAAiB;IAChD,OAAOtC,MAAM,CAACwS,OAAO,CAAClQ,KAAK,CAAC,CAACmD,GAAG,CAAC1F,CAAC,IAAIA,CAAC,CAAC0F,GAAG,CAAC6E,CAAC,IAAImI,IAAI,CAACC,SAAS,CAACpI,CAAC,CAAC,CAAC,CAACqI,IAAI,CAAC,GAAG,CAAC,CAAC;EAChF;;EAEA;EACAC,iBAAiBA,CAACtQ,KAA0B,EAAO;IACjD,IAAI,CAACA,KAAK,CAACyB,GAAG,EAAE;MACd,OAAOzB,KAAK;IACd;IACA,MAAMoN,OAAO,GAAGpN,KAAK,CAACyB,GAAG,CAAC0B,GAAG,CAACoK,CAAC,IAAI,IAAI,CAAC0C,sBAAsB,CAAC1C,CAAC,CAAC,CAAC;IAClE,IAAIgD,MAAM,GAAG,KAAK;IAClB,GAAG;MACDA,MAAM,GAAG,KAAK;MACd,KAAK,IAAIvS,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGoP,OAAO,CAACxO,MAAM,GAAG,CAAC,EAAEZ,CAAC,EAAE,EAAE;QAC3C,KAAK,IAAIwS,CAAC,GAAGxS,CAAC,GAAG,CAAC,EAAEwS,CAAC,GAAGpD,OAAO,CAACxO,MAAM,EAAE4R,CAAC,EAAE,EAAE;UAC3C,MAAM,CAACC,OAAO,EAAEC,MAAM,CAAC,GAAGtD,OAAO,CAACpP,CAAC,CAAC,CAACY,MAAM,GAAGwO,OAAO,CAACoD,CAAC,CAAC,CAAC5R,MAAM,GAAG,CAAC4R,CAAC,EAAExS,CAAC,CAAC,GAAG,CAACA,CAAC,EAAEwS,CAAC,CAAC;UACjF,MAAMG,YAAY,GAAGvD,OAAO,CAACqD,OAAO,CAAC,CAACtC,MAAM,CAC1C,CAACyC,GAAG,EAAE/P,KAAK,KAAK+P,GAAG,IAAIxD,OAAO,CAACsD,MAAM,CAAC,CAACzO,QAAQ,CAACpB,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAC/D,CACF,CAAC;UACD,MAAMgQ,cAAc,GAAGzD,OAAO,CAACqD,OAAO,CAAC,CAAC7R,MAAM;UAC9C,IAAI+R,YAAY,KAAKE,cAAc,EAAE;YACnC;YACA;YACA7Q,KAAK,CAACyB,GAAG,CAACqP,MAAM,CAACJ,MAAM,EAAE,CAAC,CAAC;YAC3BtD,OAAO,CAAC0D,MAAM,CAACJ,MAAM,EAAE,CAAC,CAAC;YACzBH,MAAM,GAAG,IAAI;YACb;UACF;QACF;MACF;IACF,CAAC,QAAQA,MAAM;IACf,IAAIvQ,KAAK,CAACyB,GAAG,CAAC7C,MAAM,KAAK,CAAC,EAAE;MAC1BoB,KAAK,GAAAtB,aAAA,CAAAA,aAAA,KAAQsB,KAAK,GAAKA,KAAK,CAACyB,GAAG,CAAC,CAAC,CAAC,CAAE;MACrC,OAAOzB,KAAK,CAACyB,GAAG;IAClB;IACA,OAAOzB,KAAK;EACd;;EAEA;EACA+Q,kBAAkBA,CAAC/Q,KAA2B,EAAO;IACnD,IAAI,CAACA,KAAK,CAAC2B,IAAI,EAAE;MACf,OAAO3B,KAAK;IACd;IACA,MAAMoN,OAAO,GAAGpN,KAAK,CAAC2B,IAAI,CAACwB,GAAG,CAACoK,CAAC,IAAI,IAAI,CAAC0C,sBAAsB,CAAC1C,CAAC,CAAC,CAAC;IACnE,IAAIgD,MAAM,GAAG,KAAK;IAClB,GAAG;MACDA,MAAM,GAAG,KAAK;MACd,KAAK,IAAIvS,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGoP,OAAO,CAACxO,MAAM,GAAG,CAAC,EAAEZ,CAAC,EAAE,EAAE;QAC3C,KAAK,IAAIwS,CAAC,GAAGxS,CAAC,GAAG,CAAC,EAAEwS,CAAC,GAAGpD,OAAO,CAACxO,MAAM,EAAE4R,CAAC,EAAE,EAAE;UAC3C,MAAM,CAACC,OAAO,EAAEC,MAAM,CAAC,GAAGtD,OAAO,CAACpP,CAAC,CAAC,CAACY,MAAM,GAAGwO,OAAO,CAACoD,CAAC,CAAC,CAAC5R,MAAM,GAAG,CAAC4R,CAAC,EAAExS,CAAC,CAAC,GAAG,CAACA,CAAC,EAAEwS,CAAC,CAAC;UACjF,MAAMG,YAAY,GAAGvD,OAAO,CAACqD,OAAO,CAAC,CAACtC,MAAM,CAC1C,CAACyC,GAAG,EAAE/P,KAAK,KAAK+P,GAAG,IAAIxD,OAAO,CAACsD,MAAM,CAAC,CAACzO,QAAQ,CAACpB,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAC/D,CACF,CAAC;UACD,MAAMgQ,cAAc,GAAGzD,OAAO,CAACqD,OAAO,CAAC,CAAC7R,MAAM;UAC9C,IAAI+R,YAAY,KAAKE,cAAc,EAAE;YACnC;YACA;YACA7Q,KAAK,CAAC2B,IAAI,CAACmP,MAAM,CAACL,OAAO,EAAE,CAAC,CAAC;YAC7BrD,OAAO,CAAC0D,MAAM,CAACL,OAAO,EAAE,CAAC,CAAC;YAC1BF,MAAM,GAAG,IAAI;YACb;UACF;QACF;MACF;IACF,CAAC,QAAQA,MAAM;IACf,IAAIvQ,KAAK,CAAC2B,IAAI,CAAC/C,MAAM,KAAK,CAAC,EAAE;MAC3BoB,KAAK,GAAAtB,aAAA,CAAAA,aAAA,KAAQsB,KAAK,GAAKA,KAAK,CAAC2B,IAAI,CAAC,CAAC,CAAC,CAAE;MACtC,OAAO3B,KAAK,CAAC2B,IAAI;IACnB;IACA,OAAO3B,KAAK;EACd;;EAEA;EACA;EACA;EACA;EACA;EACAiJ,qBAAqBA,CACnB1G,MAAyC,EACzCC,SAAiB,EACjBF,SAAiB,EACjBtC,KAAU,EACVoC,QAAe,GAAG,EAAE,EACf;IACL;IACA;IACA,IAAIG,MAAM,CAACyO,2BAA2B,CAACxO,SAAS,EAAEJ,QAAQ,EAAEE,SAAS,CAAC,EAAE;MACtE,OAAOtC,KAAK;IACd;IACA,MAAM8C,KAAK,GAAGP,MAAM,CAACQ,wBAAwB,CAACP,SAAS,CAAC;IAExD,MAAMyO,OAAO,GAAG7O,QAAQ,CAAC9D,MAAM,CAAC2B,GAAG,IAAI;MACrC,OAAOA,GAAG,CAACJ,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,IAAII,GAAG,IAAI,GAAG;IAChD,CAAC,CAAC;IAEF,MAAMiR,QAAQ,GACZ,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,CAACrR,OAAO,CAACyC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,gBAAgB,GAAG,iBAAiB;IAEzF,MAAM6O,UAAU,GAAG,EAAE;IAErB,IAAIrO,KAAK,CAACR,SAAS,CAAC,IAAIQ,KAAK,CAACR,SAAS,CAAC,CAAC8O,aAAa,EAAE;MACtDD,UAAU,CAAC3S,IAAI,CAAC,GAAGsE,KAAK,CAACR,SAAS,CAAC,CAAC8O,aAAa,CAAC;IACpD;IAEA,IAAItO,KAAK,CAACoO,QAAQ,CAAC,EAAE;MACnB,KAAK,MAAMtF,KAAK,IAAI9I,KAAK,CAACoO,QAAQ,CAAC,EAAE;QACnC,IAAI,CAACC,UAAU,CAAClP,QAAQ,CAAC2J,KAAK,CAAC,EAAE;UAC/BuF,UAAU,CAAC3S,IAAI,CAACoN,KAAK,CAAC;QACxB;MACF;IACF;IACA;IACA,IAAIuF,UAAU,CAACvS,MAAM,GAAG,CAAC,EAAE;MACzB;MACA;MACA;MACA,IAAIqS,OAAO,CAACrS,MAAM,IAAI,CAAC,EAAE;QACvB;MACF;MACA,MAAM+D,MAAM,GAAGsO,OAAO,CAAC,CAAC,CAAC;MACzB,MAAMI,WAAW,GAAG;QAClBlG,MAAM,EAAE,SAAS;QACjB3I,SAAS,EAAE,OAAO;QAClBoB,QAAQ,EAAEjB;MACZ,CAAC;MAED,MAAMyK,OAAO,GAAG+D,UAAU,CAAChO,GAAG,CAACtB,GAAG,IAAI;QACpC,MAAMyP,eAAe,GAAG/O,MAAM,CAACmF,eAAe,CAAClF,SAAS,EAAEX,GAAG,CAAC;QAC9D,MAAM0P,SAAS,GACbD,eAAe,IACf,OAAOA,eAAe,KAAK,QAAQ,IACnC5T,MAAM,CAAC8T,SAAS,CAAC1T,cAAc,CAACC,IAAI,CAACuT,eAAe,EAAE,MAAM,CAAC,GACzDA,eAAe,CAACjM,IAAI,GACpB,IAAI;QAEV,IAAIoM,WAAW;QAEf,IAAIF,SAAS,KAAK,SAAS,EAAE;UAC3B;UACAE,WAAW,GAAG;YAAE,CAAC5P,GAAG,GAAGwP;UAAY,CAAC;QACtC,CAAC,MAAM,IAAIE,SAAS,KAAK,OAAO,EAAE;UAChC;UACAE,WAAW,GAAG;YAAE,CAAC5P,GAAG,GAAG;cAAE6P,IAAI,EAAE,CAACL,WAAW;YAAE;UAAE,CAAC;QAClD,CAAC,MAAM,IAAIE,SAAS,KAAK,QAAQ,EAAE;UACjC;UACAE,WAAW,GAAG;YAAE,CAAC5P,GAAG,GAAGwP;UAAY,CAAC;QACtC,CAAC,MAAM;UACL;UACA;UACA,MAAM9P,KAAK,CACT,wEAAwEiB,SAAS,IAAIX,GAAG,EAC1F,CAAC;QACH;QACA;QACA,IAAInE,MAAM,CAAC8T,SAAS,CAAC1T,cAAc,CAACC,IAAI,CAACiC,KAAK,EAAE6B,GAAG,CAAC,EAAE;UACpD,OAAO,IAAI,CAACkP,kBAAkB,CAAC;YAAEpP,IAAI,EAAE,CAAC8P,WAAW,EAAEzR,KAAK;UAAE,CAAC,CAAC;QAChE;QACA;QACA,OAAOtC,MAAM,CAACiU,MAAM,CAAC,CAAC,CAAC,EAAE3R,KAAK,EAAEyR,WAAW,CAAC;MAC9C,CAAC,CAAC;MAEF,OAAOrE,OAAO,CAACxO,MAAM,KAAK,CAAC,GAAGwO,OAAO,CAAC,CAAC,CAAC,GAAG,IAAI,CAACkD,iBAAiB,CAAC;QAAE7O,GAAG,EAAE2L;MAAQ,CAAC,CAAC;IACrF,CAAC,MAAM;MACL,OAAOpN,KAAK;IACd;EACF;EAEAuP,kBAAkBA,CAChBhN,MAA+C,EAC/CC,SAAiB,EACjBxC,KAAU,GAAG,CAAC,CAAC,EACfoC,QAAe,GAAG,EAAE,EACpBC,IAAS,GAAG,CAAC,CAAC,EACd+J,YAA8B,GAAG,CAAC,CAAC,EAClB;IACjB,MAAMtJ,KAAK,GACTP,MAAM,IAAIA,MAAM,CAACQ,wBAAwB,GACrCR,MAAM,CAACQ,wBAAwB,CAACP,SAAS,CAAC,GAC1CD,MAAM;IACZ,IAAI,CAACO,KAAK,EAAE,OAAO,IAAI;IAEvB,MAAML,eAAe,GAAGK,KAAK,CAACL,eAAe;IAC7C,IAAI,CAACA,eAAe,EAAE,OAAO,IAAI;IAEjC,IAAIL,QAAQ,CAACvC,OAAO,CAACG,KAAK,CAAC4D,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,IAAI;;IAEtD;IACA;IACA;IACA;IACA,MAAMgO,YAAY,GAAGxF,YAAY,CAACjO,IAAI;;IAEtC;IACA;IACA;IACA,MAAM0T,cAAc,GAAG,EAAE;IAEzB,MAAMC,aAAa,GAAGzP,IAAI,CAACO,IAAI;;IAE/B;IACA,MAAMmP,KAAK,GAAG,CAAC1P,IAAI,CAAC2P,SAAS,IAAI,EAAE,EAAE7D,MAAM,CAAC,CAACyC,GAAG,EAAE3T,CAAC,KAAK;MACtD2T,GAAG,CAAC3T,CAAC,CAAC,GAAGwF,eAAe,CAACxF,CAAC,CAAC;MAC3B,OAAO2T,GAAG;IACZ,CAAC,EAAE,CAAC,CAAC,CAAC;;IAEN;IACA,MAAMqB,iBAAiB,GAAG,EAAE;IAE5B,KAAK,MAAMpQ,GAAG,IAAIY,eAAe,EAAE;MACjC;MACA,IAAIZ,GAAG,CAACqB,UAAU,CAAC,YAAY,CAAC,EAAE;QAChC,IAAI0O,YAAY,EAAE;UAChB,MAAMxM,SAAS,GAAGvD,GAAG,CAACuB,SAAS,CAAC,EAAE,CAAC;UACnC,IAAI,CAACwO,YAAY,CAAC3P,QAAQ,CAACmD,SAAS,CAAC,EAAE;YACrC;YACAgH,YAAY,CAACjO,IAAI,IAAIiO,YAAY,CAACjO,IAAI,CAACK,IAAI,CAAC4G,SAAS,CAAC;YACtD;YACAyM,cAAc,CAACrT,IAAI,CAAC4G,SAAS,CAAC;UAChC;QACF;QACA;MACF;;MAEA;MACA,IAAIvD,GAAG,KAAK,GAAG,EAAE;QACfoQ,iBAAiB,CAACzT,IAAI,CAACiE,eAAe,CAACZ,GAAG,CAAC,CAAC;QAC5C;MACF;MAEA,IAAIiQ,aAAa,EAAE;QACjB,IAAIjQ,GAAG,KAAK,eAAe,EAAE;UAC3B;UACAoQ,iBAAiB,CAACzT,IAAI,CAACiE,eAAe,CAACZ,GAAG,CAAC,CAAC;UAC5C;QACF;QAEA,IAAIkQ,KAAK,CAAClQ,GAAG,CAAC,IAAIA,GAAG,CAACqB,UAAU,CAAC,OAAO,CAAC,EAAE;UACzC;UACA+O,iBAAiB,CAACzT,IAAI,CAACuT,KAAK,CAAClQ,GAAG,CAAC,CAAC;QACpC;MACF;IACF;;IAEA;IACA,IAAIiQ,aAAa,EAAE;MACjB,MAAMnP,MAAM,GAAGN,IAAI,CAACO,IAAI,CAACC,EAAE;MAC3B,IAAIC,KAAK,CAACL,eAAe,CAACE,MAAM,CAAC,EAAE;QACjCsP,iBAAiB,CAACzT,IAAI,CAACsE,KAAK,CAACL,eAAe,CAACE,MAAM,CAAC,CAAC;MACvD;IACF;;IAEA;IACA,IAAIkP,cAAc,CAACjT,MAAM,GAAG,CAAC,EAAE;MAC7BkE,KAAK,CAACL,eAAe,CAAC4B,aAAa,GAAGwN,cAAc;IACtD;IAEA,IAAIK,aAAa,GAAGD,iBAAiB,CAAC9D,MAAM,CAAC,CAACyC,GAAG,EAAEuB,IAAI,KAAK;MAC1D,IAAIA,IAAI,EAAE;QACRvB,GAAG,CAACpS,IAAI,CAAC,GAAG2T,IAAI,CAAC;MACnB;MACA,OAAOvB,GAAG;IACZ,CAAC,EAAE,EAAE,CAAC;;IAEN;IACAqB,iBAAiB,CAACpT,OAAO,CAACgF,MAAM,IAAI;MAClC,IAAIA,MAAM,EAAE;QACVqO,aAAa,GAAGA,aAAa,CAAC5T,MAAM,CAACwF,CAAC,IAAID,MAAM,CAAC5B,QAAQ,CAAC6B,CAAC,CAAC,CAAC;MAC/D;IACF,CAAC,CAAC;IAEF,OAAOoO,aAAa;EACtB;EAEAE,0BAA0BA,CAAA,EAAG;IAC3B,OAAO,IAAI,CAAChM,OAAO,CAACgM,0BAA0B,CAAC,CAAC,CAACxL,IAAI,CAACyL,oBAAoB,IAAI;MAC5E,IAAI,CAAC9L,qBAAqB,GAAG8L,oBAAoB;IACnD,CAAC,CAAC;EACJ;EAEAC,0BAA0BA,CAAA,EAAG;IAC3B,IAAI,CAAC,IAAI,CAAC/L,qBAAqB,EAAE;MAC/B,MAAM,IAAIhF,KAAK,CAAC,6CAA6C,CAAC;IAChE;IACA,OAAO,IAAI,CAAC6E,OAAO,CAACkM,0BAA0B,CAAC,IAAI,CAAC/L,qBAAqB,CAAC,CAACK,IAAI,CAAC,MAAM;MACpF,IAAI,CAACL,qBAAqB,GAAG,IAAI;IACnC,CAAC,CAAC;EACJ;EAEAgM,yBAAyBA,CAAA,EAAG;IAC1B,IAAI,CAAC,IAAI,CAAChM,qBAAqB,EAAE;MAC/B,MAAM,IAAIhF,KAAK,CAAC,4CAA4C,CAAC;IAC/D;IACA,OAAO,IAAI,CAAC6E,OAAO,CAACmM,yBAAyB,CAAC,IAAI,CAAChM,qBAAqB,CAAC,CAACK,IAAI,CAAC,MAAM;MACnF,IAAI,CAACL,qBAAqB,GAAG,IAAI;IACnC,CAAC,CAAC;EACJ;;EAEA;EACA;EACA,MAAMiM,qBAAqBA,CAAA,EAAG;IAC5B,MAAM,IAAI,CAACpM,OAAO,CAACoM,qBAAqB,CAAC;MACvCC,sBAAsB,EAAEnW,gBAAgB,CAACmW;IAC3C,CAAC,CAAC;IACF,MAAMC,kBAAkB,GAAG;MACzB7O,MAAM,EAAAnF,aAAA,CAAAA,aAAA,KACDpC,gBAAgB,CAACqW,cAAc,CAACC,QAAQ,GACxCtW,gBAAgB,CAACqW,cAAc,CAACE,KAAK;IAE5C,CAAC;IACD,MAAMC,kBAAkB,GAAG;MACzBjP,MAAM,EAAAnF,aAAA,CAAAA,aAAA,KACDpC,gBAAgB,CAACqW,cAAc,CAACC,QAAQ,GACxCtW,gBAAgB,CAACqW,cAAc,CAACI,KAAK;IAE5C,CAAC;IACD,MAAMC,yBAAyB,GAAG;MAChCnP,MAAM,EAAAnF,aAAA,CAAAA,aAAA,KACDpC,gBAAgB,CAACqW,cAAc,CAACC,QAAQ,GACxCtW,gBAAgB,CAACqW,cAAc,CAACM,YAAY;IAEnD,CAAC;IACD,MAAM,IAAI,CAACtM,UAAU,CAAC,CAAC,CAACC,IAAI,CAACrE,MAAM,IAAIA,MAAM,CAAC8I,kBAAkB,CAAC,OAAO,CAAC,CAAC;IAC1E,MAAM,IAAI,CAAC1E,UAAU,CAAC,CAAC,CAACC,IAAI,CAACrE,MAAM,IAAIA,MAAM,CAAC8I,kBAAkB,CAAC,OAAO,CAAC,CAAC;IAC1E,MAAM,IAAI,CAAC1E,UAAU,CAAC,CAAC,CAACC,IAAI,CAACrE,MAAM,IAAIA,MAAM,CAAC8I,kBAAkB,CAAC,cAAc,CAAC,CAAC;IAEjF,MAAM,IAAI,CAACjF,OAAO,CAAC8M,gBAAgB,CAAC,OAAO,EAAER,kBAAkB,EAAE,CAAC,UAAU,CAAC,CAAC,CAACxJ,KAAK,CAACR,KAAK,IAAI;MAC5FyK,eAAM,CAACC,IAAI,CAAC,6CAA6C,EAAE1K,KAAK,CAAC;MACjE,MAAMA,KAAK;IACb,CAAC,CAAC;IAEF,IAAI,CAAC,IAAI,CAAC3C,OAAO,CAACuJ,6BAA6B,EAAE;MAC/C,MAAM,IAAI,CAAClJ,OAAO,CACfiN,WAAW,CAAC,OAAO,EAAEX,kBAAkB,EAAE,CAAC,UAAU,CAAC,EAAE,2BAA2B,EAAE,IAAI,CAAC,CACzFxJ,KAAK,CAACR,KAAK,IAAI;QACdyK,eAAM,CAACC,IAAI,CAAC,oDAAoD,EAAE1K,KAAK,CAAC;QACxE,MAAMA,KAAK;MACb,CAAC,CAAC;MAEJ,MAAM,IAAI,CAACtC,OAAO,CACfiN,WAAW,CAAC,OAAO,EAAEX,kBAAkB,EAAE,CAAC,OAAO,CAAC,EAAE,wBAAwB,EAAE,IAAI,CAAC,CACnFxJ,KAAK,CAACR,KAAK,IAAI;QACdyK,eAAM,CAACC,IAAI,CAAC,iDAAiD,EAAE1K,KAAK,CAAC;QACrE,MAAMA,KAAK;MACb,CAAC,CAAC;IACN;IAEA,MAAM,IAAI,CAACtC,OAAO,CAAC8M,gBAAgB,CAAC,OAAO,EAAER,kBAAkB,EAAE,CAAC,OAAO,CAAC,CAAC,CAACxJ,KAAK,CAACR,KAAK,IAAI;MACzFyK,eAAM,CAACC,IAAI,CAAC,wDAAwD,EAAE1K,KAAK,CAAC;MAC5E,MAAMA,KAAK;IACb,CAAC,CAAC;IAEF,MAAM,IAAI,CAACtC,OAAO,CAAC8M,gBAAgB,CAAC,OAAO,EAAEJ,kBAAkB,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC5J,KAAK,CAACR,KAAK,IAAI;MACxFyK,eAAM,CAACC,IAAI,CAAC,6CAA6C,EAAE1K,KAAK,CAAC;MACjE,MAAMA,KAAK;IACb,CAAC,CAAC;IAEF,MAAM,IAAI,CAACtC,OAAO,CACf8M,gBAAgB,CAAC,cAAc,EAAEF,yBAAyB,EAAE,CAAC,OAAO,CAAC,CAAC,CACtE9J,KAAK,CAACR,KAAK,IAAI;MACdyK,eAAM,CAACC,IAAI,CAAC,0DAA0D,EAAE1K,KAAK,CAAC;MAC9E,MAAMA,KAAK;IACb,CAAC,CAAC;IAEJ,MAAM4K,cAAc,GAAG,IAAI,CAAClN,OAAO,YAAYmN,4BAAmB;IAClE,MAAMC,iBAAiB,GAAG,IAAI,CAACpN,OAAO,YAAYqN,+BAAsB;IACxE,IAAIH,cAAc,IAAIE,iBAAiB,EAAE;MACvC,IAAIzN,OAAO,GAAG,CAAC,CAAC;MAChB,IAAIuN,cAAc,EAAE;QAClBvN,OAAO,GAAG;UACR2N,GAAG,EAAE;QACP,CAAC;MACH,CAAC,MAAM,IAAIF,iBAAiB,EAAE;QAC5BzN,OAAO,GAAG,IAAI,CAACM,kBAAkB;QACjCN,OAAO,CAAC4N,sBAAsB,GAAG,IAAI;MACvC;MACA,MAAM,IAAI,CAACvN,OAAO,CACfiN,WAAW,CAAC,cAAc,EAAEL,yBAAyB,EAAE,CAAC,QAAQ,CAAC,EAAE,KAAK,EAAE,KAAK,EAAEjN,OAAO,CAAC,CACzFmD,KAAK,CAACR,KAAK,IAAI;QACdyK,eAAM,CAACC,IAAI,CAAC,0DAA0D,EAAE1K,KAAK,CAAC;QAC9E,MAAMA,KAAK;MACb,CAAC,CAAC;IACN;IACA,MAAM,IAAI,CAACtC,OAAO,CAACwN,uBAAuB,CAAC,CAAC;EAC9C;EAEAC,sBAAsBA,CAACnR,MAAW,EAAEb,GAAW,EAAE3C,KAAU,EAAO;IAChE,IAAI2C,GAAG,CAAChC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;MACxB6C,MAAM,CAACb,GAAG,CAAC,GAAG3C,KAAK,CAAC2C,GAAG,CAAC;MACxB,OAAOa,MAAM;IACf;IACA,MAAMoR,IAAI,GAAGjS,GAAG,CAAC6D,KAAK,CAAC,GAAG,CAAC;IAC3B,MAAMqO,QAAQ,GAAGD,IAAI,CAAC,CAAC,CAAC;IACxB,MAAME,QAAQ,GAAGF,IAAI,CAACG,KAAK,CAAC,CAAC,CAAC,CAAC5D,IAAI,CAAC,GAAG,CAAC;;IAExC;IACA,IAAI,IAAI,CAACtK,OAAO,IAAI,IAAI,CAACA,OAAO,CAACmO,sBAAsB,EAAE;MACvD;MACA,KAAK,MAAMC,OAAO,IAAI,IAAI,CAACpO,OAAO,CAACmO,sBAAsB,EAAE;QACzD,MAAMlS,KAAK,GAAGwG,cAAK,CAAC4L,sBAAsB,CACxC;UAAE,CAACL,QAAQ,GAAG,IAAI;UAAE,CAACC,QAAQ,GAAG;QAAK,CAAC,EACtCG,OAAO,CAACtS,GAAG,EACX,IACF,CAAC;QACD,IAAIG,KAAK,EAAE;UACT,MAAM,IAAIV,WAAK,CAACC,KAAK,CACnBD,WAAK,CAACC,KAAK,CAACW,gBAAgB,EAC5B,uCAAuCiO,IAAI,CAACC,SAAS,CAAC+D,OAAO,CAAC,GAChE,CAAC;QACH;MACF;IACF;IAEAzR,MAAM,CAACqR,QAAQ,CAAC,GAAG,IAAI,CAACF,sBAAsB,CAC5CnR,MAAM,CAACqR,QAAQ,CAAC,IAAI,CAAC,CAAC,EACtBC,QAAQ,EACR9U,KAAK,CAAC6U,QAAQ,CAChB,CAAC;IACD,OAAOrR,MAAM,CAACb,GAAG,CAAC;IAClB,OAAOa,MAAM;EACf;EAEAoH,uBAAuBA,CAACkB,cAAmB,EAAEpK,MAAW,EAAgB;IACtE,MAAMyT,QAAQ,GAAG,CAAC,CAAC;IACnB,IAAI,CAACzT,MAAM,EAAE;MACX,OAAOsG,OAAO,CAACG,OAAO,CAACgN,QAAQ,CAAC;IAClC;IACA3W,MAAM,CAACS,IAAI,CAAC6M,cAAc,CAAC,CAACnM,OAAO,CAACgD,GAAG,IAAI;MACzC,MAAMyS,SAAS,GAAGtJ,cAAc,CAACnJ,GAAG,CAAC;MACrC;MACA,IACEyS,SAAS,IACT,OAAOA,SAAS,KAAK,QAAQ,IAC7BA,SAAS,CAAC1P,IAAI,IACd,CAAC,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAE,WAAW,EAAE,aAAa,CAAC,CAAC/E,OAAO,CAACyU,SAAS,CAAC1P,IAAI,CAAC,GAAG,CAAC,CAAC,EACvF;QACA;QACA;QACA,IAAI,CAACiP,sBAAsB,CAACQ,QAAQ,EAAExS,GAAG,EAAEjB,MAAM,CAAC;QAClD;QACA,IAAIiB,GAAG,CAACI,QAAQ,CAAC,GAAG,CAAC,EAAE;UACrB,MAAM,CAAC2J,KAAK,EAAEqB,KAAK,CAAC,GAAGpL,GAAG,CAAC6D,KAAK,CAAC,GAAG,CAAC;UACrC,MAAM6O,YAAY,GAAG7S,KAAK,CAAC8S,IAAI,CAACvH,KAAK,CAAC,CAACwH,KAAK,CAACC,CAAC,IAAIA,CAAC,IAAI,GAAG,IAAIA,CAAC,IAAI,GAAG,CAAC;UACvE,IAAIH,YAAY,IAAI7S,KAAK,CAACgC,OAAO,CAAC9C,MAAM,CAACgL,KAAK,CAAC,CAAC,IAAI,CAAClK,KAAK,CAACgC,OAAO,CAAC2Q,QAAQ,CAACzI,KAAK,CAAC,CAAC,EAAE;YACnFyI,QAAQ,CAACzI,KAAK,CAAC,GAAGhL,MAAM,CAACgL,KAAK,CAAC;UACjC;QACF;MACF;IACF,CAAC,CAAC;IACF,OAAO1E,OAAO,CAACG,OAAO,CAACgN,QAAQ,CAAC;EAClC;AAIF;AAEAM,MAAM,CAACC,OAAO,GAAG1O,kBAAkB;AACnC;AACAyO,MAAM,CAACC,OAAO,CAACC,cAAc,GAAG3T,aAAa;AAC7CyT,MAAM,CAACC,OAAO,CAACzS,mBAAmB,GAAGA,mBAAmB","ignoreList":[]}