UNPKG

@decaf-ts/core

Version:

Core persistence module for the decaf framework

661 lines 84.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.createOrUpdate = createOrUpdate; exports.oneToOneOnCreate = oneToOneOnCreate; exports.oneToOneOnUpdate = oneToOneOnUpdate; exports.oneToOneOnDelete = oneToOneOnDelete; exports.oneToManyOnCreate = oneToManyOnCreate; exports.oneToManyOnUpdate = oneToManyOnUpdate; exports.oneToManyOnDelete = oneToManyOnDelete; exports.getPopulateKey = getPopulateKey; exports.cacheModelForPopulate = cacheModelForPopulate; exports.populate = populate; exports.repositoryFromTypeMetadata = repositoryFromTypeMetadata; const decorator_validation_1 = require("@decaf-ts/decorator-validation"); const Repository_1 = require("./../repository/Repository.cjs"); const db_decorators_1 = require("@decaf-ts/db-decorators"); const constants_1 = require("./../persistence/constants.cjs"); const constants_2 = require("./../repository/constants.cjs"); /** * @description Creates or updates a model instance * @summary Determines whether to create a new model or update an existing one based on the presence of a primary key * @template M - The model type extending Model * @template F - The repository flags type * @param {M} model - The model instance to create or update * @param {Context<F>} context - The context for the operation * @param {Repo<M, F, Context<F>>} [repository] - Optional repository to use for the operation * @return {Promise<M>} A promise that resolves to the created or updated model * @function createOrUpdate * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant createOrUpdate * participant Repository * participant Model * * Caller->>createOrUpdate: model, context, repository? * alt repository not provided * createOrUpdate->>Model: get(model.constructor.name) * Model-->>createOrUpdate: constructor * createOrUpdate->>Repository: forModel(constructor) * Repository-->>createOrUpdate: repository * end * * alt primary key undefined * createOrUpdate->>Repository: create(model, context) * Repository-->>createOrUpdate: created model * else primary key defined * createOrUpdate->>Repository: update(model, context) * alt update successful * Repository-->>createOrUpdate: updated model * else NotFoundError * createOrUpdate->>Repository: create(model, context) * Repository-->>createOrUpdate: created model * end * end * * createOrUpdate-->>Caller: model */ async function createOrUpdate(model, context, alias, repository) { if (!repository) { const constructor = decorator_validation_1.Model.get(model.constructor.name); if (!constructor) throw new db_decorators_1.InternalError(`Could not find model ${model.constructor.name}`); repository = Repository_1.Repository.forModel(constructor, alias); } if (typeof model[repository.pk] === "undefined") return repository.create(model, context); else { try { return repository.update(model, context); } catch (e) { if (!(e instanceof db_decorators_1.NotFoundError)) throw e; return repository.create(model, context); } } } /** * @description Handles one-to-one relationship creation * @summary Processes a one-to-one relationship when creating a model, either by referencing an existing model or creating a new one * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param {string} key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function oneToOneOnCreate * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant oneToOneOnCreate * participant repositoryFromTypeMetadata * participant Model * participant Repository * participant cacheModelForPopulate * * Caller->>oneToOneOnCreate: this, context, data, key, model * oneToOneOnCreate->>oneToOneOnCreate: check if propertyValue exists * * alt propertyValue is not an object * oneToOneOnCreate->>repositoryFromTypeMetadata: model, key * repositoryFromTypeMetadata-->>oneToOneOnCreate: innerRepo * oneToOneOnCreate->>innerRepo: read(propertyValue) * innerRepo-->>oneToOneOnCreate: read * oneToOneOnCreate->>cacheModelForPopulate: context, model, key, propertyValue, read * oneToOneOnCreate->>oneToOneOnCreate: set model[key] = propertyValue * else propertyValue is an object * oneToOneOnCreate->>Model: get(data.class) * Model-->>oneToOneOnCreate: constructor * oneToOneOnCreate->>Repository: forModel(constructor) * Repository-->>oneToOneOnCreate: repo * oneToOneOnCreate->>repo: create(propertyValue) * repo-->>oneToOneOnCreate: created * oneToOneOnCreate->>findPrimaryKey: created * findPrimaryKey-->>oneToOneOnCreate: pk * oneToOneOnCreate->>cacheModelForPopulate: context, model, key, created[pk], created * oneToOneOnCreate->>oneToOneOnCreate: set model[key] = created[pk] * end * * oneToOneOnCreate-->>Caller: void */ async function oneToOneOnCreate(context, data, key, model) { const propertyValue = model[key]; if (!propertyValue) return; if (typeof propertyValue !== "object") { const innerRepo = repositoryFromTypeMetadata(model, key, this.adapter.alias); const read = await innerRepo.read(propertyValue); await cacheModelForPopulate(context, model, key, propertyValue, read); model[key] = propertyValue; return; } data.class = typeof data.class === "string" ? data.class : data.class().name; const constructor = decorator_validation_1.Model.get(data.class); if (!constructor) throw new db_decorators_1.InternalError(`Could not find model ${data.class}`); const repo = Repository_1.Repository.forModel(constructor, this.adapter.alias); const created = await repo.create(propertyValue); const pk = (0, db_decorators_1.findPrimaryKey)(created).id; await cacheModelForPopulate(context, model, key, created[pk], created); model[key] = created[pk]; } /** * @description Handles one-to-one relationship updates * @summary Processes a one-to-one relationship when updating a model, either by referencing an existing model or updating the related model * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function oneToOneOnUpdate * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant oneToOneOnUpdate * participant repositoryFromTypeMetadata * participant createOrUpdate * participant findPrimaryKey * participant cacheModelForPopulate * * Caller->>oneToOneOnUpdate: this, context, data, key, model * oneToOneOnUpdate->>oneToOneOnUpdate: check if propertyValue exists * oneToOneOnUpdate->>oneToOneOnUpdate: check if cascade.update is CASCADE * * alt propertyValue is not an object * oneToOneOnUpdate->>repositoryFromTypeMetadata: model, key * repositoryFromTypeMetadata-->>oneToOneOnUpdate: innerRepo * oneToOneOnUpdate->>innerRepo: read(propertyValue) * innerRepo-->>oneToOneOnUpdate: read * oneToOneOnUpdate->>cacheModelForPopulate: context, model, key, propertyValue, read * oneToOneOnUpdate->>oneToOneOnUpdate: set model[key] = propertyValue * else propertyValue is an object * oneToOneOnUpdate->>createOrUpdate: model[key], context * createOrUpdate-->>oneToOneOnUpdate: updated * oneToOneOnUpdate->>findPrimaryKey: updated * findPrimaryKey-->>oneToOneOnUpdate: pk * oneToOneOnUpdate->>cacheModelForPopulate: context, model, key, updated[pk], updated * oneToOneOnUpdate->>oneToOneOnUpdate: set model[key] = updated[pk] * end * * oneToOneOnUpdate-->>Caller: void */ async function oneToOneOnUpdate(context, data, key, model) { const propertyValue = model[key]; if (!propertyValue) return; if (data.cascade.update !== constants_2.Cascade.CASCADE) return; if (typeof propertyValue !== "object") { const innerRepo = repositoryFromTypeMetadata(model, key, this.adapter.alias); const read = await innerRepo.read(propertyValue); await cacheModelForPopulate(context, model, key, propertyValue, read); model[key] = propertyValue; return; } const updated = await createOrUpdate(model[key], context, this.adapter.alias); const pk = (0, db_decorators_1.findPrimaryKey)(updated).id; await cacheModelForPopulate(context, model, key, updated[pk], updated); model[key] = updated[pk]; } /** * @description Handles one-to-one relationship deletion * @summary Processes a one-to-one relationship when deleting a model, deleting the related model if cascade is enabled * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function oneToOneOnDelete * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant oneToOneOnDelete * participant repositoryFromTypeMetadata * participant cacheModelForPopulate * * Caller->>oneToOneOnDelete: this, context, data, key, model * oneToOneOnDelete->>oneToOneOnDelete: check if propertyValue exists * oneToOneOnDelete->>oneToOneOnDelete: check if cascade.update is CASCADE * * oneToOneOnDelete->>repositoryFromTypeMetadata: model, key * repositoryFromTypeMetadata-->>oneToOneOnDelete: innerRepo * * alt propertyValue is not a Model instance * oneToOneOnDelete->>innerRepo: delete(model[key], context) * innerRepo-->>oneToOneOnDelete: deleted * else propertyValue is a Model instance * oneToOneOnDelete->>innerRepo: delete(model[key][innerRepo.pk], context) * innerRepo-->>oneToOneOnDelete: deleted * end * * oneToOneOnDelete->>cacheModelForPopulate: context, model, key, deleted[innerRepo.pk], deleted * oneToOneOnDelete-->>Caller: void */ async function oneToOneOnDelete(context, data, key, model) { const propertyValue = model[key]; if (!propertyValue) return; if (data.cascade.update !== constants_2.Cascade.CASCADE) return; const innerRepo = repositoryFromTypeMetadata(model, key, this.adapter.alias); let deleted; if (!(propertyValue instanceof decorator_validation_1.Model)) deleted = await innerRepo.delete(model[key], context); else deleted = await innerRepo.delete(model[key][innerRepo.pk], context); await cacheModelForPopulate(context, model, key, deleted[innerRepo.pk], deleted); } /** * @description Handles one-to-many relationship creation * @summary Processes a one-to-many relationship when creating a model, either by referencing existing models or creating new ones * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function oneToManyOnCreate * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant oneToManyOnCreate * participant repositoryFromTypeMetadata * participant createOrUpdate * participant findPrimaryKey * participant cacheModelForPopulate * * Caller->>oneToManyOnCreate: this, context, data, key, model * oneToManyOnCreate->>oneToManyOnCreate: check if propertyValues exists and has length * oneToManyOnCreate->>oneToManyOnCreate: check if all elements have same type * oneToManyOnCreate->>oneToManyOnCreate: create uniqueValues set * * alt arrayType is not "object" * oneToManyOnCreate->>repositoryFromTypeMetadata: model, key * repositoryFromTypeMetadata-->>oneToManyOnCreate: repo * loop for each id in uniqueValues * oneToManyOnCreate->>repo: read(id) * repo-->>oneToManyOnCreate: read * oneToManyOnCreate->>cacheModelForPopulate: context, model, key, id, read * end * oneToManyOnCreate->>oneToManyOnCreate: set model[key] = [...uniqueValues] * else arrayType is "object" * oneToManyOnCreate->>findPrimaryKey: propertyValues[0] * findPrimaryKey-->>oneToManyOnCreate: pkName * oneToManyOnCreate->>oneToManyOnCreate: create result set * loop for each m in propertyValues * oneToManyOnCreate->>createOrUpdate: m, context * createOrUpdate-->>oneToManyOnCreate: record * oneToManyOnCreate->>cacheModelForPopulate: context, model, key, record[pkName], record * oneToManyOnCreate->>oneToManyOnCreate: add record[pkName] to result * end * oneToManyOnCreate->>oneToManyOnCreate: set model[key] = [...result] * end * * oneToManyOnCreate-->>Caller: void */ async function oneToManyOnCreate(context, data, key, model) { const propertyValues = model[key]; if (!propertyValues || !propertyValues.length) return; const arrayType = typeof propertyValues[0]; if (!propertyValues.every((item) => typeof item === arrayType)) throw new db_decorators_1.InternalError(`Invalid operation. All elements of property ${key} must match the same type.`); const uniqueValues = new Set([...propertyValues]); if (arrayType !== "object") { const repo = repositoryFromTypeMetadata(model, key, this.adapter.alias); for (const id of uniqueValues) { const read = await repo.read(id); await cacheModelForPopulate(context, model, key, id, read); } model[key] = [...uniqueValues]; return; } const pkName = (0, db_decorators_1.findPrimaryKey)(propertyValues[0]).id; const result = new Set(); for (const m of propertyValues) { const record = await createOrUpdate(m, context, this.adapter.alias); await cacheModelForPopulate(context, model, key, record[pkName], record); result.add(record[pkName]); } model[key] = [...result]; } /** * @description Handles one-to-many relationship updates * @summary Processes a one-to-many relationship when updating a model, delegating to oneToManyOnCreate if cascade update is enabled * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function oneToManyOnUpdate * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant oneToManyOnUpdate * participant oneToManyOnCreate * * Caller->>oneToManyOnUpdate: this, context, data, key, model * oneToManyOnUpdate->>oneToManyOnUpdate: check if cascade.update is CASCADE * * alt cascade.update is CASCADE * oneToManyOnUpdate->>oneToManyOnCreate: apply(this, [context, data, key, model]) * oneToManyOnCreate-->>oneToManyOnUpdate: void * end * * oneToManyOnUpdate-->>Caller: void */ async function oneToManyOnUpdate(context, data, key, model) { const { cascade } = data; if (cascade.update !== constants_2.Cascade.CASCADE) return; return oneToManyOnCreate.apply(this, [ context, data, key, model, ]); } /** * @description Handles one-to-many relationship deletion * @summary Processes a one-to-many relationship when deleting a model, deleting all related models if cascade delete is enabled * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function oneToManyOnDelete * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant oneToManyOnDelete * participant Repository * participant repositoryFromTypeMetadata * participant cacheModelForPopulate * * Caller->>oneToManyOnDelete: this, context, data, key, model * oneToManyOnDelete->>oneToManyOnDelete: check if cascade.delete is CASCADE * oneToManyOnDelete->>oneToManyOnDelete: check if values exists and has length * oneToManyOnDelete->>oneToManyOnDelete: check if all elements have same type * * alt isInstantiated (arrayType is "object") * oneToManyOnDelete->>Repository: forModel(values[0]) * Repository-->>oneToManyOnDelete: repo * else not instantiated * oneToManyOnDelete->>repositoryFromTypeMetadata: model, key * repositoryFromTypeMetadata-->>oneToManyOnDelete: repo * end * * oneToManyOnDelete->>oneToManyOnDelete: create uniqueValues set * * loop for each id in uniqueValues * oneToManyOnDelete->>repo: delete(id, context) * repo-->>oneToManyOnDelete: deleted * oneToManyOnDelete->>cacheModelForPopulate: context, model, key, id, deleted * end * * oneToManyOnDelete->>oneToManyOnDelete: set model[key] = [...uniqueValues] * oneToManyOnDelete-->>Caller: void */ async function oneToManyOnDelete(context, data, key, model) { if (data.cascade.delete !== constants_2.Cascade.CASCADE) return; const values = model[key]; if (!values || !values.length) return; const arrayType = typeof values[0]; const areAllSameType = values.every((item) => typeof item === arrayType); if (!areAllSameType) throw new db_decorators_1.InternalError(`Invalid operation. All elements of property ${key} must match the same type.`); const isInstantiated = arrayType === "object"; const repo = isInstantiated ? Repository_1.Repository.forModel(values[0], this.adapter.alias) : repositoryFromTypeMetadata(model, key, this.adapter.alias); const uniqueValues = new Set([ ...(isInstantiated ? values.map((v) => v[repo.pk]) : values), ]); for (const id of uniqueValues.values()) { const deleted = await repo.delete(id, context); await cacheModelForPopulate(context, model, key, id, deleted); } model[key] = [...uniqueValues]; } /** * @description Generates a key for caching populated model relationships * @summary Creates a unique key for storing and retrieving populated model relationships in the cache * @param {string} tableName - The name of the table or model * @param {string} fieldName - The name of the field or property * @param {string|number} id - The identifier of the related model * @return {string} A dot-separated string that uniquely identifies the relationship * @function getPopulateKey * @memberOf module:core */ function getPopulateKey(tableName, fieldName, id) { return [constants_1.PersistenceKeys.POPULATE, tableName, fieldName, id].join("."); } /** * @description Caches a model for later population * @summary Stores a model in the context cache for efficient retrieval during relationship population * @template M - The model type extending Model * @template F - The repository flags type * @param {Context<F>} context - The context for the operation * @param {M} parentModel - The parent model that contains the relationship * @param propertyKey - The property key of the relationship * @param {string | number} pkValue - The primary key value of the related model * @param {any} cacheValue - The model instance to cache * @return {Promise<any>} A promise that resolves with the result of the cache operation * @function cacheModelForPopulate * @memberOf module:core */ async function cacheModelForPopulate(context, parentModel, propertyKey, pkValue, cacheValue) { const cacheKey = getPopulateKey(parentModel.constructor.name, propertyKey, pkValue); return context.accumulate({ [cacheKey]: cacheValue }); } /** * @description Populates a model's relationship * @summary Retrieves and attaches related models to a model's relationship property * @template M - The model type extending Model * @template R - The repository type extending Repo<M, F, C> * @template V - The relations metadata type extending RelationsMetadata * @template F - The repository flags type * @template C - The context type extending Context<F> * @param {R} this - The repository instance * @param {Context<F>} context - The context for the operation * @param {V} data - The relations metadata * @param key - The property key of the relationship * @param {M} model - The model instance * @return {Promise<void>} A promise that resolves when the operation is complete * @function populate * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant populate * participant fetchPopulateValues * participant getPopulateKey * participant Context * participant repositoryFromTypeMetadata * * Caller->>populate: this, context, data, key, model * populate->>populate: check if data.populate is true * populate->>populate: get nested value and check if it exists * * populate->>fetchPopulateValues: context, model, key, isArr ? nested : [nested] * * fetchPopulateValues->>fetchPopulateValues: initialize variables * * loop for each proKeyValue in propKeyValues * fetchPopulateValues->>getPopulateKey: model.constructor.name, propName, proKeyValue * getPopulateKey-->>fetchPopulateValues: cacheKey * * alt try to get from cache * fetchPopulateValues->>Context: get(cacheKey) * Context-->>fetchPopulateValues: val * else catch error * fetchPopulateValues->>repositoryFromTypeMetadata: model, propName * repositoryFromTypeMetadata-->>fetchPopulateValues: repo * fetchPopulateValues->>repo: read(proKeyValue) * repo-->>fetchPopulateValues: val * end * * fetchPopulateValues->>fetchPopulateValues: add val to results * end * * fetchPopulateValues-->>populate: results * populate->>populate: set model[key] = isArr ? res : res[0] * populate-->>Caller: void */ async function populate(context, data, key, model) { if (!data.populate) return; const nested = model[key]; const isArr = Array.isArray(nested); if (typeof nested === "undefined" || (isArr && nested.length === 0)) return; async function fetchPopulateValues(c, model, propName, propKeyValues, alias) { let cacheKey; let val; const results = []; for (const proKeyValue of propKeyValues) { cacheKey = getPopulateKey(model.constructor.name, propName, proKeyValue); try { val = await c.get(cacheKey); // eslint-disable-next-line @typescript-eslint/no-unused-vars } catch (e) { const repo = repositoryFromTypeMetadata(model, propName, alias); if (!repo) throw new db_decorators_1.InternalError("Could not find repo"); val = await repo.read(proKeyValue); } results.push(val); } return results; } const res = await fetchPopulateValues(context, model, key, isArr ? nested : [nested], this.adapter.alias); model[key] = isArr ? res : res[0]; } /** * @description List of common JavaScript types * @summary An array of strings representing common JavaScript types that are not custom model types * @const commomTypes * @memberOf module:core */ const commomTypes = [ "array", "string", "number", "boolean", "symbol", "function", "object", "undefined", "null", "bigint", ]; /** * @description Retrieves a repository for a model property based on its type metadata * @summary Examines a model property's type metadata to determine the appropriate repository for related models * @template M - The model type extending Model * @param {any} model - The model instance containing the property * @param propertyKey - The property key to examine * @return {Repo<M>} A repository for the model type associated with the property * @function repositoryFromTypeMetadata * @memberOf module:core * @mermaid * sequenceDiagram * participant Caller * participant repositoryFromTypeMetadata * participant Reflect * participant Validation * participant Model * participant Repository * * Caller->>repositoryFromTypeMetadata: model, propertyKey * * repositoryFromTypeMetadata->>Validation: key(Array.isArray(model[propertyKey]) ? ValidationKeys.LIST : ValidationKeys.TYPE) * Validation-->>repositoryFromTypeMetadata: validationKey * * repositoryFromTypeMetadata->>Reflect: getMetadata(validationKey, model, propertyKey) * Reflect-->>repositoryFromTypeMetadata: types * * repositoryFromTypeMetadata->>repositoryFromTypeMetadata: determine customTypes based on property type * repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if types and customTypes exist * * repositoryFromTypeMetadata->>repositoryFromTypeMetadata: create allowedTypes array * repositoryFromTypeMetadata->>repositoryFromTypeMetadata: find constructorName not in commomTypes * repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if constructorName exists * * repositoryFromTypeMetadata->>Model: get(constructorName) * Model-->>repositoryFromTypeMetadata: constructor * repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if constructor exists * * repositoryFromTypeMetadata->>Repository: forModel(constructor) * Repository-->>repositoryFromTypeMetadata: repo * * repositoryFromTypeMetadata-->>Caller: repo */ function repositoryFromTypeMetadata(model, propertyKey, alias) { const types = Reflect.getMetadata(decorator_validation_1.Validation.key(Array.isArray(model[propertyKey]) ? decorator_validation_1.ValidationKeys.LIST : decorator_validation_1.ValidationKeys.TYPE), model, propertyKey); const customTypes = Array.isArray(model[propertyKey]) ? types.clazz : types.customTypes; if (!types || !customTypes) throw new db_decorators_1.InternalError(`Failed to find types decorators for property ${propertyKey}`); const allowedTypes = (Array.isArray(customTypes) ? [...customTypes] : [customTypes]).map((t) => (typeof t === "function" ? t() : t)); const constructorName = allowedTypes.find((t) => !commomTypes.includes(`${t}`.toLowerCase())); if (!constructorName) throw new db_decorators_1.InternalError(`Property key ${propertyKey} does not have a valid constructor type`); const constructor = decorator_validation_1.Model.get(constructorName); if (!constructor) throw new db_decorators_1.InternalError(`No registered model found for ${constructorName}`); return Repository_1.Repository.forModel(constructor, alias); } //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RydWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL21vZGVsL2NvbnN0cnVjdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQTREQSx3Q0E0QkM7QUFvREQsNENBdUNDO0FBaURELDRDQTJDQztBQTJDRCw0Q0FvQ0M7QUF3REQsOENBMENDO0FBa0NELDhDQXFCQztBQWtERCw4Q0FzQ0M7QUFZRCx3Q0FNQztBQWdCRCxzREFnQkM7QUF3REQsNEJBa0RDO0FBK0RELGdFQXFDQztBQS8wQkQseUVBTXdDO0FBQ3hDLCtEQUE0RDtBQUU1RCwyREFLaUM7QUFDakMsOERBQTJEO0FBQzNELDZEQUFrRDtBQUdsRDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdDRztBQUNJLEtBQUssVUFBVSxjQUFjLENBSWxDLEtBQVEsRUFDUixPQUFtQixFQUNuQixLQUFjLEVBQ2QsVUFBbUM7SUFFbkMsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO1FBQ2hCLE1BQU0sV0FBVyxHQUFHLDRCQUFLLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDdEQsSUFBSSxDQUFDLFdBQVc7WUFDZCxNQUFNLElBQUksNkJBQWEsQ0FBQyx3QkFBd0IsS0FBSyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO1FBQzVFLFVBQVUsR0FBRyx1QkFBVSxDQUFDLFFBQVEsQ0FDOUIsV0FBNkMsRUFDN0MsS0FBSyxDQUNOLENBQUM7SUFDSixDQUFDO0lBQ0QsSUFBSSxPQUFPLEtBQUssQ0FBQyxVQUFVLENBQUMsRUFBRSxDQUFDLEtBQUssV0FBVztRQUM3QyxPQUFPLFVBQVUsQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1NBQ3RDLENBQUM7UUFDSixJQUFJLENBQUM7WUFDSCxPQUFPLFVBQVUsQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFF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QUFDLFdBQVcsRUFBRSxLQUFLLENBQUMsQ0FBQztBQUNqRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29uc3RydWN0b3IsXG4gIE1vZGVsLFxuICBNb2RlbENvbnN0cnVjdG9yLFxuICBWYWxpZGF0aW9uLFxuICBWYWxpZGF0aW9uS2V5cyxcbn0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgUmVwbywgUmVwb3NpdG9yeSB9IGZyb20gXCIuLi9yZXBvc2l0b3J5L1JlcG9zaXRvcnlcIjtcbmltcG9ydCB7IFJlbGF0aW9uc01ldGFkYXRhIH0gZnJvbSBcIi4vdHlwZXNcIjtcbmltcG9ydCB7XG4gIGZpbmRQcmltYXJ5S2V5LFxuICBJbnRlcm5hbEVycm9yLFxuICBOb3RGb3VuZEVycm9yLFxuICBSZXBvc2l0b3J5RmxhZ3MsXG59IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgUGVyc2lzdGVuY2VLZXlzIH0gZnJvbSBcIi4uL3BlcnNpc3RlbmNlL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgQ2FzY2FkZSB9IGZyb20gXCIuLi9yZXBvc2l0b3J5L2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgQ29udGV4dCB9IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDcmVhdGVzIG9yIHVwZGF0ZXMgYSBtb2RlbCBpbnN0YW5jZVxuICogQHN1bW1hcnkgRGV0ZXJtaW5lcyB3aGV0aGVyIHRvIGNyZWF0ZSBhIG5ldyBtb2RlbCBvciB1cGRhdGUgYW4gZXhpc3Rpbmcgb25lIGJhc2VkIG9uIHRoZSBwcmVzZW5jZSBvZiBhIHByaW1hcnkga2V5XG4gKiBAdGVtcGxhdGUgTSAtIFRoZSBtb2RlbCB0eXBlIGV4dGVuZGluZyBNb2RlbFxuICogQHRlbXBsYXRlIEYgLSBUaGUgcmVwb3NpdG9yeSBmbGFncyB0eXBlXG4gKiBAcGFyYW0ge019IG1vZGVsIC0gVGhlIG1vZGVsIGluc3RhbmNlIHRvIGNyZWF0ZSBvciB1cGRhdGVcbiAqIEBwYXJhbSB7Q29udGV4dDxGPn0gY29udGV4dCAtIFRoZSBjb250ZXh0IGZvciB0aGUgb3BlcmF0aW9uXG4gKiBAcGFyYW0ge1JlcG88TSwgRiwgQ29udGV4dDxGPj59IFtyZXBvc2l0b3J5XSAtIE9wdGlvbmFsIHJlcG9zaXRvcnkgdG8gdXNlIGZvciB0aGUgb3BlcmF0aW9uXG4gKiBAcmV0dXJuIHtQcm9taXNlPE0+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgY3JlYXRlZCBvciB1cGRhdGVkIG1vZGVsXG4gKiBAZnVuY3Rpb24gY3JlYXRlT3JVcGRhdGVcbiAqIEBtZW1iZXJPZiBtb2R1bGU6Y29yZVxuICogQG1lcm1haWRcbiAqIHNlcXVlbmNlRGlhZ3JhbVxuICogICBwYXJ0aWNpcGFudCBDYWxsZXJcbiAqICAgcGFydGljaXBhbnQgY3JlYXRlT3JVcGRhdGVcbiAqICAgcGFydGljaXBhbnQgUmVwb3NpdG9yeVxuICogICBwYXJ0aWNpcGFudCBNb2RlbFxuICpcbiAqICAgQ2FsbGVyLT4+Y3JlYXRlT3JVcGRhdGU6IG1vZGVsLCBjb250ZXh0LCByZXBvc2l0b3J5P1xuICogICBhbHQgcmVwb3NpdG9yeSBub3QgcHJvdmlkZWRcbiAqICAgICBjcmVhdGVPclVwZGF0ZS0+Pk1vZGVsOiBnZXQobW9kZWwuY29uc3RydWN0b3IubmFtZSlcbiAqICAgICBNb2RlbC0tPj5jcmVhdGVPclVwZGF0ZTogY29uc3RydWN0b3JcbiAqICAgICBjcmVhdGVPclVwZGF0ZS0+PlJlcG9zaXRvcnk6IGZvck1vZGVsKGNvbnN0cnVjdG9yKVxuICogICAgIFJlcG9zaXRvcnktLT4+Y3JlYXRlT3JVcGRhdGU6IHJlcG9zaXRvcnlcbiAqICAgZW5kXG4gKlxuICogICBhbHQgcHJpbWFyeSBrZXkgdW5kZWZpbmVkXG4gKiAgICAgY3JlYXRlT3JVcGRhdGUtPj5SZXBvc2l0b3J5OiBjcmVhdGUobW9kZWwsIGNvbnRleHQpXG4gKiAgICAgUmVwb3NpdG9yeS0tPj5jcmVhdGVPclVwZGF0ZTogY3JlYXRlZCBtb2RlbFxuICogICBlbHNlIHByaW1hcnkga2V5IGRlZmluZWRcbiAqICAgICBjcmVhdGVPclVwZGF0ZS0+PlJlcG9zaXRvcnk6IHVwZGF0ZShtb2RlbCwgY29udGV4dClcbiAqICAgICBhbHQgdXBkYXRlIHN1Y2Nlc3NmdWxcbiAqICAgICAgIFJlcG9zaXRvcnktLT4+Y3JlYXRlT3JVcGRhdGU6IHVwZGF0ZWQgbW9kZWxcbiAqICAgICBlbHNlIE5vdEZvdW5kRXJyb3JcbiAqICAgICAgIGNyZWF0ZU9yVXBkYXRlLT4+UmVwb3NpdG9yeTogY3JlYXRlKG1vZGVsLCBjb250ZXh0KVxuICogICAgICAgUmVwb3NpdG9yeS0tPj5jcmVhdGVPclVwZGF0ZTogY3JlYXRlZCBtb2RlbFxuICogICAgIGVuZFxuICogICBlbmRcbiAqXG4gKiAgIGNyZWF0ZU9yVXBkYXRlLS0+PkNhbGxlcjogbW9kZWxcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNyZWF0ZU9yVXBkYXRlPFxuICBNIGV4dGVuZHMgTW9kZWwsXG4gIEYgZXh0ZW5kcyBSZXBvc2l0b3J5RmxhZ3MsXG4+KFxuICBtb2RlbDogTSxcbiAgY29udGV4dDogQ29udGV4dDxGPixcbiAgYWxpYXM/OiBzdHJpbmcsXG4gIHJlcG9zaXRvcnk/OiBSZXBvPE0sIEYsIENvbnRleHQ8Rj4+XG4pOiBQcm9taXNlPE0+IHtcbiAgaWYgKCFyZXBvc2l0b3J5KSB7XG4gICAgY29uc3QgY29uc3RydWN0b3IgPSBNb2RlbC5nZXQobW9kZWwuY29uc3RydWN0b3IubmFtZSk7XG4gICAgaWYgKCFjb25zdHJ1Y3RvcilcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGBDb3VsZCBub3QgZmluZCBtb2RlbCAke21vZGVsLmNvbnN0cnVjdG9yLm5hbWV9YCk7XG4gICAgcmVwb3NpdG9yeSA9IFJlcG9zaXRvcnkuZm9yTW9kZWw8TSwgUmVwbzxNPj4oXG4gICAgICBjb25zdHJ1Y3RvciBhcyB1bmtub3duIGFzIE1vZGVsQ29uc3RydWN0b3I8TT4sXG4gICAgICBhbGlhc1xuICAgICk7XG4gIH1cbiAgaWYgKHR5cGVvZiBtb2RlbFtyZXBvc2l0b3J5LnBrXSA9PT0gXCJ1bmRlZmluZWRcIilcbiAgICByZXR1cm4gcmVwb3NpdG9yeS5jcmVhdGUobW9kZWwsIGNvbnRleHQpO1xuICBlbHNlIHtcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIHJlcG9zaXRvcnkudXBkYXRlKG1vZGVsLCBjb250ZXh0KTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIGlmICghKGUgaW5zdGFuY2VvZiBOb3RGb3VuZEVycm9yKSkgdGhyb3cgZTtcbiAgICAgIHJldHVybiByZXBvc2l0b3J5LmNyZWF0ZShtb2RlbCwgY29udGV4dCk7XG4gICAgfVxuICB9XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEhhbmRsZXMgb25lLXRvLW9uZSByZWxhdGlvbnNoaXAgY3JlYXRpb25cbiAqIEBzdW1tYXJ5IFByb2Nlc3NlcyBhIG9uZS10by1vbmUgcmVsYXRpb25zaGlwIHdoZW4gY3JlYXRpbmcgYSBtb2RlbCwgZWl0aGVyIGJ5IHJlZmVyZW5jaW5nIGFuIGV4aXN0aW5nIG1vZGVsIG9yIGNyZWF0aW5nIGEgbmV3IG9uZVxuICogQHRlbXBsYXRlIE0gLSBUaGUgbW9kZWwgdHlwZSBleHRlbmRpbmcgTW9kZWxcbiAqIEB0ZW1wbGF0ZSBSIC0gVGhlIHJlcG9zaXRvcnkgdHlwZSBleHRlbmRpbmcgUmVwbzxNLCBGLCBDPlxuICogQHRlbXBsYXRlIFYgLSBUaGUgcmVsYXRpb25zIG1ldGFkYXRhIHR5cGUgZXh0ZW5kaW5nIFJlbGF0aW9uc01ldGFkYXRhXG4gKiBAdGVtcGxhdGUgRiAtIFRoZSByZXBvc2l0b3J5IGZsYWdzIHR5cGVcbiAqIEB0ZW1wbGF0ZSBDIC0gVGhlIGNvbnRleHQgdHlwZSBleHRlbmRpbmcgQ29udGV4dDxGPlxuICogQHBhcmFtIHtSfSB0aGlzIC0gVGhlIHJlcG9zaXRvcnkgaW5zdGFuY2VcbiAqIEBwYXJhbSB7Q29udGV4dDxGPn0gY29udGV4dCAtIFRoZSBjb250ZXh0IGZvciB0aGUgb3BlcmF0aW9uXG4gKiBAcGFyYW0ge1Z9IGRhdGEgLSBUaGUgcmVsYXRpb25zIG1ldGFkYXRhXG4gKiBAcGFyYW0ge3N0cmluZ30ga2V5IC0gVGhlIHByb3BlcnR5IGtleSBvZiB0aGUgcmVsYXRpb25zaGlwXG4gKiBAcGFyYW0ge019IG1vZGVsIC0gVGhlIG1vZGVsIGluc3RhbmNlXG4gKiBAcmV0dXJuIHtQcm9taXNlPHZvaWQ+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB3aGVuIHRoZSBvcGVyYXRpb24gaXMgY29tcGxldGVcbiAqIEBmdW5jdGlvbiBvbmVUb09uZU9uQ3JlYXRlXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmNvcmVcbiAqIEBtZXJtYWlkXG4gKiBzZXF1ZW5jZURpYWdyYW1cbiAqICAgcGFydGljaXBhbnQgQ2FsbGVyXG4gKiAgIHBhcnRpY2lwYW50IG9uZVRvT25lT25DcmVhdGVcbiAqICAgcGFydGljaXBhbnQgcmVwb3NpdG9yeUZyb21UeXBlTWV0YWRhdGFcbiAqICAgcGFydGljaXBhbnQgTW9kZWxcbiAqICAgcGFydGljaXBhbnQgUmVwb3NpdG9yeVxuICogICBwYXJ0aWNpcGFudCBjYWNoZU1vZGVsRm9yUG9wdWxhdGVcbiAqXG4gKiAgIENhbGxlci0+Pm9uZVRvT25lT25DcmVhdGU6IHRoaXMsIGNvbnRleHQsIGRhdGEsIGtleSwgbW9kZWxcbiAqICAgb25lVG9PbmVPbkNyZWF0ZS0+Pm9uZVRvT25lT25DcmVhdGU6IGNoZWNrIGlmIHByb3BlcnR5VmFsdWUgZXhpc3RzXG4gKlxuICogICBhbHQgcHJvcGVydHlWYWx1ZSBpcyBub3QgYW4gb2JqZWN0XG4gKiAgICAgb25lVG9PbmVPbkNyZWF0ZS0+PnJlcG9zaXRvcnlGcm9tVHlwZU1ldGFkYXRhOiBtb2RlbCwga2V5XG4gKiAgICAgcmVwb3NpdG9yeUZyb21UeXBlTWV0YWRhdGEtLT4+b25lVG9PbmVPbkNyZWF0ZTogaW5uZXJSZXBvXG4gKiAgICAgb25lVG9PbmVPbkNyZWF0ZS0+PmlubmVyUmVwbzogcmVhZChwcm9wZXJ0eVZhbHVlKVxuICogICAgIGlubmVyUmVwby0tPj5vbmVUb09uZU9uQ3JlYXRlOiByZWFkXG4gKiAgICAgb25lVG9PbmVPbkNyZWF0ZS0+PmNhY2hlTW9kZWxGb3JQb3B1bGF0ZTogY29udGV4dCwgbW9kZWwsIGtleSwgcHJvcGVydHlWYWx1ZSwgcmVhZFxuICogICAgIG9uZVRvT25lT25DcmVhdGUtPj5vbmVUb09uZU9uQ3JlYXRlOiBzZXQgbW9kZWxba2V5XSA9IHByb3BlcnR5VmFsdWVcbiAqICAgZWxzZSBwcm9wZXJ0eVZhbHVlIGlzIGFuIG9iamVjdFxuICogICAgIG9uZVRvT25lT25DcmVhdGUtPj5Nb2RlbDogZ2V0KGRhdGEuY2xhc3MpXG4gKiAgICAgTW9kZWwtLT4+b25lVG9PbmVPbkNyZWF0ZTogY29uc3RydWN0b3JcbiAqICAgICBvbmVUb09uZU9uQ3JlYXRlLT4+UmVwb3NpdG9yeTogZm9yTW9kZWwoY29uc3RydWN0b3IpXG4gKiAgICAgUmVwb3NpdG9yeS0tPj5vbmVUb09uZU9uQ3JlYXRlOiByZXBvXG4gKiAgICAgb25lVG9PbmVPbkNyZWF0ZS0+PnJlcG86IGNyZWF0ZShwcm9wZXJ0eVZhbHVlKVxuICogICAgIHJlcG8tLT4+b25lVG9PbmVPbkNyZWF0ZTogY3JlYXRlZFxuICogICAgIG9uZVRvT25lT25DcmVhdGUtPj5maW5kUHJpbWFyeUtleTogY3JlYXRlZFxuICogICAgIGZpbmRQcmltYXJ5S2V5LS0+Pm9uZVRvT25lT25DcmVhdGU6IHBrXG4gKiAgICAgb25lVG9PbmVPbkNyZWF0ZS0+PmNhY2hlTW9kZWxGb3JQb3B1bGF0ZTogY29udGV4dCwgbW9kZWwsIGtleSwgY3JlYXRlZFtwa10sIGNyZWF0ZWRcbiAqICAgICBvbmVUb09uZU9uQ3JlYXRlLT4+b25lVG9PbmVPbkNyZWF0ZTogc2V0IG1vZGVsW2tleV0gPSBjcmVhdGVkW3BrXVxuICogICBlbmRcbiAqXG4gKiAgIG9uZVRvT25lT25DcmVhdGUtLT4+Q2FsbGVyOiB2b2lkXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBvbmVUb09uZU9uQ3JlYXRlPFxuICBNIGV4dGVuZHMgTW9kZWwsXG4gIFIgZXh0ZW5kcyBSZXBvPE0sIEYsIEM+LFxuICBWIGV4dGVuZHMgUmVsYXRpb25zTWV0YWRhdGEsXG4gIEYgZXh0ZW5kcyBSZXBvc2l0b3J5RmxhZ3MsXG4gIEMgZXh0ZW5kcyBDb250ZXh0PEY+LFxuPihcbiAgdGhpczogUixcbiAgY29udGV4dDogQ29