@decaf-ts/core 0.5.2 → 0.5.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/core.cjs +1 -1
- package/dist/core.esm.cjs +1 -1
- package/lib/esm/identity/decorators.js +6 -6
- package/lib/esm/identity/index.js +3 -3
- package/lib/esm/identity/utils.js +3 -3
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +10 -10
- package/lib/esm/interfaces/index.js +8 -8
- package/lib/esm/model/construction.js +4 -4
- package/lib/esm/model/decorators.js +8 -8
- package/lib/esm/model/index.js +5 -5
- package/lib/esm/persistence/Adapter.d.ts +1 -1
- package/lib/esm/persistence/Adapter.js +6 -6
- package/lib/esm/persistence/Dispatch.js +3 -3
- package/lib/esm/persistence/Sequence.js +2 -2
- package/lib/esm/persistence/decorators.js +3 -3
- package/lib/esm/persistence/index.js +9 -9
- package/lib/esm/query/Condition.js +3 -3
- package/lib/esm/query/Paginator.js +2 -2
- package/lib/esm/query/Statement.js +4 -4
- package/lib/esm/query/index.js +8 -8
- package/lib/esm/ram/RamAdapter.js +8 -8
- package/lib/esm/ram/RamPaginator.js +2 -2
- package/lib/esm/ram/RamSequence.js +4 -4
- package/lib/esm/ram/RamStatement.js +4 -4
- package/lib/esm/ram/handlers.js +2 -2
- package/lib/esm/ram/index.js +11 -11
- package/lib/esm/ram/model/RamSequence.js +3 -3
- package/lib/esm/ram/model/index.js +2 -2
- package/lib/esm/repository/Repository.js +9 -9
- package/lib/esm/repository/decorators.js +2 -2
- package/lib/esm/repository/index.js +8 -8
- package/lib/esm/repository/injectables.js +5 -5
- package/lib/esm/repository/utils.js +4 -4
- package/lib/esm/utils/index.js +3 -3
- package/lib/index.cjs +1 -1
- package/lib/index.d.ts +1 -1
- package/lib/persistence/Adapter.d.ts +1 -1
- package/package.json +1 -1
@@ -1,10 +1,10 @@
|
|
1
1
|
import { Decoration, propMetadata, required, } from "@decaf-ts/decorator-validation";
|
2
|
-
import { DefaultSequenceOptions, } from "
|
2
|
+
import { DefaultSequenceOptions, } from "./../interfaces/SequenceOptions.js";
|
3
3
|
import { DBKeys, InternalError, onCreate, readonly, } from "@decaf-ts/db-decorators";
|
4
|
-
import { Repository } from "
|
5
|
-
import { index } from "
|
6
|
-
import { sequenceNameForModel } from "./utils";
|
7
|
-
import { OrderDirection } from "
|
4
|
+
import { Repository } from "./../repository/Repository.js";
|
5
|
+
import { index } from "./../model/decorators.js";
|
6
|
+
import { sequenceNameForModel } from "./utils.js";
|
7
|
+
import { OrderDirection } from "./../repository/index.js";
|
8
8
|
/**
|
9
9
|
* @description Callback function for primary key creation
|
10
10
|
* @summary Handles the creation of primary key values for models using sequences
|
@@ -94,4 +94,4 @@ export function pk(opts = DefaultSequenceOptions) {
|
|
94
94
|
propMetadata(key, opts), onCreate(pkOnCreate, opts))
|
95
95
|
.apply();
|
96
96
|
}
|
97
|
-
//# sourceMappingURL=data:application/json;base64,
|
97
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../../src/identity/decorators.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,UAAU,EAEV,YAAY,EACZ,QAAQ,GACT,MAAM,gCAAgC,CAAC;AACxC,OAAO,EACL,sBAAsB,GAEvB,2CAAsC;AACvC,OAAO,EACL,MAAM,EACN,aAAa,EACb,QAAQ,EACR,QAAQ,GAET,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAQ,UAAU,EAAE,sCAAiC;AAC5D,OAAO,EAAE,KAAK,EAAE,iCAA4B;AAC5C,OAAO,EAAE,oBAAoB,EAAE,mBAAgB;AAG/C,OAAO,EAAE,cAAc,EAAE,iCAAsB;AAE/C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmCG;AACH,MAAM,CAAC,KAAK,UAAU,UAAU,CAQ9B,OAAmB,EACnB,IAAO,EACP,GAAY,EACZ,KAAQ;IAER,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC;QAC7B,OAAO;IACT,CAAC;IAED,MAAM,kBAAkB,GAAG,UACzB,MAAS,EACT,WAAmB,EACnB,KAA+B;QAE/B,MAAM,CAAC,cAAc,CAAC,MAAM,EAAE,WAAW,EAAE;YACzC,UAAU,EAAE,IAAI;YAChB,QAAQ,EAAE,KAAK;YACf,YAAY,EAAE,IAAI;YAClB,KAAK,EAAE,KAAK;SACb,CAAC,CAAC;IACL,CAAC,CAAC;IACF,IAAI,CAAC,IAAI,CAAC,IAAI;QAAE,IAAI,CAAC,IAAI,GAAG,oBAAoB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;IAC9D,IAAI,QAAkB,CAAC;IACvB,IAAI,CAAC;QACH,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;IAC/C,CAAC;IAAC,OAAO,CAAM,EAAE,CAAC;QAChB,MAAM,IAAI,aAAa,CACrB,kCAAkC,IAAI,CAAC,IAAI,KAAK,CAAC,EAAE,CACpD,CAAC;IACJ,CAAC;IAED,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;IACnC,kBAAkB,CAAC,KAAK,EAAE,GAAa,EAAE,IAAI,CAAC,CAAC;AACjD,CAAC;AAED;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,UAAU,EAAE,CAChB,OAGI,sBAAsB;IAE1B,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,sBAAsB,EAAE,IAAI,CAAoB,CAAC;IAC1E,MAAM,GAAG,GAAG,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IACtC,OAAO,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC;SACvB,MAAM,CACL,KAAK,CAAC,CAAC,cAAc,CAAC,GAAG,EAAE,cAAc,CAAC,GAAG,CAAC,CAAC,EAC/C,QAAQ,EAAE,EACV,QAAQ,EAAE;IACV,iDAAiD;IACjD,YAAY,CAAC,GAAG,EAAE,IAAuB,CAAC,EAC1C,QAAQ,CAAC,UAAU,EAAE,IAAuB,CAAC,CAC9C;SACA,KAAK,EAAE,CAAC;AACb,CAAC","sourcesContent":["import {\n  Decoration,\n  Model,\n  propMetadata,\n  required,\n} from \"@decaf-ts/decorator-validation\";\nimport {\n  DefaultSequenceOptions,\n  SequenceOptions,\n} from \"../interfaces/SequenceOptions\";\nimport {\n  DBKeys,\n  InternalError,\n  onCreate,\n  readonly,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport { Repo, Repository } from \"../repository/Repository\";\nimport { index } from \"../model/decorators\";\nimport { sequenceNameForModel } from \"./utils\";\nimport { Sequence } from \"../persistence/Sequence\";\nimport { Context } from \"@decaf-ts/db-decorators\";\nimport { OrderDirection } from \"../repository\";\n\n/**\n * @description Callback function for primary key creation\n * @summary Handles the creation of primary key values for models using sequences\n * @template M - Type that extends Model\n * @template R - Type that extends Repo<M, F, C>\n * @template V - Type that extends SequenceOptions\n * @template F - Type that extends RepositoryFlags\n * @template C - Type that extends Context<F>\n * @param {Context<F>} context - The execution context\n * @param {V} data - The sequence options\n * @param key - The property key to set as primary key\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the primary key is set\n * @function pkOnCreate\n * @category Property Decorators\n * @mermaid\n * sequenceDiagram\n *   participant Model\n *   participant pkOnCreate\n *   participant Adapter\n *   participant Sequence\n *\n *   Model->>pkOnCreate: Call with model instance\n *   Note over pkOnCreate: Check if key already exists\n *   alt Key exists or no type specified\n *     pkOnCreate-->>Model: Return early\n *   else Key needs to be created\n *     pkOnCreate->>pkOnCreate: Generate sequence name if not provided\n *     pkOnCreate->>Adapter: Request Sequence(data)\n *     Adapter->>Sequence: Create sequence\n *     Sequence-->>pkOnCreate: Return sequence\n *     pkOnCreate->>Sequence: Call next()\n *     Sequence-->>pkOnCreate: Return next value\n *     pkOnCreate->>Model: Set primary key value\n *   end\n */\nexport async function pkOnCreate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends SequenceOptions,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!data.type || model[key]) {\n    return;\n  }\n\n  const setPrimaryKeyValue = function <M extends Model>(\n    target: M,\n    propertyKey: string,\n    value: string | number | bigint\n  ) {\n    Object.defineProperty(target, propertyKey, {\n      enumerable: true,\n      writable: false,\n      configurable: true,\n      value: value,\n    });\n  };\n  if (!data.name) data.name = sequenceNameForModel(model, \"pk\");\n  let sequence: Sequence;\n  try {\n    sequence = await this.adapter.Sequence(data);\n  } catch (e: any) {\n    throw new InternalError(\n      `Failed to instantiate Sequence ${data.name}: ${e}`\n    );\n  }\n\n  const next = await sequence.next();\n  setPrimaryKeyValue(model, key as string, next);\n}\n\n/**\n * @description Primary Key Decorator\n * @summary Marks a property as the model's primary key with automatic sequence generation\n * This decorator combines multiple behaviors: it marks the property as unique, required,\n * and ensures the index is created properly according to the provided sequence options.\n * @param {Omit<SequenceOptions, \"cycle\" | \"startWith\" | \"incrementBy\">} opts - Options for the sequence generation\n * @return {PropertyDecorator} A property decorator that can be applied to model properties\n * @function pk\n * @category Property Decorators\n * @example\n * ```typescript\n * class User extends BaseModel {\n *   @pk()\n *   id!: string;\n *\n *   @required()\n *   username!: string;\n * }\n * ```\n */\nexport function pk(\n  opts: Omit<\n    SequenceOptions,\n    \"cycle\" | \"startWith\" | \"incrementBy\"\n  > = DefaultSequenceOptions\n) {\n  opts = Object.assign({}, DefaultSequenceOptions, opts) as SequenceOptions;\n  const key = Repository.key(DBKeys.ID);\n  return Decoration.for(key)\n    .define(\n      index([OrderDirection.ASC, OrderDirection.DSC]),\n      required(),\n      readonly(),\n      // type([String.name, Number.name, BigInt.name]),\n      propMetadata(key, opts as SequenceOptions),\n      onCreate(pkOnCreate, opts as SequenceOptions)\n    )\n    .apply();\n}\n"]}
|
@@ -1,3 +1,3 @@
|
|
1
|
-
export * from "./decorators";
|
2
|
-
export * from "./utils";
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
1
|
+
export * from "./decorators.js";
|
2
|
+
export * from "./utils.js";
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaWRlbnRpdHkvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsZ0NBQTZCO0FBQzdCLDJCQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG4iXX0=
|
@@ -1,6 +1,6 @@
|
|
1
1
|
import { Model } from "@decaf-ts/decorator-validation";
|
2
|
-
import { Adapter } from "
|
3
|
-
import { PersistenceKeys } from "
|
2
|
+
import { Adapter } from "./../persistence/Adapter.js";
|
3
|
+
import { PersistenceKeys } from "./../persistence/constants.js";
|
4
4
|
/**
|
5
5
|
* @description Gets the table name for a model
|
6
6
|
* @summary Retrieves the table name associated with a model by checking metadata or falling back to the constructor name
|
@@ -34,4 +34,4 @@ export function getTableName(model) {
|
|
34
34
|
export function sequenceNameForModel(model, ...args) {
|
35
35
|
return [getTableName(model), ...args].join("_");
|
36
36
|
}
|
37
|
-
//# sourceMappingURL=data:application/json;base64,
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaWRlbnRpdHkvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFlLEtBQUssRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBQ3BFLE9BQU8sRUFBRSxPQUFPLEVBQUUsb0NBQStCO0FBQ2pELE9BQU8sRUFBRSxlQUFlLEVBQUUsc0NBQWlDO0FBRTNEOzs7Ozs7OztHQVFHO0FBQ0gsTUFBTSxVQUFVLFlBQVksQ0FDMUIsS0FBeUI7SUFFekIsTUFBTSxHQUFHLEdBQUcsS0FBSyxZQUFZLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDO0lBRS9ELE1BQU0sUUFBUSxHQUFHLE9BQU8sQ0FBQyxjQUFjLENBQ3JDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLEtBQUssQ0FBQyxFQUNsQyxHQUFHLENBQ0osQ0FBQztJQUNGLElBQUksUUFBUSxFQUFFLENBQUM7UUFDYixPQUFPLFFBQVEsQ0FBQztJQUNsQixDQUFDO0lBQ0QsSUFBSSxLQUFLLFlBQVksS0FBSyxFQUFFLENBQUM7UUFDM0IsT0FBTyxLQUFLLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQztJQUNoQyxDQUFDO0lBQ0QsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDO0FBQ3BCLENBQUM7QUFFRDs7Ozs7Ozs7O0dBU0c7QUFDSCxNQUFNLFVBQVUsb0JBQW9CLENBQ2xDLEtBQXlCLEVBQ3pCLEdBQUcsSUFBYztJQUVqQixPQUFPLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxFQUFFLEdBQUcsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBQ2xELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb25zdHJ1Y3RvciwgTW9kZWwgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBBZGFwdGVyIH0gZnJvbSBcIi4uL3BlcnNpc3RlbmNlL0FkYXB0ZXJcIjtcbmltcG9ydCB7IFBlcnNpc3RlbmNlS2V5cyB9IGZyb20gXCIuLi9wZXJzaXN0ZW5jZS9jb25zdGFudHNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gR2V0cyB0aGUgdGFibGUgbmFtZSBmb3IgYSBtb2RlbFxuICogQHN1bW1hcnkgUmV0cmlldmVzIHRoZSB0YWJsZSBuYW1lIGFzc29jaWF0ZWQgd2l0aCBhIG1vZGVsIGJ5IGNoZWNraW5nIG1ldGFkYXRhIG9yIGZhbGxpbmcgYmFjayB0byB0aGUgY29uc3RydWN0b3IgbmFtZVxuICogQHRlbXBsYXRlIE0gLSBUeXBlIHRoYXQgZXh0ZW5kcyBNb2RlbFxuICogQHBhcmFtIHtNIHwgQ29uc3RydWN0b3I8TT59IG1vZGVsIC0gVGhlIG1vZGVsIGluc3RhbmNlIG9yIGNvbnN0cnVjdG9yIHRvIGdldCB0aGUgdGFibGUgbmFtZSBmb3JcbiAqIEByZXR1cm4ge3N0cmluZ30gVGhlIHRhYmxlIG5hbWUgZm9yIHRoZSBtb2RlbFxuICogQGZ1bmN0aW9uIGdldFRhYmxlTmFtZVxuICogQG1lbWJlck9mIG1vZHVsZTpjb3JlXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRUYWJsZU5hbWU8TSBleHRlbmRzIE1vZGVsPihcbiAgbW9kZWw6IE0gfCBDb25zdHJ1Y3RvcjxNPlxuKTogc3RyaW5nIHtcbiAgY29uc3Qgb2JqID0gbW9kZWwgaW5zdGFuY2VvZiBNb2RlbCA/IG1vZGVsLmNvbnN0cnVjdG9yIDogbW9kZWw7XG5cbiAgY29uc3QgbWV0YWRhdGEgPSBSZWZsZWN0LmdldE93bk1ldGFkYXRhKFxuICAgIEFkYXB0ZXIua2V5KFBlcnNpc3RlbmNlS2V5cy5UQUJMRSksXG4gICAgb2JqXG4gICk7XG4gIGlmIChtZXRhZGF0YSkge1xuICAgIHJldHVybiBtZXRhZGF0YTtcbiAgfVxuICBpZiAobW9kZWwgaW5zdGFuY2VvZiBNb2RlbCkge1xuICAgIHJldHVybiBtb2RlbC5jb25zdHJ1Y3Rvci5uYW1lO1xuICB9XG4gIHJldHVybiBtb2RlbC5uYW1lO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBHZW5lcmF0ZXMgYSBzZXF1ZW5jZSBuYW1lIGZvciBhIG1vZGVsXG4gKiBAc3VtbWFyeSBDcmVhdGVzIGEgc3RhbmRhcmRpemVkIHNlcXVlbmNlIG5hbWUgYnkgY29tYmluaW5nIHRoZSB0YWJsZSBuYW1lIHdpdGggYWRkaXRpb25hbCBhcmd1bWVudHNcbiAqIEB0ZW1wbGF0ZSBNIC0gVHlwZSB0aGF0IGV4dGVuZHMgTW9kZWxcbiAqIEBwYXJhbSB7TSB8IENvbnN0cnVjdG9yPE0+fSBtb2RlbCAtIFRoZSBtb2RlbCBpbnN0YW5jZSBvciBjb25zdHJ1Y3RvciB0byBnZW5lcmF0ZSB0aGUgc2VxdWVuY2UgbmFtZSBmb3JcbiAqIEBwYXJhbSB7Li4uc3RyaW5nfSBhcmdzIC0gQWRkaXRpb25hbCBzdHJpbmcgYXJndW1lbnRzIHRvIGFwcGVuZCB0byB0aGUgc2VxdWVuY2UgbmFtZVxuICogQHJldHVybiB7c3RyaW5nfSBUaGUgZ2VuZXJhdGVkIHNlcXVlbmNlIG5hbWVcbiAqIEBmdW5jdGlvbiBzZXF1ZW5jZU5hbWVGb3JNb2RlbFxuICogQG1lbWJlck9mIG1vZHVsZTpjb3JlXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzZXF1ZW5jZU5hbWVGb3JNb2RlbDxNIGV4dGVuZHMgTW9kZWw+KFxuICBtb2RlbDogTSB8IENvbnN0cnVjdG9yPE0+LFxuICAuLi5hcmdzOiBzdHJpbmdbXVxuKSB7XG4gIHJldHVybiBbZ2V0VGFibGVOYW1lKG1vZGVsKSwgLi4uYXJnc10uam9pbihcIl9cIik7XG59XG4iXX0=
|
package/lib/esm/index.d.ts
CHANGED
package/lib/esm/index.js
CHANGED
@@ -5,23 +5,23 @@
|
|
5
5
|
* model definitions, repository patterns, persistence layer, query building, and utility functions.
|
6
6
|
* It exports functionality from various submodules and sets up the injectable registry for repository decorators.
|
7
7
|
*/
|
8
|
-
import { InjectablesRegistry } from "./repository";
|
8
|
+
import { InjectablesRegistry } from "./repository/index.js";
|
9
9
|
import { Injectables } from "@decaf-ts/injectable-decorators";
|
10
10
|
// overrides the previous Injectables registry to enable the @repository decorator
|
11
11
|
Injectables.setRegistry(new InjectablesRegistry());
|
12
|
-
export * from "./identity";
|
13
|
-
export * from "./interfaces";
|
14
|
-
export * from "./model";
|
15
|
-
export * from "./query";
|
16
|
-
export * from "./repository";
|
17
|
-
export * from "./utils";
|
12
|
+
export * from "./identity/index.js";
|
13
|
+
export * from "./interfaces/index.js";
|
14
|
+
export * from "./model/index.js";
|
15
|
+
export * from "./query/index.js";
|
16
|
+
export * from "./repository/index.js";
|
17
|
+
export * from "./utils/index.js";
|
18
18
|
//left to last on purpose
|
19
|
-
export * from "./persistence";
|
19
|
+
export * from "./persistence/index.js";
|
20
20
|
/**
|
21
21
|
* @description Stores the current package version
|
22
22
|
* @summary A constant representing the version of the core package
|
23
23
|
* @const VERSION
|
24
24
|
* @memberOf module:core
|
25
25
|
*/
|
26
|
-
export const VERSION = "0.5.
|
27
|
-
//# sourceMappingURL=data:application/json;base64,
|
26
|
+
export const VERSION = "0.5.3";
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUgsT0FBTyxFQUFFLG1CQUFtQixFQUFFLDhCQUFxQjtBQUNuRCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFFOUQsa0ZBQWtGO0FBQ2xGLFdBQVcsQ0FBQyxXQUFXLENBQUMsSUFBSSxtQkFBbUIsRUFBRSxDQUFDLENBQUM7QUFFbkQsb0NBQTJCO0FBQzNCLHNDQUE2QjtBQUM3QixpQ0FBd0I7QUFDeEIsaUNBQXdCO0FBQ3hCLHNDQUE2QjtBQUM3QixpQ0FBd0I7QUFDeEIseUJBQXlCO0FBQ3pCLHVDQUE4QjtBQUU5Qjs7Ozs7R0FLRztBQUNILE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBtb2R1bGUgY29yZVxuICogQGRlc2NyaXB0aW9uIENvcmUgbW9kdWxlIGZvciB0aGUgRGVjYWYgVHlwZVNjcmlwdCBmcmFtZXdvcmtcbiAqIEBzdW1tYXJ5IFRoaXMgbW9kdWxlIHByb3ZpZGVzIHRoZSBmb3VuZGF0aW9uYWwgY29tcG9uZW50cyBvZiB0aGUgRGVjYWYgZnJhbWV3b3JrLCBpbmNsdWRpbmcgaWRlbnRpdHkgbWFuYWdlbWVudCwgXG4gKiBtb2RlbCBkZWZpbml0aW9ucywgcmVwb3NpdG9yeSBwYXR0ZXJucywgcGVyc2lzdGVuY2UgbGF5ZXIsIHF1ZXJ5IGJ1aWxkaW5nLCBhbmQgdXRpbGl0eSBmdW5jdGlvbnMuXG4gKiBJdCBleHBvcnRzIGZ1bmN0aW9uYWxpdHkgZnJvbSB2YXJpb3VzIHN1Ym1vZHVsZXMgYW5kIHNldHMgdXAgdGhlIGluamVjdGFibGUgcmVnaXN0cnkgZm9yIHJlcG9zaXRvcnkgZGVjb3JhdG9ycy5cbiAqL1xuXG5pbXBvcnQgeyBJbmplY3RhYmxlc1JlZ2lzdHJ5IH0gZnJvbSBcIi4vcmVwb3NpdG9yeVwiO1xuaW1wb3J0IHsgSW5qZWN0YWJsZXMgfSBmcm9tIFwiQGRlY2FmLXRzL2luamVjdGFibGUtZGVjb3JhdG9yc1wiO1xuXG4vLyBvdmVycmlkZXMgdGhlIHByZXZpb3VzIEluamVjdGFibGVzIHJlZ2lzdHJ5IHRvIGVuYWJsZSB0aGUgQHJlcG9zaXRvcnkgZGVjb3JhdG9yXG5JbmplY3RhYmxlcy5zZXRSZWdpc3RyeShuZXcgSW5qZWN0YWJsZXNSZWdpc3RyeSgpKTtcblxuZXhwb3J0ICogZnJvbSBcIi4vaWRlbnRpdHlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2ludGVyZmFjZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL21vZGVsXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9xdWVyeVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vcmVwb3NpdG9yeVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHNcIjtcbi8vbGVmdCB0byBsYXN0IG9uIHB1cnBvc2VcbmV4cG9ydCAqIGZyb20gXCIuL3BlcnNpc3RlbmNlXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFN0b3JlcyB0aGUgY3VycmVudCBwYWNrYWdlIHZlcnNpb25cbiAqIEBzdW1tYXJ5IEEgY29uc3RhbnQgcmVwcmVzZW50aW5nIHRoZSB2ZXJzaW9uIG9mIHRoZSBjb3JlIHBhY2thZ2VcbiAqIEBjb25zdCBWRVJTSU9OXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmNvcmVcbiAqL1xuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBcIiMjVkVSU0lPTiMjXCI7XG4iXX0=
|
@@ -1,8 +1,8 @@
|
|
1
|
-
export * from "./ErrorParser";
|
2
|
-
export * from "./Executor";
|
3
|
-
export * from "./Observable";
|
4
|
-
export * from "./Observer";
|
5
|
-
export * from "./Queriable";
|
6
|
-
export * from "./RawExecutor";
|
7
|
-
export * from "./SequenceOptions";
|
8
|
-
//# sourceMappingURL=data:application/json;base64,
|
1
|
+
export * from "./ErrorParser.js";
|
2
|
+
export * from "./Executor.js";
|
3
|
+
export * from "./Observable.js";
|
4
|
+
export * from "./Observer.js";
|
5
|
+
export * from "./Queriable.js";
|
6
|
+
export * from "./RawExecutor.js";
|
7
|
+
export * from "./SequenceOptions.js";
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaW50ZXJmYWNlcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxpQ0FBOEI7QUFDOUIsOEJBQTJCO0FBQzNCLGdDQUE2QjtBQUM3Qiw4QkFBMkI7QUFDM0IsK0JBQTRCO0FBQzVCLGlDQUE4QjtBQUM5QixxQ0FBa0MiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9FcnJvclBhcnNlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vRXhlY3V0b3JcIjtcbmV4cG9ydCAqIGZyb20gXCIuL09ic2VydmFibGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL09ic2VydmVyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9RdWVyaWFibGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL1Jhd0V4ZWN1dG9yXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9TZXF1ZW5jZU9wdGlvbnNcIjtcbiJdfQ==
|
@@ -1,8 +1,8 @@
|
|
1
1
|
import { Model, Validation, ValidationKeys, } from "@decaf-ts/decorator-validation";
|
2
|
-
import { Repository } from "
|
2
|
+
import { Repository } from "./../repository/Repository.js";
|
3
3
|
import { findPrimaryKey, InternalError, NotFoundError, } from "@decaf-ts/db-decorators";
|
4
|
-
import { PersistenceKeys } from "
|
5
|
-
import { Cascade } from "
|
4
|
+
import { PersistenceKeys } from "./../persistence/constants.js";
|
5
|
+
import { Cascade } from "./../repository/constants.js";
|
6
6
|
/**
|
7
7
|
* @description Creates or updates a model instance
|
8
8
|
* @summary Determines whether to create a new model or update an existing one based on the presence of a primary key
|
@@ -645,4 +645,4 @@ export function repositoryFromTypeMetadata(model, propertyKey) {
|
|
645
645
|
throw new InternalError(`No registered model found for ${constructorName}`);
|
646
646
|
return Repository.forModel(constructor);
|
647
647
|
}
|
648
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construction.js","sourceRoot":"","sources":["../../../src/model/construction.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Constructor,\n  Model,\n  ModelConstructor,\n  Validation,\n  ValidationKeys,\n} from \"@decaf-ts/decorator-validation\";\nimport { Repo, Repository } from \"../repository/Repository\";\nimport { RelationsMetadata } from \"./types\";\nimport {\n  findPrimaryKey,\n  InternalError,\n  NotFoundError,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport { PersistenceKeys } from \"../persistence/constants\";\nimport { Cascade } from \"../repository/constants\";\nimport { Context } from \"@decaf-ts/db-decorators\";\n\n/**\n * @description Creates or updates a model instance\n * @summary Determines whether to create a new model or update an existing one based on the presence of a primary key\n * @template M - The model type extending Model\n * @template F - The repository flags type\n * @param {M} model - The model instance to create or update\n * @param {Context<F>} context - The context for the operation\n * @param {Repo<M, F, Context<F>>} [repository] - Optional repository to use for the operation\n * @return {Promise<M>} A promise that resolves to the created or updated model\n * @function createOrUpdate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant createOrUpdate\n *   participant Repository\n *   participant Model\n *\n *   Caller->>createOrUpdate: model, context, repository?\n *   alt repository not provided\n *     createOrUpdate->>Model: get(model.constructor.name)\n *     Model-->>createOrUpdate: constructor\n *     createOrUpdate->>Repository: forModel(constructor)\n *     Repository-->>createOrUpdate: repository\n *   end\n *\n *   alt primary key undefined\n *     createOrUpdate->>Repository: create(model, context)\n *     Repository-->>createOrUpdate: created model\n *   else primary key defined\n *     createOrUpdate->>Repository: update(model, context)\n *     alt update successful\n *       Repository-->>createOrUpdate: updated model\n *     else NotFoundError\n *       createOrUpdate->>Repository: create(model, context)\n *       Repository-->>createOrUpdate: created model\n *     end\n *   end\n *\n *   createOrUpdate-->>Caller: model\n */\nexport async function createOrUpdate<\n  M extends Model,\n  F extends RepositoryFlags,\n>(\n  model: M,\n  context: Context<F>,\n  repository?: Repo<M, F, Context<F>>\n): Promise<M> {\n  if (!repository) {\n    const constructor = Model.get(model.constructor.name);\n    if (!constructor)\n      throw new InternalError(`Could not find model ${model.constructor.name}`);\n    repository = Repository.forModel<M, Repo<M>>(\n      constructor as unknown as ModelConstructor<M>\n    );\n  }\n  if (typeof model[repository.pk] === \"undefined\")\n    return repository.create(model, context);\n  else {\n    try {\n      return repository.update(model, context);\n    } catch (e: any) {\n      if (!(e instanceof NotFoundError)) throw e;\n      return repository.create(model, context);\n    }\n  }\n}\n\n/**\n * @description Handles one-to-one relationship creation\n * @summary Processes a one-to-one relationship when creating a model, either by referencing an existing model or creating a new one\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param {string} key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToOneOnCreate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToOneOnCreate\n *   participant repositoryFromTypeMetadata\n *   participant Model\n *   participant Repository\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToOneOnCreate: this, context, data, key, model\n *   oneToOneOnCreate->>oneToOneOnCreate: check if propertyValue exists\n *\n *   alt propertyValue is not an object\n *     oneToOneOnCreate->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToOneOnCreate: innerRepo\n *     oneToOneOnCreate->>innerRepo: read(propertyValue)\n *     innerRepo-->>oneToOneOnCreate: read\n *     oneToOneOnCreate->>cacheModelForPopulate: context, model, key, propertyValue, read\n *     oneToOneOnCreate->>oneToOneOnCreate: set model[key] = propertyValue\n *   else propertyValue is an object\n *     oneToOneOnCreate->>Model: get(data.class)\n *     Model-->>oneToOneOnCreate: constructor\n *     oneToOneOnCreate->>Repository: forModel(constructor)\n *     Repository-->>oneToOneOnCreate: repo\n *     oneToOneOnCreate->>repo: create(propertyValue)\n *     repo-->>oneToOneOnCreate: created\n *     oneToOneOnCreate->>findPrimaryKey: created\n *     findPrimaryKey-->>oneToOneOnCreate: pk\n *     oneToOneOnCreate->>cacheModelForPopulate: context, model, key, created[pk], created\n *     oneToOneOnCreate->>oneToOneOnCreate: set model[key] = created[pk]\n *   end\n *\n *   oneToOneOnCreate-->>Caller: void\n */\nexport async function oneToOneOnCreate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValue: any = model[key];\n  if (!propertyValue) return;\n\n  if (typeof propertyValue !== \"object\") {\n    const innerRepo = repositoryFromTypeMetadata(model, key);\n    const read = await innerRepo.read(propertyValue);\n    await cacheModelForPopulate(context, model, key, propertyValue, read);\n    (model as any)[key] = propertyValue;\n    return;\n  }\n\n  const constructor = Model.get(data.class);\n  if (!constructor)\n    throw new InternalError(`Could not find model ${data.class}`);\n  const repo: Repo<any> = Repository.forModel(constructor);\n  const created = await repo.create(propertyValue);\n  const pk = findPrimaryKey(created).id;\n  await cacheModelForPopulate(context, model, key, created[pk], created);\n  (model as any)[key] = created[pk];\n}\n\n/**\n * @description Handles one-to-one relationship updates\n * @summary Processes a one-to-one relationship when updating a model, either by referencing an existing model or updating the related model\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToOneOnUpdate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToOneOnUpdate\n *   participant repositoryFromTypeMetadata\n *   participant createOrUpdate\n *   participant findPrimaryKey\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToOneOnUpdate: this, context, data, key, model\n *   oneToOneOnUpdate->>oneToOneOnUpdate: check if propertyValue exists\n *   oneToOneOnUpdate->>oneToOneOnUpdate: check if cascade.update is CASCADE\n *\n *   alt propertyValue is not an object\n *     oneToOneOnUpdate->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToOneOnUpdate: innerRepo\n *     oneToOneOnUpdate->>innerRepo: read(propertyValue)\n *     innerRepo-->>oneToOneOnUpdate: read\n *     oneToOneOnUpdate->>cacheModelForPopulate: context, model, key, propertyValue, read\n *     oneToOneOnUpdate->>oneToOneOnUpdate: set model[key] = propertyValue\n *   else propertyValue is an object\n *     oneToOneOnUpdate->>createOrUpdate: model[key], context\n *     createOrUpdate-->>oneToOneOnUpdate: updated\n *     oneToOneOnUpdate->>findPrimaryKey: updated\n *     findPrimaryKey-->>oneToOneOnUpdate: pk\n *     oneToOneOnUpdate->>cacheModelForPopulate: context, model, key, updated[pk], updated\n *     oneToOneOnUpdate->>oneToOneOnUpdate: set model[key] = updated[pk]\n *   end\n *\n *   oneToOneOnUpdate-->>Caller: void\n */\nexport async function oneToOneOnUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValue: any = model[key];\n  if (!propertyValue) return;\n  if (data.cascade.update !== Cascade.CASCADE) return;\n\n  if (typeof propertyValue !== \"object\") {\n    const innerRepo = repositoryFromTypeMetadata(model, key);\n    const read = await innerRepo.read(propertyValue);\n    await cacheModelForPopulate(context, model, key, propertyValue, read);\n    (model as any)[key] = propertyValue;\n    return;\n  }\n\n  const updated = await createOrUpdate(model[key] as M, context);\n  const pk = findPrimaryKey(updated).id;\n  await cacheModelForPopulate(\n    context,\n    model,\n    key,\n    updated[pk] as string,\n    updated\n  );\n  model[key] = updated[pk];\n}\n\n/**\n * @description Handles one-to-one relationship deletion\n * @summary Processes a one-to-one relationship when deleting a model, deleting the related model if cascade is enabled\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToOneOnDelete\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToOneOnDelete\n *   participant repositoryFromTypeMetadata\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToOneOnDelete: this, context, data, key, model\n *   oneToOneOnDelete->>oneToOneOnDelete: check if propertyValue exists\n *   oneToOneOnDelete->>oneToOneOnDelete: check if cascade.update is CASCADE\n *\n *   oneToOneOnDelete->>repositoryFromTypeMetadata: model, key\n *   repositoryFromTypeMetadata-->>oneToOneOnDelete: innerRepo\n *\n *   alt propertyValue is not a Model instance\n *     oneToOneOnDelete->>innerRepo: delete(model[key], context)\n *     innerRepo-->>oneToOneOnDelete: deleted\n *   else propertyValue is a Model instance\n *     oneToOneOnDelete->>innerRepo: delete(model[key][innerRepo.pk], context)\n *     innerRepo-->>oneToOneOnDelete: deleted\n *   end\n *\n *   oneToOneOnDelete->>cacheModelForPopulate: context, model, key, deleted[innerRepo.pk], deleted\n *   oneToOneOnDelete-->>Caller: void\n */\nexport async function oneToOneOnDelete<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValue: any = model[key];\n  if (!propertyValue) return;\n  if (data.cascade.update !== Cascade.CASCADE) return;\n  const innerRepo: Repo<M> = repositoryFromTypeMetadata(model, key);\n  let deleted: M;\n  if (!(propertyValue instanceof Model))\n    deleted = await innerRepo.delete(model[key] as string, context);\n  else\n    deleted = await innerRepo.delete(\n      (model[key] as M)[innerRepo.pk as keyof M] as string,\n      context\n    );\n  await cacheModelForPopulate(\n    context,\n    model,\n    key,\n    deleted[innerRepo.pk] as string,\n    deleted\n  );\n}\n\n/**\n * @description Handles one-to-many relationship creation\n * @summary Processes a one-to-many relationship when creating a model, either by referencing existing models or creating new ones\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToManyOnCreate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToManyOnCreate\n *   participant repositoryFromTypeMetadata\n *   participant createOrUpdate\n *   participant findPrimaryKey\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToManyOnCreate: this, context, data, key, model\n *   oneToManyOnCreate->>oneToManyOnCreate: check if propertyValues exists and has length\n *   oneToManyOnCreate->>oneToManyOnCreate: check if all elements have same type\n *   oneToManyOnCreate->>oneToManyOnCreate: create uniqueValues set\n *\n *   alt arrayType is not \"object\"\n *     oneToManyOnCreate->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToManyOnCreate: repo\n *     loop for each id in uniqueValues\n *       oneToManyOnCreate->>repo: read(id)\n *       repo-->>oneToManyOnCreate: read\n *       oneToManyOnCreate->>cacheModelForPopulate: context, model, key, id, read\n *     end\n *     oneToManyOnCreate->>oneToManyOnCreate: set model[key] = [...uniqueValues]\n *   else arrayType is \"object\"\n *     oneToManyOnCreate->>findPrimaryKey: propertyValues[0]\n *     findPrimaryKey-->>oneToManyOnCreate: pkName\n *     oneToManyOnCreate->>oneToManyOnCreate: create result set\n *     loop for each m in propertyValues\n *       oneToManyOnCreate->>createOrUpdate: m, context\n *       createOrUpdate-->>oneToManyOnCreate: record\n *       oneToManyOnCreate->>cacheModelForPopulate: context, model, key, record[pkName], record\n *       oneToManyOnCreate->>oneToManyOnCreate: add record[pkName] to result\n *     end\n *     oneToManyOnCreate->>oneToManyOnCreate: set model[key] = [...result]\n *   end\n *\n *   oneToManyOnCreate-->>Caller: void\n */\nexport async function oneToManyOnCreate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValues: any = model[key];\n  if (!propertyValues || !propertyValues.length) return;\n  const arrayType = typeof propertyValues[0];\n  if (!propertyValues.every((item: any) => typeof item === arrayType))\n    throw new InternalError(\n      `Invalid operation. All elements of property ${key as string} must match the same type.`\n    );\n  const uniqueValues = new Set([...propertyValues]);\n  if (arrayType !== \"object\") {\n    const repo = repositoryFromTypeMetadata(model, key);\n    for (const id of uniqueValues) {\n      const read = await repo.read(id);\n      await cacheModelForPopulate(context, model, key, id, read);\n    }\n    (model as any)[key] = [...uniqueValues];\n    return;\n  }\n\n  const pkName = findPrimaryKey(propertyValues[0]).id;\n\n  const result: Set<string> = new Set();\n\n  for (const m of propertyValues) {\n    const record = await createOrUpdate(m, context);\n    await cacheModelForPopulate(context, model, key, record[pkName], record);\n    result.add(record[pkName]);\n  }\n\n  (model as any)[key] = [...result];\n}\n\n/**\n * @description Handles one-to-many relationship updates\n * @summary Processes a one-to-many relationship when updating a model, delegating to oneToManyOnCreate if cascade update is enabled\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToManyOnUpdate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToManyOnUpdate\n *   participant oneToManyOnCreate\n *\n *   Caller->>oneToManyOnUpdate: this, context, data, key, model\n *   oneToManyOnUpdate->>oneToManyOnUpdate: check if cascade.update is CASCADE\n *\n *   alt cascade.update is CASCADE\n *     oneToManyOnUpdate->>oneToManyOnCreate: apply(this, [context, data, key, model])\n *     oneToManyOnCreate-->>oneToManyOnUpdate: void\n *   end\n *\n *   oneToManyOnUpdate-->>Caller: void\n */\nexport async function oneToManyOnUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const { cascade } = data;\n  if (cascade.update !== Cascade.CASCADE) return;\n  return oneToManyOnCreate.apply(this as any, [\n    context,\n    data,\n    key as keyof Model,\n    model,\n  ]);\n}\n\n/**\n * @description Handles one-to-many relationship deletion\n * @summary Processes a one-to-many relationship when deleting a model, deleting all related models if cascade delete is enabled\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToManyOnDelete\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToManyOnDelete\n *   participant Repository\n *   participant repositoryFromTypeMetadata\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToManyOnDelete: this, context, data, key, model\n *   oneToManyOnDelete->>oneToManyOnDelete: check if cascade.delete is CASCADE\n *   oneToManyOnDelete->>oneToManyOnDelete: check if values exists and has length\n *   oneToManyOnDelete->>oneToManyOnDelete: check if all elements have same type\n *\n *   alt isInstantiated (arrayType is \"object\")\n *     oneToManyOnDelete->>Repository: forModel(values[0])\n *     Repository-->>oneToManyOnDelete: repo\n *   else not instantiated\n *     oneToManyOnDelete->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToManyOnDelete: repo\n *   end\n *\n *   oneToManyOnDelete->>oneToManyOnDelete: create uniqueValues set\n *\n *   loop for each id in uniqueValues\n *     oneToManyOnDelete->>repo: delete(id, context)\n *     repo-->>oneToManyOnDelete: deleted\n *     oneToManyOnDelete->>cacheModelForPopulate: context, model, key, id, deleted\n *   end\n *\n *   oneToManyOnDelete->>oneToManyOnDelete: set model[key] = [...uniqueValues]\n *   oneToManyOnDelete-->>Caller: void\n */\nexport async function oneToManyOnDelete<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (data.cascade.delete !== Cascade.CASCADE) return;\n  const values = model[key] as any;\n  if (!values || !values.length) return;\n  const arrayType = typeof values[0];\n  const areAllSameType = values.every((item: any) => typeof item === arrayType);\n  if (!areAllSameType)\n    throw new InternalError(\n      `Invalid operation. All elements of property ${key as string} must match the same type.`\n    );\n  const isInstantiated = arrayType === \"object\";\n  const repo = isInstantiated\n    ? Repository.forModel(values[0],this.adapter.alias)\n    : repositoryFromTypeMetadata(model, key);\n\n  const uniqueValues = new Set([\n    ...(isInstantiated\n      ? values.map((v: Record<string, any>) => v[repo.pk as string])\n      : values),\n  ]);\n\n  for (const id of uniqueValues.values()) {\n    const deleted = await repo.delete(id, context);\n    await cacheModelForPopulate(context, model, key, id, deleted);\n  }\n  (model as any)[key] = [...uniqueValues];\n}\n\n/**\n * @description Generates a key for caching populated model relationships\n * @summary Creates a unique key for storing and retrieving populated model relationships in the cache\n * @param {string} tableName - The name of the table or model\n * @param {string} fieldName - The name of the field or property\n * @param {string|number} id - The identifier of the related model\n * @return {string} A dot-separated string that uniquely identifies the relationship\n * @function getPopulateKey\n * @memberOf module:core\n */\nexport function getPopulateKey(\n  tableName: string,\n  fieldName: string,\n  id: string | number\n) {\n  return [PersistenceKeys.POPULATE, tableName, fieldName, id].join(\".\");\n}\n\n/**\n * @description Caches a model for later population\n * @summary Stores a model in the context cache for efficient retrieval during relationship population\n * @template M - The model type extending Model\n * @template F - The repository flags type\n * @param {Context<F>} context - The context for the operation\n * @param {M} parentModel - The parent model that contains the relationship\n * @param propertyKey - The property key of the relationship\n * @param {string | number} pkValue - The primary key value of the related model\n * @param {any} cacheValue - The model instance to cache\n * @return {Promise<any>} A promise that resolves with the result of the cache operation\n * @function cacheModelForPopulate\n * @memberOf module:core\n */\nexport async function cacheModelForPopulate<\n  M extends Model,\n  F extends RepositoryFlags,\n>(\n  context: Context<F>,\n  parentModel: M,\n  propertyKey: keyof M | string,\n  pkValue: string | number,\n  cacheValue: any\n) {\n  const cacheKey = getPopulateKey(\n    parentModel.constructor.name,\n    propertyKey as string,\n    pkValue\n  );\n  return context.accumulate({ [cacheKey]: cacheValue });\n}\n\n/**\n * @description Populates a model's relationship\n * @summary Retrieves and attaches related models to a model's relationship property\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function populate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant populate\n *   participant fetchPopulateValues\n *   participant getPopulateKey\n *   participant Context\n *   participant repositoryFromTypeMetadata\n *\n *   Caller->>populate: this, context, data, key, model\n *   populate->>populate: check if data.populate is true\n *   populate->>populate: get nested value and check if it exists\n *\n *   populate->>fetchPopulateValues: context, model, key, isArr ? nested : [nested]\n *\n *   fetchPopulateValues->>fetchPopulateValues: initialize variables\n *\n *   loop for each proKeyValue in propKeyValues\n *     fetchPopulateValues->>getPopulateKey: model.constructor.name, propName, proKeyValue\n *     getPopulateKey-->>fetchPopulateValues: cacheKey\n *\n *     alt try to get from cache\n *       fetchPopulateValues->>Context: get(cacheKey)\n *       Context-->>fetchPopulateValues: val\n *     else catch error\n *       fetchPopulateValues->>repositoryFromTypeMetadata: model, propName\n *       repositoryFromTypeMetadata-->>fetchPopulateValues: repo\n *       fetchPopulateValues->>repo: read(proKeyValue)\n *       repo-->>fetchPopulateValues: val\n *     end\n *\n *     fetchPopulateValues->>fetchPopulateValues: add val to results\n *   end\n *\n *   fetchPopulateValues-->>populate: results\n *   populate->>populate: set model[key] = isArr ? res : res[0]\n *   populate-->>Caller: void\n */\nexport async function populate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!data.populate) return;\n  const nested: any = model[key];\n  const isArr = Array.isArray(nested);\n  if (typeof nested === \"undefined\" || (isArr && nested.length === 0)) return;\n\n  async function fetchPopulateValues(\n    c: Context<F>,\n    model: M,\n    propName: string,\n    propKeyValues: any[]\n  ) {\n    let cacheKey: string;\n    let val: any;\n    const results: M[] = [];\n    for (const proKeyValue of propKeyValues) {\n      cacheKey = getPopulateKey(model.constructor.name, propName, proKeyValue);\n      try {\n        val = await c.get(cacheKey as any);\n        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      } catch (e: any) {\n        const repo = repositoryFromTypeMetadata(model, propName);\n        if (!repo) throw new InternalError(\"Could not find repo\");\n        val = await repo.read(proKeyValue);\n      }\n      results.push(val);\n    }\n    return results;\n  }\n  const res = await fetchPopulateValues(\n    context,\n    model,\n    key as string,\n    isArr ? nested : [nested]\n  );\n  (model as any)[key] = isArr ? res : res[0];\n}\n\n/**\n * @description List of common JavaScript types\n * @summary An array of strings representing common JavaScript types that are not custom model types\n * @const commomTypes\n * @memberOf module:core\n */\nconst commomTypes = [\n  \"array\",\n  \"string\",\n  \"number\",\n  \"boolean\",\n  \"symbol\",\n  \"function\",\n  \"object\",\n  \"undefined\",\n  \"null\",\n  \"bigint\",\n];\n\n/**\n * @description Retrieves a repository for a model property based on its type metadata\n * @summary Examines a model property's type metadata to determine the appropriate repository for related models\n * @template M - The model type extending Model\n * @param {any} model - The model instance containing the property\n * @param propertyKey - The property key to examine\n * @return {Repo<M>} A repository for the model type associated with the property\n * @function repositoryFromTypeMetadata\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant repositoryFromTypeMetadata\n *   participant Reflect\n *   participant Validation\n *   participant Model\n *   participant Repository\n *\n *   Caller->>repositoryFromTypeMetadata: model, propertyKey\n *\n *   repositoryFromTypeMetadata->>Validation: key(Array.isArray(model[propertyKey]) ? ValidationKeys.LIST : ValidationKeys.TYPE)\n *   Validation-->>repositoryFromTypeMetadata: validationKey\n *\n *   repositoryFromTypeMetadata->>Reflect: getMetadata(validationKey, model, propertyKey)\n *   Reflect-->>repositoryFromTypeMetadata: types\n *\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: determine customTypes based on property type\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if types and customTypes exist\n *\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: create allowedTypes array\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: find constructorName not in commomTypes\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if constructorName exists\n *\n *   repositoryFromTypeMetadata->>Model: get(constructorName)\n *   Model-->>repositoryFromTypeMetadata: constructor\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if constructor exists\n *\n *   repositoryFromTypeMetadata->>Repository: forModel(constructor)\n *   Repository-->>repositoryFromTypeMetadata: repo\n *\n *   repositoryFromTypeMetadata-->>Caller: repo\n */\nexport function repositoryFromTypeMetadata<M extends Model>(\n  model: any,\n  propertyKey: string | keyof M\n): Repo<M> {\n  const types = Reflect.getMetadata(\n    Validation.key(\n      Array.isArray(model[propertyKey])\n        ? ValidationKeys.LIST\n        : ValidationKeys.TYPE\n    ),\n    model,\n    propertyKey as string\n  );\n  const customTypes: any = Array.isArray(model[propertyKey])\n    ? types.clazz\n    : types.customTypes;\n  if (!types || !customTypes)\n    throw new InternalError(\n      `Failed to find types decorators for property ${propertyKey as string}`\n    );\n\n  const allowedTypes: string[] = Array.isArray(customTypes)\n    ? [...customTypes]\n    : [customTypes];\n  const constructorName = allowedTypes.find(\n    (t) => !commomTypes.includes(`${t}`.toLowerCase())\n  );\n  if (!constructorName)\n    throw new InternalError(\n      `Property key ${propertyKey as string} does not have a valid constructor type`\n    );\n  const constructor: Constructor<M> | undefined = Model.get(constructorName);\n  if (!constructor)\n    throw new InternalError(`No registered model found for ${constructorName}`);\n\n  return Repository.forModel(constructor);\n}\n\n"]}
|
648
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construction.js","sourceRoot":"","sources":["../../../src/model/construction.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Constructor,\n  Model,\n  ModelConstructor,\n  Validation,\n  ValidationKeys,\n} from \"@decaf-ts/decorator-validation\";\nimport { Repo, Repository } from \"../repository/Repository\";\nimport { RelationsMetadata } from \"./types\";\nimport {\n  findPrimaryKey,\n  InternalError,\n  NotFoundError,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport { PersistenceKeys } from \"../persistence/constants\";\nimport { Cascade } from \"../repository/constants\";\nimport { Context } from \"@decaf-ts/db-decorators\";\n\n/**\n * @description Creates or updates a model instance\n * @summary Determines whether to create a new model or update an existing one based on the presence of a primary key\n * @template M - The model type extending Model\n * @template F - The repository flags type\n * @param {M} model - The model instance to create or update\n * @param {Context<F>} context - The context for the operation\n * @param {Repo<M, F, Context<F>>} [repository] - Optional repository to use for the operation\n * @return {Promise<M>} A promise that resolves to the created or updated model\n * @function createOrUpdate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant createOrUpdate\n *   participant Repository\n *   participant Model\n *\n *   Caller->>createOrUpdate: model, context, repository?\n *   alt repository not provided\n *     createOrUpdate->>Model: get(model.constructor.name)\n *     Model-->>createOrUpdate: constructor\n *     createOrUpdate->>Repository: forModel(constructor)\n *     Repository-->>createOrUpdate: repository\n *   end\n *\n *   alt primary key undefined\n *     createOrUpdate->>Repository: create(model, context)\n *     Repository-->>createOrUpdate: created model\n *   else primary key defined\n *     createOrUpdate->>Repository: update(model, context)\n *     alt update successful\n *       Repository-->>createOrUpdate: updated model\n *     else NotFoundError\n *       createOrUpdate->>Repository: create(model, context)\n *       Repository-->>createOrUpdate: created model\n *     end\n *   end\n *\n *   createOrUpdate-->>Caller: model\n */\nexport async function createOrUpdate<\n  M extends Model,\n  F extends RepositoryFlags,\n>(\n  model: M,\n  context: Context<F>,\n  repository?: Repo<M, F, Context<F>>\n): Promise<M> {\n  if (!repository) {\n    const constructor = Model.get(model.constructor.name);\n    if (!constructor)\n      throw new InternalError(`Could not find model ${model.constructor.name}`);\n    repository = Repository.forModel<M, Repo<M>>(\n      constructor as unknown as ModelConstructor<M>\n    );\n  }\n  if (typeof model[repository.pk] === \"undefined\")\n    return repository.create(model, context);\n  else {\n    try {\n      return repository.update(model, context);\n    } catch (e: any) {\n      if (!(e instanceof NotFoundError)) throw e;\n      return repository.create(model, context);\n    }\n  }\n}\n\n/**\n * @description Handles one-to-one relationship creation\n * @summary Processes a one-to-one relationship when creating a model, either by referencing an existing model or creating a new one\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param {string} key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToOneOnCreate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToOneOnCreate\n *   participant repositoryFromTypeMetadata\n *   participant Model\n *   participant Repository\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToOneOnCreate: this, context, data, key, model\n *   oneToOneOnCreate->>oneToOneOnCreate: check if propertyValue exists\n *\n *   alt propertyValue is not an object\n *     oneToOneOnCreate->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToOneOnCreate: innerRepo\n *     oneToOneOnCreate->>innerRepo: read(propertyValue)\n *     innerRepo-->>oneToOneOnCreate: read\n *     oneToOneOnCreate->>cacheModelForPopulate: context, model, key, propertyValue, read\n *     oneToOneOnCreate->>oneToOneOnCreate: set model[key] = propertyValue\n *   else propertyValue is an object\n *     oneToOneOnCreate->>Model: get(data.class)\n *     Model-->>oneToOneOnCreate: constructor\n *     oneToOneOnCreate->>Repository: forModel(constructor)\n *     Repository-->>oneToOneOnCreate: repo\n *     oneToOneOnCreate->>repo: create(propertyValue)\n *     repo-->>oneToOneOnCreate: created\n *     oneToOneOnCreate->>findPrimaryKey: created\n *     findPrimaryKey-->>oneToOneOnCreate: pk\n *     oneToOneOnCreate->>cacheModelForPopulate: context, model, key, created[pk], created\n *     oneToOneOnCreate->>oneToOneOnCreate: set model[key] = created[pk]\n *   end\n *\n *   oneToOneOnCreate-->>Caller: void\n */\nexport async function oneToOneOnCreate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValue: any = model[key];\n  if (!propertyValue) return;\n\n  if (typeof propertyValue !== \"object\") {\n    const innerRepo = repositoryFromTypeMetadata(model, key);\n    const read = await innerRepo.read(propertyValue);\n    await cacheModelForPopulate(context, model, key, propertyValue, read);\n    (model as any)[key] = propertyValue;\n    return;\n  }\n\n  const constructor = Model.get(data.class);\n  if (!constructor)\n    throw new InternalError(`Could not find model ${data.class}`);\n  const repo: Repo<any> = Repository.forModel(constructor);\n  const created = await repo.create(propertyValue);\n  const pk = findPrimaryKey(created).id;\n  await cacheModelForPopulate(context, model, key, created[pk], created);\n  (model as any)[key] = created[pk];\n}\n\n/**\n * @description Handles one-to-one relationship updates\n * @summary Processes a one-to-one relationship when updating a model, either by referencing an existing model or updating the related model\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToOneOnUpdate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToOneOnUpdate\n *   participant repositoryFromTypeMetadata\n *   participant createOrUpdate\n *   participant findPrimaryKey\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToOneOnUpdate: this, context, data, key, model\n *   oneToOneOnUpdate->>oneToOneOnUpdate: check if propertyValue exists\n *   oneToOneOnUpdate->>oneToOneOnUpdate: check if cascade.update is CASCADE\n *\n *   alt propertyValue is not an object\n *     oneToOneOnUpdate->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToOneOnUpdate: innerRepo\n *     oneToOneOnUpdate->>innerRepo: read(propertyValue)\n *     innerRepo-->>oneToOneOnUpdate: read\n *     oneToOneOnUpdate->>cacheModelForPopulate: context, model, key, propertyValue, read\n *     oneToOneOnUpdate->>oneToOneOnUpdate: set model[key] = propertyValue\n *   else propertyValue is an object\n *     oneToOneOnUpdate->>createOrUpdate: model[key], context\n *     createOrUpdate-->>oneToOneOnUpdate: updated\n *     oneToOneOnUpdate->>findPrimaryKey: updated\n *     findPrimaryKey-->>oneToOneOnUpdate: pk\n *     oneToOneOnUpdate->>cacheModelForPopulate: context, model, key, updated[pk], updated\n *     oneToOneOnUpdate->>oneToOneOnUpdate: set model[key] = updated[pk]\n *   end\n *\n *   oneToOneOnUpdate-->>Caller: void\n */\nexport async function oneToOneOnUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValue: any = model[key];\n  if (!propertyValue) return;\n  if (data.cascade.update !== Cascade.CASCADE) return;\n\n  if (typeof propertyValue !== \"object\") {\n    const innerRepo = repositoryFromTypeMetadata(model, key);\n    const read = await innerRepo.read(propertyValue);\n    await cacheModelForPopulate(context, model, key, propertyValue, read);\n    (model as any)[key] = propertyValue;\n    return;\n  }\n\n  const updated = await createOrUpdate(model[key] as M, context);\n  const pk = findPrimaryKey(updated).id;\n  await cacheModelForPopulate(\n    context,\n    model,\n    key,\n    updated[pk] as string,\n    updated\n  );\n  model[key] = updated[pk];\n}\n\n/**\n * @description Handles one-to-one relationship deletion\n * @summary Processes a one-to-one relationship when deleting a model, deleting the related model if cascade is enabled\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToOneOnDelete\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToOneOnDelete\n *   participant repositoryFromTypeMetadata\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToOneOnDelete: this, context, data, key, model\n *   oneToOneOnDelete->>oneToOneOnDelete: check if propertyValue exists\n *   oneToOneOnDelete->>oneToOneOnDelete: check if cascade.update is CASCADE\n *\n *   oneToOneOnDelete->>repositoryFromTypeMetadata: model, key\n *   repositoryFromTypeMetadata-->>oneToOneOnDelete: innerRepo\n *\n *   alt propertyValue is not a Model instance\n *     oneToOneOnDelete->>innerRepo: delete(model[key], context)\n *     innerRepo-->>oneToOneOnDelete: deleted\n *   else propertyValue is a Model instance\n *     oneToOneOnDelete->>innerRepo: delete(model[key][innerRepo.pk], context)\n *     innerRepo-->>oneToOneOnDelete: deleted\n *   end\n *\n *   oneToOneOnDelete->>cacheModelForPopulate: context, model, key, deleted[innerRepo.pk], deleted\n *   oneToOneOnDelete-->>Caller: void\n */\nexport async function oneToOneOnDelete<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValue: any = model[key];\n  if (!propertyValue) return;\n  if (data.cascade.update !== Cascade.CASCADE) return;\n  const innerRepo: Repo<M> = repositoryFromTypeMetadata(model, key);\n  let deleted: M;\n  if (!(propertyValue instanceof Model))\n    deleted = await innerRepo.delete(model[key] as string, context);\n  else\n    deleted = await innerRepo.delete(\n      (model[key] as M)[innerRepo.pk as keyof M] as string,\n      context\n    );\n  await cacheModelForPopulate(\n    context,\n    model,\n    key,\n    deleted[innerRepo.pk] as string,\n    deleted\n  );\n}\n\n/**\n * @description Handles one-to-many relationship creation\n * @summary Processes a one-to-many relationship when creating a model, either by referencing existing models or creating new ones\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToManyOnCreate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToManyOnCreate\n *   participant repositoryFromTypeMetadata\n *   participant createOrUpdate\n *   participant findPrimaryKey\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToManyOnCreate: this, context, data, key, model\n *   oneToManyOnCreate->>oneToManyOnCreate: check if propertyValues exists and has length\n *   oneToManyOnCreate->>oneToManyOnCreate: check if all elements have same type\n *   oneToManyOnCreate->>oneToManyOnCreate: create uniqueValues set\n *\n *   alt arrayType is not \"object\"\n *     oneToManyOnCreate->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToManyOnCreate: repo\n *     loop for each id in uniqueValues\n *       oneToManyOnCreate->>repo: read(id)\n *       repo-->>oneToManyOnCreate: read\n *       oneToManyOnCreate->>cacheModelForPopulate: context, model, key, id, read\n *     end\n *     oneToManyOnCreate->>oneToManyOnCreate: set model[key] = [...uniqueValues]\n *   else arrayType is \"object\"\n *     oneToManyOnCreate->>findPrimaryKey: propertyValues[0]\n *     findPrimaryKey-->>oneToManyOnCreate: pkName\n *     oneToManyOnCreate->>oneToManyOnCreate: create result set\n *     loop for each m in propertyValues\n *       oneToManyOnCreate->>createOrUpdate: m, context\n *       createOrUpdate-->>oneToManyOnCreate: record\n *       oneToManyOnCreate->>cacheModelForPopulate: context, model, key, record[pkName], record\n *       oneToManyOnCreate->>oneToManyOnCreate: add record[pkName] to result\n *     end\n *     oneToManyOnCreate->>oneToManyOnCreate: set model[key] = [...result]\n *   end\n *\n *   oneToManyOnCreate-->>Caller: void\n */\nexport async function oneToManyOnCreate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const propertyValues: any = model[key];\n  if (!propertyValues || !propertyValues.length) return;\n  const arrayType = typeof propertyValues[0];\n  if (!propertyValues.every((item: any) => typeof item === arrayType))\n    throw new InternalError(\n      `Invalid operation. All elements of property ${key as string} must match the same type.`\n    );\n  const uniqueValues = new Set([...propertyValues]);\n  if (arrayType !== \"object\") {\n    const repo = repositoryFromTypeMetadata(model, key);\n    for (const id of uniqueValues) {\n      const read = await repo.read(id);\n      await cacheModelForPopulate(context, model, key, id, read);\n    }\n    (model as any)[key] = [...uniqueValues];\n    return;\n  }\n\n  const pkName = findPrimaryKey(propertyValues[0]).id;\n\n  const result: Set<string> = new Set();\n\n  for (const m of propertyValues) {\n    const record = await createOrUpdate(m, context);\n    await cacheModelForPopulate(context, model, key, record[pkName], record);\n    result.add(record[pkName]);\n  }\n\n  (model as any)[key] = [...result];\n}\n\n/**\n * @description Handles one-to-many relationship updates\n * @summary Processes a one-to-many relationship when updating a model, delegating to oneToManyOnCreate if cascade update is enabled\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToManyOnUpdate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToManyOnUpdate\n *   participant oneToManyOnCreate\n *\n *   Caller->>oneToManyOnUpdate: this, context, data, key, model\n *   oneToManyOnUpdate->>oneToManyOnUpdate: check if cascade.update is CASCADE\n *\n *   alt cascade.update is CASCADE\n *     oneToManyOnUpdate->>oneToManyOnCreate: apply(this, [context, data, key, model])\n *     oneToManyOnCreate-->>oneToManyOnUpdate: void\n *   end\n *\n *   oneToManyOnUpdate-->>Caller: void\n */\nexport async function oneToManyOnUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  const { cascade } = data;\n  if (cascade.update !== Cascade.CASCADE) return;\n  return oneToManyOnCreate.apply(this as any, [\n    context,\n    data,\n    key as keyof Model,\n    model,\n  ]);\n}\n\n/**\n * @description Handles one-to-many relationship deletion\n * @summary Processes a one-to-many relationship when deleting a model, deleting all related models if cascade delete is enabled\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function oneToManyOnDelete\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant oneToManyOnDelete\n *   participant Repository\n *   participant repositoryFromTypeMetadata\n *   participant cacheModelForPopulate\n *\n *   Caller->>oneToManyOnDelete: this, context, data, key, model\n *   oneToManyOnDelete->>oneToManyOnDelete: check if cascade.delete is CASCADE\n *   oneToManyOnDelete->>oneToManyOnDelete: check if values exists and has length\n *   oneToManyOnDelete->>oneToManyOnDelete: check if all elements have same type\n *\n *   alt isInstantiated (arrayType is \"object\")\n *     oneToManyOnDelete->>Repository: forModel(values[0])\n *     Repository-->>oneToManyOnDelete: repo\n *   else not instantiated\n *     oneToManyOnDelete->>repositoryFromTypeMetadata: model, key\n *     repositoryFromTypeMetadata-->>oneToManyOnDelete: repo\n *   end\n *\n *   oneToManyOnDelete->>oneToManyOnDelete: create uniqueValues set\n *\n *   loop for each id in uniqueValues\n *     oneToManyOnDelete->>repo: delete(id, context)\n *     repo-->>oneToManyOnDelete: deleted\n *     oneToManyOnDelete->>cacheModelForPopulate: context, model, key, id, deleted\n *   end\n *\n *   oneToManyOnDelete->>oneToManyOnDelete: set model[key] = [...uniqueValues]\n *   oneToManyOnDelete-->>Caller: void\n */\nexport async function oneToManyOnDelete<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (data.cascade.delete !== Cascade.CASCADE) return;\n  const values = model[key] as any;\n  if (!values || !values.length) return;\n  const arrayType = typeof values[0];\n  const areAllSameType = values.every((item: any) => typeof item === arrayType);\n  if (!areAllSameType)\n    throw new InternalError(\n      `Invalid operation. All elements of property ${key as string} must match the same type.`\n    );\n  const isInstantiated = arrayType === \"object\";\n  const repo = isInstantiated\n    ? Repository.forModel(values[0],this.adapter.alias)\n    : repositoryFromTypeMetadata(model, key);\n\n  const uniqueValues = new Set([\n    ...(isInstantiated\n      ? values.map((v: Record<string, any>) => v[repo.pk as string])\n      : values),\n  ]);\n\n  for (const id of uniqueValues.values()) {\n    const deleted = await repo.delete(id, context);\n    await cacheModelForPopulate(context, model, key, id, deleted);\n  }\n  (model as any)[key] = [...uniqueValues];\n}\n\n/**\n * @description Generates a key for caching populated model relationships\n * @summary Creates a unique key for storing and retrieving populated model relationships in the cache\n * @param {string} tableName - The name of the table or model\n * @param {string} fieldName - The name of the field or property\n * @param {string|number} id - The identifier of the related model\n * @return {string} A dot-separated string that uniquely identifies the relationship\n * @function getPopulateKey\n * @memberOf module:core\n */\nexport function getPopulateKey(\n  tableName: string,\n  fieldName: string,\n  id: string | number\n) {\n  return [PersistenceKeys.POPULATE, tableName, fieldName, id].join(\".\");\n}\n\n/**\n * @description Caches a model for later population\n * @summary Stores a model in the context cache for efficient retrieval during relationship population\n * @template M - The model type extending Model\n * @template F - The repository flags type\n * @param {Context<F>} context - The context for the operation\n * @param {M} parentModel - The parent model that contains the relationship\n * @param propertyKey - The property key of the relationship\n * @param {string | number} pkValue - The primary key value of the related model\n * @param {any} cacheValue - The model instance to cache\n * @return {Promise<any>} A promise that resolves with the result of the cache operation\n * @function cacheModelForPopulate\n * @memberOf module:core\n */\nexport async function cacheModelForPopulate<\n  M extends Model,\n  F extends RepositoryFlags,\n>(\n  context: Context<F>,\n  parentModel: M,\n  propertyKey: keyof M | string,\n  pkValue: string | number,\n  cacheValue: any\n) {\n  const cacheKey = getPopulateKey(\n    parentModel.constructor.name,\n    propertyKey as string,\n    pkValue\n  );\n  return context.accumulate({ [cacheKey]: cacheValue });\n}\n\n/**\n * @description Populates a model's relationship\n * @summary Retrieves and attaches related models to a model's relationship property\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The relations metadata\n * @param key - The property key of the relationship\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the operation is complete\n * @function populate\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant populate\n *   participant fetchPopulateValues\n *   participant getPopulateKey\n *   participant Context\n *   participant repositoryFromTypeMetadata\n *\n *   Caller->>populate: this, context, data, key, model\n *   populate->>populate: check if data.populate is true\n *   populate->>populate: get nested value and check if it exists\n *\n *   populate->>fetchPopulateValues: context, model, key, isArr ? nested : [nested]\n *\n *   fetchPopulateValues->>fetchPopulateValues: initialize variables\n *\n *   loop for each proKeyValue in propKeyValues\n *     fetchPopulateValues->>getPopulateKey: model.constructor.name, propName, proKeyValue\n *     getPopulateKey-->>fetchPopulateValues: cacheKey\n *\n *     alt try to get from cache\n *       fetchPopulateValues->>Context: get(cacheKey)\n *       Context-->>fetchPopulateValues: val\n *     else catch error\n *       fetchPopulateValues->>repositoryFromTypeMetadata: model, propName\n *       repositoryFromTypeMetadata-->>fetchPopulateValues: repo\n *       fetchPopulateValues->>repo: read(proKeyValue)\n *       repo-->>fetchPopulateValues: val\n *     end\n *\n *     fetchPopulateValues->>fetchPopulateValues: add val to results\n *   end\n *\n *   fetchPopulateValues-->>populate: results\n *   populate->>populate: set model[key] = isArr ? res : res[0]\n *   populate-->>Caller: void\n */\nexport async function populate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!data.populate) return;\n  const nested: any = model[key];\n  const isArr = Array.isArray(nested);\n  if (typeof nested === \"undefined\" || (isArr && nested.length === 0)) return;\n\n  async function fetchPopulateValues(\n    c: Context<F>,\n    model: M,\n    propName: string,\n    propKeyValues: any[]\n  ) {\n    let cacheKey: string;\n    let val: any;\n    const results: M[] = [];\n    for (const proKeyValue of propKeyValues) {\n      cacheKey = getPopulateKey(model.constructor.name, propName, proKeyValue);\n      try {\n        val = await c.get(cacheKey as any);\n        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      } catch (e: any) {\n        const repo = repositoryFromTypeMetadata(model, propName);\n        if (!repo) throw new InternalError(\"Could not find repo\");\n        val = await repo.read(proKeyValue);\n      }\n      results.push(val);\n    }\n    return results;\n  }\n  const res = await fetchPopulateValues(\n    context,\n    model,\n    key as string,\n    isArr ? nested : [nested]\n  );\n  (model as any)[key] = isArr ? res : res[0];\n}\n\n/**\n * @description List of common JavaScript types\n * @summary An array of strings representing common JavaScript types that are not custom model types\n * @const commomTypes\n * @memberOf module:core\n */\nconst commomTypes = [\n  \"array\",\n  \"string\",\n  \"number\",\n  \"boolean\",\n  \"symbol\",\n  \"function\",\n  \"object\",\n  \"undefined\",\n  \"null\",\n  \"bigint\",\n];\n\n/**\n * @description Retrieves a repository for a model property based on its type metadata\n * @summary Examines a model property's type metadata to determine the appropriate repository for related models\n * @template M - The model type extending Model\n * @param {any} model - The model instance containing the property\n * @param propertyKey - The property key to examine\n * @return {Repo<M>} A repository for the model type associated with the property\n * @function repositoryFromTypeMetadata\n * @memberOf module:core\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant repositoryFromTypeMetadata\n *   participant Reflect\n *   participant Validation\n *   participant Model\n *   participant Repository\n *\n *   Caller->>repositoryFromTypeMetadata: model, propertyKey\n *\n *   repositoryFromTypeMetadata->>Validation: key(Array.isArray(model[propertyKey]) ? ValidationKeys.LIST : ValidationKeys.TYPE)\n *   Validation-->>repositoryFromTypeMetadata: validationKey\n *\n *   repositoryFromTypeMetadata->>Reflect: getMetadata(validationKey, model, propertyKey)\n *   Reflect-->>repositoryFromTypeMetadata: types\n *\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: determine customTypes based on property type\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if types and customTypes exist\n *\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: create allowedTypes array\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: find constructorName not in commomTypes\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if constructorName exists\n *\n *   repositoryFromTypeMetadata->>Model: get(constructorName)\n *   Model-->>repositoryFromTypeMetadata: constructor\n *   repositoryFromTypeMetadata->>repositoryFromTypeMetadata: check if constructor exists\n *\n *   repositoryFromTypeMetadata->>Repository: forModel(constructor)\n *   Repository-->>repositoryFromTypeMetadata: repo\n *\n *   repositoryFromTypeMetadata-->>Caller: repo\n */\nexport function repositoryFromTypeMetadata<M extends Model>(\n  model: any,\n  propertyKey: string | keyof M\n): Repo<M> {\n  const types = Reflect.getMetadata(\n    Validation.key(\n      Array.isArray(model[propertyKey])\n        ? ValidationKeys.LIST\n        : ValidationKeys.TYPE\n    ),\n    model,\n    propertyKey as string\n  );\n  const customTypes: any = Array.isArray(model[propertyKey])\n    ? types.clazz\n    : types.customTypes;\n  if (!types || !customTypes)\n    throw new InternalError(\n      `Failed to find types decorators for property ${propertyKey as string}`\n    );\n\n  const allowedTypes: string[] = Array.isArray(customTypes)\n    ? [...customTypes]\n    : [customTypes];\n  const constructorName = allowedTypes.find(\n    (t) => !commomTypes.includes(`${t}`.toLowerCase())\n  );\n  if (!constructorName)\n    throw new InternalError(\n      `Property key ${propertyKey as string} does not have a valid constructor type`\n    );\n  const constructor: Constructor<M> | undefined = Model.get(constructorName);\n  if (!constructor)\n    throw new InternalError(`No registered model found for ${constructorName}`);\n\n  return Repository.forModel(constructor);\n}\n\n"]}
|
@@ -1,13 +1,13 @@
|
|
1
1
|
import { ConflictError, onCreate, onCreateUpdate, onDelete, onUpdate, afterAny, } from "@decaf-ts/db-decorators";
|
2
2
|
import { apply, metadata } from "@decaf-ts/reflection";
|
3
|
-
import { PersistenceKeys } from "
|
4
|
-
import { DefaultCascade } from "
|
3
|
+
import { PersistenceKeys } from "./../persistence/constants.js";
|
4
|
+
import { DefaultCascade } from "./../repository/constants.js";
|
5
5
|
import { Decoration, list, Model, prop, propMetadata, type, } from "@decaf-ts/decorator-validation";
|
6
|
-
import { Adapter } from "
|
7
|
-
import { Repository } from "
|
8
|
-
import { Condition } from "
|
9
|
-
import { oneToManyOnCreate, oneToManyOnDelete, oneToManyOnUpdate, oneToOneOnCreate, oneToOneOnDelete, oneToOneOnUpdate, populate as pop, } from "./construction";
|
10
|
-
import { AuthorizationError } from "
|
6
|
+
import { Adapter } from "./../persistence/Adapter.js";
|
7
|
+
import { Repository } from "./../repository/Repository.js";
|
8
|
+
import { Condition } from "./../query/Condition.js";
|
9
|
+
import { oneToManyOnCreate, oneToManyOnDelete, oneToManyOnUpdate, oneToOneOnCreate, oneToOneOnDelete, oneToOneOnUpdate, populate as pop, } from "./construction.js";
|
10
|
+
import { AuthorizationError } from "./../utils/index.js";
|
11
11
|
/**
|
12
12
|
* @description Specifies the database table name for a model
|
13
13
|
* @summary Decorator that sets the table name for a model class in the database
|
@@ -281,4 +281,4 @@ export function manyToOne(clazz, cascadeOptions = DefaultCascade, populate = tru
|
|
281
281
|
propMetadata(key, metadata))
|
282
282
|
.apply();
|
283
283
|
}
|
284
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../../src/model/decorators.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,aAAa,EACb,QAAQ,EACR,cAAc,EACd,QAAQ,EACR,QAAQ,EACR,QAAQ,GAGT,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AACvD,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAE3D,OAAO,EAAE,cAAc,EAAkB,MAAM,yBAAyB,CAAC;AACzE,OAAO,EAEL,UAAU,EACV,IAAI,EACJ,KAAK,EACL,IAAI,EACJ,YAAY,EACZ,IAAI,GACL,MAAM,gCAAgC,CAAC;AACxC,OAAO,EAAE,OAAO,EAAE,MAAM,wBAAwB,CAAC;AACjD,OAAO,EAAQ,UAAU,EAAE,MAAM,0BAA0B,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAE/C,OAAO,EACL,iBAAiB,EACjB,iBAAiB,EACjB,iBAAiB,EACjB,gBAAgB,EAChB,gBAAgB,EAChB,gBAAgB,EAChB,QAAQ,IAAI,GAAG,GAChB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,kBAAkB,EAAE,MAAM,UAAU,CAAC;AAE9C;;;;;;;GAOG;AACH,MAAM,UAAU,KAAK,CAAC,SAAiB;IACrC,OAAO,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,KAAK,CAAC,EAAE,SAAS,CAAC,CAAC;AACjE,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,MAAM,CAAC,UAAkB;IACvC,OAAO,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE,UAAU,CAAC,CAAC;AACvE,CAAC;AAED;;;;;;;;GAQG;AACH,MAAM,UAAU,KAAK,CAAC,UAA6B,EAAE,YAAuB;IAC1E,OAAO,YAAY,CACjB,UAAU,CAAC,GAAG,CACZ,GAAG,eAAe,CAAC,KAAK,GAAG,YAAY,IAAI,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CACrG,EACD;QACE,UAAU,EAAE,UAAU;QACtB,YAAY,EAAE,YAAY;KACV,CACnB,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;;GAgBG;AACH,MAAM,CAAC,KAAK,UAAU,oBAAoB,CAQxC,OAAmB,EACnB,IAAO,EACP,GAAY,EACZ,KAAQ;IAER,IAAI,CAAE,KAAa,CAAC,GAAG,CAAC;QAAE,OAAO;IACjC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,EAAE;SACjC,KAAK,CAAC,SAAS,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;SAC9C,OAAO,EAAE,CAAC;IACb,IAAI,QAAQ,CAAC,MAAM;QACjB,MAAM,IAAI,aAAa,CACrB,sCAAsC,GAAa,aAAa,IAAI,CAAC,SAAS,CAAE,KAAa,CAAC,GAAG,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC,EAAE,CACpH,CAAC;AACN,CAAC;AAED;;;;;;;;;;;;;;GAcG;AACH,MAAM,UAAU,MAAM;IACpB,OAAO,KAAK,CACV,cAAc,CAAC,oBAAoB,CAAC,EACpC,YAAY,CAAC,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,CACzD,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;;GAgBG;AACH,MAAM,CAAC,KAAK,UAAU,uBAAuB;AAQ3C,6DAA6D;AAC7D,OAAmB;AACnB,6DAA6D;AAC7D,IAAO;AACP,6DAA6D;AAC7D,GAAY;AACZ,6DAA6D;AAC7D,KAAQ;IAER,MAAM,IAAI,kBAAkB,CAC1B,mDAAmD,CACpD,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;GAaG;AACH,MAAM,UAAU,SAAS;IACvB,MAAM,GAAG,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;IACvD,OAAO,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC;SACvB,MAAM,CAAC,QAAQ,CAAC,uBAAuB,CAAC,EAAE,YAAY,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;SAChE,KAAK,EAAE,CAAC;AACb,CAAC;AAED;;;;;;;;;;;;;GAaG;AACH,MAAM,UAAU,SAAS;IACvB,MAAM,GAAG,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;IACvD,OAAO,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC;SACvB,MAAM,CAAC,cAAc,CAAC,uBAAuB,CAAC,EAAE,YAAY,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;SACtE,KAAK,EAAE,CAAC;AACb,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AACH,MAAM,UAAU,QAAQ,CACtB,KAAqB,EACrB,iBAAkC,cAAc,EAChD,WAAoB,IAAI;IAExB,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACtB,MAAM,QAAQ,GAAsB;QAClC,KAAK,EAAE,KAAK,CAAC,IAAI;QACjB,OAAO,EAAE,cAAc;QACvB,QAAQ,EAAE,QAAQ;KACnB,CAAC;IACF,MAAM,GAAG,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;IACvD,OAAO,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC;SACvB,MAAM,CACL,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,EAC/B,IAAI,CAAC,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,EACzD,QAAQ,CAAC,gBAAgB,EAAE,QAAQ,CAAC,EACpC,QAAQ,CAAC,gBAAgB,EAAE,QAAQ,CAAC,EACpC,QAAQ,CAAC,gBAAgB,EAAE,QAAQ,CAAC,EACpC,QAAQ,CAAC,GAAG,EAAE,QAAQ,CAAC,EACvB,YAAY,CAAC,GAAG,EAAE,QAAQ,CAAC,CAC5B;SACA,KAAK,EAAE,CAAC;AACb,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AACH,MAAM,UAAU,SAAS,CACvB,KAAqB,EACrB,iBAAkC,cAAc,EAChD,WAAoB,IAAI;IAExB,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACtB,MAAM,QAAQ,GAAsB;QAClC,KAAK,EAAE,KAAK,CAAC,IAAI;QACjB,OAAO,EAAE,cAAc;QACvB,QAAQ,EAAE,QAAQ;KACnB,CAAC;IACF,MAAM,GAAG,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,WAAW,CAAC,CAAC;IACxD,OAAO,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC;SACvB,MAAM,CACL,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;IAC/B,uCAAuC;IACvC,IAAI,CAAC,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC,EACrC,QAAQ,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EACrC,QAAQ,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EACrC,QAAQ,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EACrC,QAAQ,CAAC,GAAG,EAAE,QAAQ,CAAC,EACvB,YAAY,CAAC,GAAG,EAAE,QAAQ,CAAC,CAC5B;SACA,KAAK,EAAE,CAAC;AACb,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AACH,MAAM,UAAU,SAAS,CACvB,KAAqB,EACrB,iBAAkC,cAAc,EAChD,QAAQ,GAAG,IAAI;IAEf,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACtB,MAAM,QAAQ,GAAsB;QAClC,KAAK,EAAE,KAAK,CAAC,IAAI;QACjB,OAAO,EAAE,cAAc;QACvB,QAAQ,EAAE,QAAQ;KACnB,CAAC;IACF,MAAM,GAAG,GAAG,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,WAAW,CAAC,CAAC;IACxD,OAAO,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC;SACvB,MAAM,CACL,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,EAC/B,IAAI,CAAC,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;IACzD,yCAAyC;IACzC,yCAAyC;IACzC,yCAAyC;IACzC,gCAAgC;IAChC,YAAY,CAAC,GAAG,EAAE,QAAQ,CAAC,CAC5B;SACA,KAAK,EAAE,CAAC;AACb,CAAC","sourcesContent":["import {\n  ConflictError,\n  onCreate,\n  onCreateUpdate,\n  onDelete,\n  onUpdate,\n  afterAny,\n  RepositoryFlags,\n  Context,\n} from \"@decaf-ts/db-decorators\";\nimport { apply, metadata } from \"@decaf-ts/reflection\";\nimport { PersistenceKeys } from \"../persistence/constants\";\nimport { CascadeMetadata, IndexMetadata } from \"../repository/types\";\nimport { DefaultCascade, OrderDirection } from \"../repository/constants\";\nimport {\n  Constructor,\n  Decoration,\n  list,\n  Model,\n  prop,\n  propMetadata,\n  type,\n} from \"@decaf-ts/decorator-validation\";\nimport { Adapter } from \"../persistence/Adapter\";\nimport { Repo, Repository } from \"../repository/Repository\";\nimport { Condition } from \"../query/Condition\";\nimport { RelationsMetadata } from \"./types\";\nimport {\n  oneToManyOnCreate,\n  oneToManyOnDelete,\n  oneToManyOnUpdate,\n  oneToOneOnCreate,\n  oneToOneOnDelete,\n  oneToOneOnUpdate,\n  populate as pop,\n} from \"./construction\";\nimport { AuthorizationError } from \"../utils\";\n\n/**\n * @description Specifies the database table name for a model\n * @summary Decorator that sets the table name for a model class in the database\n * @param {string} tableName - The name of the table in the database\n * @return {Function} A decorator function that can be applied to a class\n * @function table\n * @category Class Decorators\n */\nexport function table(tableName: string) {\n  return metadata(Adapter.key(PersistenceKeys.TABLE), tableName);\n}\n\n/**\n * @description Specifies the database column name for a model property\n * @summary Decorator that maps a model property to a specific column name in the database\n * @param {string} columnName - The name of the column in the database\n * @return {Function} A decorator function that can be applied to a class property\n * @function column\n * @category Property Decorators\n */\nexport function column(columnName: string) {\n  return propMetadata(Adapter.key(PersistenceKeys.COLUMN), columnName);\n}\n\n/**\n * @description Creates an index on a model property for improved query performance\n * @summary Decorator that marks a property to be indexed in the database, optionally with specific directions and compositions\n * @param {OrderDirection[]} [directions] - Optional array of sort directions for the index\n * @param {string[]} [compositions] - Optional array of property names to create a composite index\n * @return {Function} A decorator function that can be applied to a class property\n * @function index\n * @category Property Decorators\n */\nexport function index(directions?: OrderDirection[], compositions?: string[]) {\n  return propMetadata(\n    Repository.key(\n      `${PersistenceKeys.INDEX}${compositions && compositions.length ? `.${compositions.join(\".\")}` : \"\"}`\n    ),\n    {\n      directions: directions,\n      compositions: compositions,\n    } as IndexMetadata\n  );\n}\n\n/**\n * @description Enforces uniqueness constraint during model creation and update\n * @summary Internal function used by the unique decorator to check if a property value already exists in the database\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The metadata type\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The metadata for the property\n * @param key - The property key to check for uniqueness\n * @param {M} model - The model instance being created or updated\n * @return {Promise<void>} A promise that resolves when the check is complete or rejects with a ConflictError\n * @function uniqueOnCreateUpdate\n * @memberOf module:core\n */\nexport async function uniqueOnCreateUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends object,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!(model as any)[key]) return;\n  const existing = await this.select()\n    .where(Condition.attribute(key).eq(model[key]))\n    .execute();\n  if (existing.length)\n    throw new ConflictError(\n      `model already exists with property ${key as string} equal to ${JSON.stringify((model as any)[key], undefined, 2)}`\n    );\n}\n\n/**\n * @description Tags a property as unique\n * @summary Decorator that ensures a property value is unique across all instances of a model in the database\n * @return {Function} A decorator function that can be applied to a class property\n * @function unique\n * @category Property Decorators\n * @example\n * ```typescript\n * class User extends BaseModel {\n *   @unique()\n *   @required()\n *   username!: string;\n * }\n * ```\n */\nexport function unique() {\n  return apply(\n    onCreateUpdate(uniqueOnCreateUpdate),\n    propMetadata(Repository.key(PersistenceKeys.UNIQUE), {})\n  );\n}\n\n/**\n * @description Handles user identification for ownership tracking\n * @summary Internal function used by the createdBy and updatedBy decorators to set ownership information\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The metadata for the property\n * @param key - The property key to store the user identifier\n * @param {M} model - The model instance being created or updated\n * @return {Promise<void>} A promise that rejects with an AuthorizationError if user identification is not supported\n * @function createdByOnCreateUpdate\n * @memberOf module:core\n */\nexport async function createdByOnCreateUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  context: Context<F>,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  data: V,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  key: keyof M,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  model: M\n): Promise<void> {\n  throw new AuthorizationError(\n    \"This adapter does not support user identification\"\n  );\n}\n\n/**\n * @description Tracks the creator of a model instance\n * @summary Decorator that marks a property to store the identifier of the user who created the model instance\n * @return {Function} A decorator function that can be applied to a class property\n * @function createdBy\n * @category Property Decorators\n * @example\n * ```typescript\n * class Document extends BaseModel {\n *   @createdBy()\n *   creator!: string;\n * }\n * ```\n */\nexport function createdBy() {\n  const key = Repository.key(PersistenceKeys.CREATED_BY);\n  return Decoration.for(key)\n    .define(onCreate(createdByOnCreateUpdate), propMetadata(key, {}))\n    .apply();\n}\n\n/**\n * @description Tracks the last updater of a model instance\n * @summary Decorator that marks a property to store the identifier of the user who last updated the model instance\n * @return {Function} A decorator function that can be applied to a class property\n * @function updatedBy\n * @category Property Decorators\n * @example\n * ```typescript\n * class Document extends BaseModel {\n *   @updatedBy()\n *   lastEditor!: string;\n * }\n * ```\n */\nexport function updatedBy() {\n  const key = Repository.key(PersistenceKeys.UPDATED_BY);\n  return Decoration.for(key)\n    .define(onCreateUpdate(createdByOnCreateUpdate), propMetadata(key, {}))\n    .apply();\n}\n\n/**\n * @description Defines a one-to-one relationship between models\n * @summary Decorator that establishes a one-to-one relationship between the current model and another model\n * @template M - The related model type extending Model\n * @param {Constructor<M>} clazz - The constructor of the related model class\n * @param {CascadeMetadata} [cascadeOptions=DefaultCascade] - Options for cascading operations (create, update, delete)\n * @param {boolean} [populate=true] - If true, automatically populates the relationship when the model is retrieved\n * @return {Function} A decorator function that can be applied to a class property\n * @function oneToOne\n * @category Property Decorators\n * @example\n * ```typescript\n * class User extends BaseModel {\n *   @oneToOne(Profile)\n *   profile!: string | Profile;\n * }\n *\n * class Profile extends BaseModel {\n *   @required()\n *   bio!: string;\n * }\n * ```\n * @see oneToMany\n * @see manyToOne\n */\nexport function oneToOne<M extends Model>(\n  clazz: Constructor<M>,\n  cascadeOptions: CascadeMetadata = DefaultCascade,\n  populate: boolean = true\n) {\n  Model.register(clazz);\n  const metadata: RelationsMetadata = {\n    class: clazz.name,\n    cascade: cascadeOptions,\n    populate: populate,\n  };\n  const key = Repository.key(PersistenceKeys.ONE_TO_ONE);\n  return Decoration.for(key)\n    .define(\n      prop(PersistenceKeys.RELATIONS),\n      type([clazz.name, String.name, Number.name, BigInt.name]),\n      onCreate(oneToOneOnCreate, metadata),\n      onUpdate(oneToOneOnUpdate, metadata),\n      onDelete(oneToOneOnDelete, metadata),\n      afterAny(pop, metadata),\n      propMetadata(key, metadata)\n    )\n    .apply();\n}\n\n/**\n * @description Defines a one-to-many relationship between models\n * @summary Decorator that establishes a one-to-many relationship between the current model and multiple instances of another model\n * @template M - The related model type extending Model\n * @param {Constructor<M>} clazz - The constructor of the related model class\n * @param {CascadeMetadata} [cascadeOptions=DefaultCascade] - Options for cascading operations (create, update, delete)\n * @param {boolean} [populate=true] - If true, automatically populates the relationship when the model is retrieved\n * @return {Function} A decorator function that can be applied to a class property\n * @function oneToMany\n * @category Property Decorators\n * @example\n * ```typescript\n * class Author extends BaseModel {\n *   @required()\n *   name!: string;\n *\n *   @oneToMany(Book)\n *   books!: string[] | Book[];\n * }\n *\n * class Book extends BaseModel {\n *   @required()\n *   title!: string;\n * }\n * ```\n * @see oneToOne\n * @see manyToOne\n */\nexport function oneToMany<M extends Model>(\n  clazz: Constructor<M>,\n  cascadeOptions: CascadeMetadata = DefaultCascade,\n  populate: boolean = true\n) {\n  Model.register(clazz);\n  const metadata: RelationsMetadata = {\n    class: clazz.name,\n    cascade: cascadeOptions,\n    populate: populate,\n  };\n  const key = Repository.key(PersistenceKeys.ONE_TO_MANY);\n  return Decoration.for(key)\n    .define(\n      prop(PersistenceKeys.RELATIONS),\n      // @ts-expect-error purposeful override\n      list([clazz, String, Number, BigInt]),\n      onCreate(oneToManyOnCreate, metadata),\n      onUpdate(oneToManyOnUpdate, metadata),\n      onDelete(oneToManyOnDelete, metadata),\n      afterAny(pop, metadata),\n      propMetadata(key, metadata)\n    )\n    .apply();\n}\n\n/**\n * @description Defines a many-to-one relationship between models\n * @summary Decorator that establishes a many-to-one relationship between multiple instances of the current model and another model\n * @template M - The related model type extending Model\n * @param {Constructor<M>} clazz - The constructor of the related model class\n * @param {CascadeMetadata} [cascadeOptions=DefaultCascade] - Options for cascading operations (create, update, delete)\n * @param {boolean} [populate=true] - If true, automatically populates the relationship when the model is retrieved\n * @return {Function} A decorator function that can be applied to a class property\n * @function manyToOne\n * @category Property Decorators\n * @example\n * ```typescript\n * class Book extends BaseModel {\n *   @required()\n *   title!: string;\n *\n *   @manyToOne(Author)\n *   author!: string | Author;\n * }\n *\n * class Author extends BaseModel {\n *   @required()\n *   name!: string;\n * }\n * ```\n * @see oneToMany\n * @see oneToOne\n */\nexport function manyToOne<M extends Model>(\n  clazz: Constructor<M>,\n  cascadeOptions: CascadeMetadata = DefaultCascade,\n  populate = true\n) {\n  Model.register(clazz);\n  const metadata: RelationsMetadata = {\n    class: clazz.name,\n    cascade: cascadeOptions,\n    populate: populate,\n  };\n  const key = Repository.key(PersistenceKeys.MANY_TO_ONE);\n  return Decoration.for(key)\n    .define(\n      prop(PersistenceKeys.RELATIONS),\n      type([clazz.name, String.name, Number.name, BigInt.name]),\n      // onCreate(oneToManyOnCreate, metadata),\n      // onUpdate(oneToManyOnUpdate, metadata),\n      // onDelete(oneToManyOnDelete, metadata),\n      // afterAll(populate, metadata),\n      propMetadata(key, metadata)\n    )\n    .apply();\n}\n"]}
|
284
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../../src/model/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ConflictError,\n  onCreate,\n  onCreateUpdate,\n  onDelete,\n  onUpdate,\n  afterAny,\n  RepositoryFlags,\n  Context,\n} from \"@decaf-ts/db-decorators\";\nimport { apply, metadata } from \"@decaf-ts/reflection\";\nimport { PersistenceKeys } from \"../persistence/constants\";\nimport { CascadeMetadata, IndexMetadata } from \"../repository/types\";\nimport { DefaultCascade, OrderDirection } from \"../repository/constants\";\nimport {\n  Constructor,\n  Decoration,\n  list,\n  Model,\n  prop,\n  propMetadata,\n  type,\n} from \"@decaf-ts/decorator-validation\";\nimport { Adapter } from \"../persistence/Adapter\";\nimport { Repo, Repository } from \"../repository/Repository\";\nimport { Condition } from \"../query/Condition\";\nimport { RelationsMetadata } from \"./types\";\nimport {\n  oneToManyOnCreate,\n  oneToManyOnDelete,\n  oneToManyOnUpdate,\n  oneToOneOnCreate,\n  oneToOneOnDelete,\n  oneToOneOnUpdate,\n  populate as pop,\n} from \"./construction\";\nimport { AuthorizationError } from \"../utils\";\n\n/**\n * @description Specifies the database table name for a model\n * @summary Decorator that sets the table name for a model class in the database\n * @param {string} tableName - The name of the table in the database\n * @return {Function} A decorator function that can be applied to a class\n * @function table\n * @category Class Decorators\n */\nexport function table(tableName: string) {\n  return metadata(Adapter.key(PersistenceKeys.TABLE), tableName);\n}\n\n/**\n * @description Specifies the database column name for a model property\n * @summary Decorator that maps a model property to a specific column name in the database\n * @param {string} columnName - The name of the column in the database\n * @return {Function} A decorator function that can be applied to a class property\n * @function column\n * @category Property Decorators\n */\nexport function column(columnName: string) {\n  return propMetadata(Adapter.key(PersistenceKeys.COLUMN), columnName);\n}\n\n/**\n * @description Creates an index on a model property for improved query performance\n * @summary Decorator that marks a property to be indexed in the database, optionally with specific directions and compositions\n * @param {OrderDirection[]} [directions] - Optional array of sort directions for the index\n * @param {string[]} [compositions] - Optional array of property names to create a composite index\n * @return {Function} A decorator function that can be applied to a class property\n * @function index\n * @category Property Decorators\n */\nexport function index(directions?: OrderDirection[], compositions?: string[]) {\n  return propMetadata(\n    Repository.key(\n      `${PersistenceKeys.INDEX}${compositions && compositions.length ? `.${compositions.join(\".\")}` : \"\"}`\n    ),\n    {\n      directions: directions,\n      compositions: compositions,\n    } as IndexMetadata\n  );\n}\n\n/**\n * @description Enforces uniqueness constraint during model creation and update\n * @summary Internal function used by the unique decorator to check if a property value already exists in the database\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The metadata type\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The metadata for the property\n * @param key - The property key to check for uniqueness\n * @param {M} model - The model instance being created or updated\n * @return {Promise<void>} A promise that resolves when the check is complete or rejects with a ConflictError\n * @function uniqueOnCreateUpdate\n * @memberOf module:core\n */\nexport async function uniqueOnCreateUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends object,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!(model as any)[key]) return;\n  const existing = await this.select()\n    .where(Condition.attribute(key).eq(model[key]))\n    .execute();\n  if (existing.length)\n    throw new ConflictError(\n      `model already exists with property ${key as string} equal to ${JSON.stringify((model as any)[key], undefined, 2)}`\n    );\n}\n\n/**\n * @description Tags a property as unique\n * @summary Decorator that ensures a property value is unique across all instances of a model in the database\n * @return {Function} A decorator function that can be applied to a class property\n * @function unique\n * @category Property Decorators\n * @example\n * ```typescript\n * class User extends BaseModel {\n *   @unique()\n *   @required()\n *   username!: string;\n * }\n * ```\n */\nexport function unique() {\n  return apply(\n    onCreateUpdate(uniqueOnCreateUpdate),\n    propMetadata(Repository.key(PersistenceKeys.UNIQUE), {})\n  );\n}\n\n/**\n * @description Handles user identification for ownership tracking\n * @summary Internal function used by the createdBy and updatedBy decorators to set ownership information\n * @template M - The model type extending Model\n * @template R - The repository type extending Repo<M, F, C>\n * @template V - The relations metadata type extending RelationsMetadata\n * @template F - The repository flags type\n * @template C - The context type extending Context<F>\n * @param {R} this - The repository instance\n * @param {Context<F>} context - The context for the operation\n * @param {V} data - The metadata for the property\n * @param key - The property key to store the user identifier\n * @param {M} model - The model instance being created or updated\n * @return {Promise<void>} A promise that rejects with an AuthorizationError if user identification is not supported\n * @function createdByOnCreateUpdate\n * @memberOf module:core\n */\nexport async function createdByOnCreateUpdate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends RelationsMetadata,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  context: Context<F>,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  data: V,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  key: keyof M,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  model: M\n): Promise<void> {\n  throw new AuthorizationError(\n    \"This adapter does not support user identification\"\n  );\n}\n\n/**\n * @description Tracks the creator of a model instance\n * @summary Decorator that marks a property to store the identifier of the user who created the model instance\n * @return {Function} A decorator function that can be applied to a class property\n * @function createdBy\n * @category Property Decorators\n * @example\n * ```typescript\n * class Document extends BaseModel {\n *   @createdBy()\n *   creator!: string;\n * }\n * ```\n */\nexport function createdBy() {\n  const key = Repository.key(PersistenceKeys.CREATED_BY);\n  return Decoration.for(key)\n    .define(onCreate(createdByOnCreateUpdate), propMetadata(key, {}))\n    .apply();\n}\n\n/**\n * @description Tracks the last updater of a model instance\n * @summary Decorator that marks a property to store the identifier of the user who last updated the model instance\n * @return {Function} A decorator function that can be applied to a class property\n * @function updatedBy\n * @category Property Decorators\n * @example\n * ```typescript\n * class Document extends BaseModel {\n *   @updatedBy()\n *   lastEditor!: string;\n * }\n * ```\n */\nexport function updatedBy() {\n  const key = Repository.key(PersistenceKeys.UPDATED_BY);\n  return Decoration.for(key)\n    .define(onCreateUpdate(createdByOnCreateUpdate), propMetadata(key, {}))\n    .apply();\n}\n\n/**\n * @description Defines a one-to-one relationship between models\n * @summary Decorator that establishes a one-to-one relationship between the current model and another model\n * @template M - The related model type extending Model\n * @param {Constructor<M>} clazz - The constructor of the related model class\n * @param {CascadeMetadata} [cascadeOptions=DefaultCascade] - Options for cascading operations (create, update, delete)\n * @param {boolean} [populate=true] - If true, automatically populates the relationship when the model is retrieved\n * @return {Function} A decorator function that can be applied to a class property\n * @function oneToOne\n * @category Property Decorators\n * @example\n * ```typescript\n * class User extends BaseModel {\n *   @oneToOne(Profile)\n *   profile!: string | Profile;\n * }\n *\n * class Profile extends BaseModel {\n *   @required()\n *   bio!: string;\n * }\n * ```\n * @see oneToMany\n * @see manyToOne\n */\nexport function oneToOne<M extends Model>(\n  clazz: Constructor<M>,\n  cascadeOptions: CascadeMetadata = DefaultCascade,\n  populate: boolean = true\n) {\n  Model.register(clazz);\n  const metadata: RelationsMetadata = {\n    class: clazz.name,\n    cascade: cascadeOptions,\n    populate: populate,\n  };\n  const key = Repository.key(PersistenceKeys.ONE_TO_ONE);\n  return Decoration.for(key)\n    .define(\n      prop(PersistenceKeys.RELATIONS),\n      type([clazz.name, String.name, Number.name, BigInt.name]),\n      onCreate(oneToOneOnCreate, metadata),\n      onUpdate(oneToOneOnUpdate, metadata),\n      onDelete(oneToOneOnDelete, metadata),\n      afterAny(pop, metadata),\n      propMetadata(key, metadata)\n    )\n    .apply();\n}\n\n/**\n * @description Defines a one-to-many relationship between models\n * @summary Decorator that establishes a one-to-many relationship between the current model and multiple instances of another model\n * @template M - The related model type extending Model\n * @param {Constructor<M>} clazz - The constructor of the related model class\n * @param {CascadeMetadata} [cascadeOptions=DefaultCascade] - Options for cascading operations (create, update, delete)\n * @param {boolean} [populate=true] - If true, automatically populates the relationship when the model is retrieved\n * @return {Function} A decorator function that can be applied to a class property\n * @function oneToMany\n * @category Property Decorators\n * @example\n * ```typescript\n * class Author extends BaseModel {\n *   @required()\n *   name!: string;\n *\n *   @oneToMany(Book)\n *   books!: string[] | Book[];\n * }\n *\n * class Book extends BaseModel {\n *   @required()\n *   title!: string;\n * }\n * ```\n * @see oneToOne\n * @see manyToOne\n */\nexport function oneToMany<M extends Model>(\n  clazz: Constructor<M>,\n  cascadeOptions: CascadeMetadata = DefaultCascade,\n  populate: boolean = true\n) {\n  Model.register(clazz);\n  const metadata: RelationsMetadata = {\n    class: clazz.name,\n    cascade: cascadeOptions,\n    populate: populate,\n  };\n  const key = Repository.key(PersistenceKeys.ONE_TO_MANY);\n  return Decoration.for(key)\n    .define(\n      prop(PersistenceKeys.RELATIONS),\n      // @ts-expect-error purposeful override\n      list([clazz, String, Number, BigInt]),\n      onCreate(oneToManyOnCreate, metadata),\n      onUpdate(oneToManyOnUpdate, metadata),\n      onDelete(oneToManyOnDelete, metadata),\n      afterAny(pop, metadata),\n      propMetadata(key, metadata)\n    )\n    .apply();\n}\n\n/**\n * @description Defines a many-to-one relationship between models\n * @summary Decorator that establishes a many-to-one relationship between multiple instances of the current model and another model\n * @template M - The related model type extending Model\n * @param {Constructor<M>} clazz - The constructor of the related model class\n * @param {CascadeMetadata} [cascadeOptions=DefaultCascade] - Options for cascading operations (create, update, delete)\n * @param {boolean} [populate=true] - If true, automatically populates the relationship when the model is retrieved\n * @return {Function} A decorator function that can be applied to a class property\n * @function manyToOne\n * @category Property Decorators\n * @example\n * ```typescript\n * class Book extends BaseModel {\n *   @required()\n *   title!: string;\n *\n *   @manyToOne(Author)\n *   author!: string | Author;\n * }\n *\n * class Author extends BaseModel {\n *   @required()\n *   name!: string;\n * }\n * ```\n * @see oneToMany\n * @see oneToOne\n */\nexport function manyToOne<M extends Model>(\n  clazz: Constructor<M>,\n  cascadeOptions: CascadeMetadata = DefaultCascade,\n  populate = true\n) {\n  Model.register(clazz);\n  const metadata: RelationsMetadata = {\n    class: clazz.name,\n    cascade: cascadeOptions,\n    populate: populate,\n  };\n  const key = Repository.key(PersistenceKeys.MANY_TO_ONE);\n  return Decoration.for(key)\n    .define(\n      prop(PersistenceKeys.RELATIONS),\n      type([clazz.name, String.name, Number.name, BigInt.name]),\n      // onCreate(oneToManyOnCreate, metadata),\n      // onUpdate(oneToManyOnUpdate, metadata),\n      // onDelete(oneToManyOnDelete, metadata),\n      // afterAll(populate, metadata),\n      propMetadata(key, metadata)\n    )\n    .apply();\n}\n"]}
|
package/lib/esm/model/index.js
CHANGED
@@ -1,5 +1,5 @@
|
|
1
|
-
export * from "./BaseModel";
|
2
|
-
export * from "./construction";
|
3
|
-
export * from "./decorators";
|
4
|
-
export * from "./types";
|
5
|
-
//# sourceMappingURL=data:application/json;base64,
|
1
|
+
export * from "./BaseModel.js";
|
2
|
+
export * from "./construction.js";
|
3
|
+
export * from "./decorators.js";
|
4
|
+
export * from "./types.js";
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbW9kZWwvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsK0JBQTRCO0FBQzVCLGtDQUErQjtBQUMvQixnQ0FBNkI7QUFDN0IsMkJBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vQmFzZU1vZGVsXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9jb25zdHJ1Y3Rpb25cIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3R5cGVzXCI7XG4iXX0=
|
@@ -207,7 +207,7 @@ export declare abstract class Adapter<Y, Q, F extends RepositoryFlags, C extends
|
|
207
207
|
* @summary Reference to the context class constructor used by this adapter
|
208
208
|
*/
|
209
209
|
protected Context: {
|
210
|
-
new (
|
210
|
+
new (): Context<F>;
|
211
211
|
factory: import("@decaf-ts/db-decorators").ContextFactory<any>;
|
212
212
|
childFrom<F_1 extends RepositoryFlags, C_1 extends Context<F_1>>(context: C_1, overrides?: Partial<F_1>): C_1;
|
213
213
|
from<M extends Model, F_1 extends RepositoryFlags, C_1 extends Context<F_1>>(operation: OperationKeys.CREATE | OperationKeys.READ | OperationKeys.UPDATE | OperationKeys.DELETE, overrides: Partial<F_1>, model: Constructor<M>, ...args: any[]): Promise<C_1>;
|