UNPKG

@decaf-ts/db-decorators

Version:

Agnostic database decorators and repository

230 lines 25.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.Context = exports.DefaultContextFactory = void 0; const constants_1 = require("./constants.cjs"); const typed_object_accumulator_1 = require("typed-object-accumulator"); /** * @description Default factory for creating context instances. * @summary A factory function that creates new Context instances with the provided repository flags. * It automatically adds a timestamp to the context and returns a properly typed context instance. * @const DefaultContextFactory * @memberOf module:db-decorators */ const DefaultContextFactory = (arg) => { return new Context().accumulate(Object.assign({}, arg, { timestamp: new Date() })); }; exports.DefaultContextFactory = DefaultContextFactory; /** * @description A context management class for handling repository operations. * @summary The Context class provides a mechanism for managing repository operations with flags, * parent-child relationships, and state accumulation. It allows for hierarchical context chains * and maintains operation-specific configurations while supporting type safety through generics. * * @template F - Type extending RepositoryFlags that defines the context configuration * * @param {ObjectAccumulator<F>} cache - The internal cache storing accumulated values * * @class * * @example * ```typescript * // Creating a new context with repository flags * const context = new Context<RepositoryFlags>(); * * // Accumulating values * const enrichedContext = context.accumulate({ * writeOperation: true, * affectedTables: ['users'], * operation: OperationKeys.CREATE * }); * * // Accessing values * const isWrite = enrichedContext.get('writeOperation'); // true * const tables = enrichedContext.get('affectedTables'); // ['users'] * ``` * * @mermaid * sequenceDiagram * participant C as Client * participant Ctx as Context * participant Cache as ObjectAccumulator * * C->>Ctx: new Context() * Ctx->>Cache: create cache * * C->>Ctx: accumulate(value) * Ctx->>Cache: accumulate(value) * Cache-->>Ctx: updated cache * Ctx-->>C: updated context * * C->>Ctx: get(key) * Ctx->>Cache: get(key) * alt Key exists in cache * Cache-->>Ctx: value * else Key not found * Ctx->>Ctx: check parent context * alt Parent exists * Ctx->>Parent: get(key) * Parent-->>Ctx: value * else No parent * Ctx-->>C: throw error * end * end * Ctx-->>C: requested value */ class Context { constructor() { this.cache = new typed_object_accumulator_1.ObjectAccumulator(); Object.defineProperty(this, "cache", { value: new typed_object_accumulator_1.ObjectAccumulator(), writable: false, enumerable: false, configurable: true, }); } static { this.factory = exports.DefaultContextFactory; } /** * @description Accumulates new values into the context. * @summary Merges the provided value object with the existing context state, * creating a new immutable cache state. * * @template F - current accumulator type * @template V - Type extending object for the values to accumulate * @param {V} value - The object containing values to accumulate * @returns A new context instance with accumulated values */ accumulate(value) { Object.defineProperty(this, "cache", { value: this.cache.accumulate(value), writable: false, enumerable: false, configurable: true, }); return this; } get timestamp() { return this.cache.timestamp; } /** * @description Retrieves a value from the context by key. * @summary Attempts to get a value from the current context's cache. * If not found, traverses up the parent context chain. * * @template K - Type extending keyof F for the key to retrieve * @template F - Accumulator type * @param {K} key - The key to retrieve from the context * @returns The value associated with the key * @throws {Error} If the key is not found in the context chain */ get(key) { try { return this.cache.get(key); } catch (e) { if (this.cache.parentContext) return this.cache.parentContext.get(key); throw e; } } /** * @description Creates a child context * @summary Generates a new context instance with current context as parent * * @template M - Type extending Model * @param {OperationKeys} operation - The operation type * @param {Constructor<M>} [model] - Optional model constructor * @returns {C} New child context instance */ child(operation, model) { return Context.childFrom(this, { operation: operation, affectedTables: model ? [model] : [], }); } /** * @description Creates a child context from another context * @summary Generates a new context instance with parent reference * * @template F - Type extending Repository Flags * @template C - Type extending Context<F> * @param {C} context - The parent context * @param {Partial<F>} [overrides] - Optional flag overrides * @returns {C} New child context instance */ static childFrom(context, overrides) { return Context.factory(Object.assign({}, context.cache, overrides || {})); } /** * @description Creates a new context from operation parameters * @summary Generates a context instance for specific operation * * @template F - Type extending Repository Flags * @template M - Type extending Model * @param {OperationKeys.DELETE} operation - The operation type * @param {Partial<F>} overrides - Flag overrides * @param {Constructor<M>} model - The model constructor * @param {any} args - Operation arguments * @returns {Promise<C>} Promise resolving to new context */ static async from(operation, overrides, model, // eslint-disable-next-line @typescript-eslint/no-unused-vars ...args) { return Context.factory(Object.assign({}, constants_1.DefaultRepositoryFlags, overrides, { operation: operation, model: model, })); } /** * @description Prepares arguments for context operations * @summary Creates a context args object with the specified operation parameters * * @template F - Type extending {@link RepositoryFlags} * @template M - Type extending {@link Model} * @param {OperationKeys.DELETE} operation - The operation type * @param {Constructor<M>} model - The model constructor * @param {any[]} args - Operation arguments * @param {Contextual<F>} [contextual] - Optional contextual object * @param {Partial<F>} [overrides] - Optional flag overrides * @returns {Promise<ContextArgs>} Promise resolving to context arguments * * @mermaid * sequenceDiagram * participant C as Context * participant M as Model * participant A as Args * * C->>C: Receive operation request * C->>M: Validate model constructor * C->>C: Create child context * C->>A: Process operation args * A->>C: Return context args * C->>C: Apply overrides * C->>C: Return final context */ static async args(operation, model, args, contextual, overrides) { const last = args.pop(); async function getContext() { if (contextual) return contextual.context(operation, overrides || {}, model, ...args); return Context.from(operation, overrides || {}, model, ...args); } let c; if (last) { if (last instanceof Context) { c = last; args.push(last); } else { c = (await getContext()); args.push(last, c); } } else { c = (await getContext()); args.push(c); } return { context: c, args: args }; } } exports.Context = Context; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Context.js","sourceRoot":"","sources":["../../src/repository/Context.ts"],"names":[],"mappings":"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","sourcesContent":["import { ContextArgs } from \"./utils\";\nimport { Contextual } from \"../interfaces/Contextual\";\nimport { OperationKeys } from \"../operations/constants\";\nimport { Constructor, Model } from \"@decaf-ts/decorator-validation\";\nimport { DefaultRepositoryFlags } from \"./constants\";\nimport { ObjectAccumulator } from \"typed-object-accumulator\";\nimport { RepositoryFlags } from \"./types\";\n\n/**\n * @description Factory type for creating context instances.\n * @summary Defines a function type that creates context instances with specific repository flags.\n * @template F - The repository flags type extending RepositoryFlags\n * @typedef {Function} ContextFactory\n * @memberOf module:db-decorators\n */\nexport type ContextFactory<F extends RepositoryFlags> = <C extends Context<F>>(\n  arg: Omit<F, \"timestamp\">\n) => C;\n\n/**\n * @description Default factory for creating context instances.\n * @summary A factory function that creates new Context instances with the provided repository flags.\n * It automatically adds a timestamp to the context and returns a properly typed context instance.\n * @const DefaultContextFactory\n * @memberOf module:db-decorators\n */\nexport const DefaultContextFactory: ContextFactory<any> = <\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  arg: Omit<F, \"timestamp\">\n) => {\n  return new Context<F>().accumulate(\n    Object.assign({}, arg, { timestamp: new Date() }) as F\n  ) as C;\n};\n\n/**\n * @description A context management class for handling repository operations.\n * @summary The Context class provides a mechanism for managing repository operations with flags,\n * parent-child relationships, and state accumulation. It allows for hierarchical context chains\n * and maintains operation-specific configurations while supporting type safety through generics.\n *\n * @template F - Type extending RepositoryFlags that defines the context configuration\n *\n * @param {ObjectAccumulator<F>} cache - The internal cache storing accumulated values\n *\n * @class\n *\n * @example\n * ```typescript\n * // Creating a new context with repository flags\n * const context = new Context<RepositoryFlags>();\n *\n * // Accumulating values\n * const enrichedContext = context.accumulate({\n *   writeOperation: true,\n *   affectedTables: ['users'],\n *   operation: OperationKeys.CREATE\n * });\n *\n * // Accessing values\n * const isWrite = enrichedContext.get('writeOperation'); // true\n * const tables = enrichedContext.get('affectedTables'); // ['users']\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant Ctx as Context\n *   participant Cache as ObjectAccumulator\n *\n *   C->>Ctx: new Context()\n *   Ctx->>Cache: create cache\n *\n *   C->>Ctx: accumulate(value)\n *   Ctx->>Cache: accumulate(value)\n *   Cache-->>Ctx: updated cache\n *   Ctx-->>C: updated context\n *\n *   C->>Ctx: get(key)\n *   Ctx->>Cache: get(key)\n *   alt Key exists in cache\n *     Cache-->>Ctx: value\n *   else Key not found\n *     Ctx->>Ctx: check parent context\n *     alt Parent exists\n *       Ctx->>Parent: get(key)\n *       Parent-->>Ctx: value\n *     else No parent\n *       Ctx-->>C: throw error\n *     end\n *   end\n *   Ctx-->>C: requested value\n */\nexport class Context<F extends RepositoryFlags> {\n  constructor() {\n    Object.defineProperty(this, \"cache\", {\n      value: new ObjectAccumulator<F>(),\n      writable: false,\n      enumerable: false,\n      configurable: true,\n    });\n  }\n\n  static factory: ContextFactory<any> = DefaultContextFactory;\n\n  private readonly cache: F & ObjectAccumulator<F> =\n    new ObjectAccumulator() as F & ObjectAccumulator<F>;\n\n  /**\n   * @description Accumulates new values into the context.\n   * @summary Merges the provided value object with the existing context state,\n   * creating a new immutable cache state.\n   *\n   * @template F - current accumulator type\n   * @template V - Type extending object for the values to accumulate\n   * @param {V} value - The object containing values to accumulate\n   * @returns A new context instance with accumulated values\n   */\n  accumulate<V extends object>(value: V) {\n    Object.defineProperty(this, \"cache\", {\n      value: this.cache.accumulate(value),\n      writable: false,\n      enumerable: false,\n      configurable: true,\n    });\n    return this as unknown as Context<F & V>;\n  }\n\n  get timestamp() {\n    return this.cache.timestamp;\n  }\n\n  /**\n   * @description Retrieves a value from the context by key.\n   * @summary Attempts to get a value from the current context's cache.\n   * If not found, traverses up the parent context chain.\n   *\n   * @template K - Type extending keyof F for the key to retrieve\n   * @template F - Accumulator type\n   * @param {K} key - The key to retrieve from the context\n   * @returns The value associated with the key\n   * @throws {Error} If the key is not found in the context chain\n   */\n  get<K extends keyof F>(key: K): F[K] {\n    try {\n      return this.cache.get(key);\n    } catch (e: unknown) {\n      if (this.cache.parentContext) return this.cache.parentContext.get(key);\n      throw e;\n    }\n  }\n\n  /**\n   * @description Creates a child context\n   * @summary Generates a new context instance with current context as parent\n   *\n   * @template M - Type extending Model\n   * @param {OperationKeys} operation - The operation type\n   * @param {Constructor<M>} [model] - Optional model constructor\n   * @returns {C} New child context instance\n   */\n  child<M extends Model, C extends Context<F>>(\n    operation: OperationKeys,\n    model?: Constructor<M>\n  ): C {\n    return Context.childFrom<F, C>(\n      this as unknown as C,\n      {\n        operation: operation,\n        affectedTables: model ? [model] : [],\n      } as unknown as Partial<F>\n    );\n  }\n\n  /**\n   * @description Creates a child context from another context\n   * @summary Generates a new context instance with parent reference\n   *\n   * @template F - Type extending Repository Flags\n   * @template C - Type extending Context<F>\n   * @param {C} context - The parent context\n   * @param {Partial<F>} [overrides] - Optional flag overrides\n   * @returns {C} New child context instance\n   */\n  static childFrom<F extends RepositoryFlags, C extends Context<F>>(\n    context: C,\n    overrides?: Partial<F>\n  ): C {\n    return Context.factory(\n      Object.assign({}, context.cache, overrides || {})\n    ) as unknown as C;\n  }\n\n  /**\n   * @description Creates a new context from operation parameters\n   * @summary Generates a context instance for specific operation\n   *\n   * @template F - Type extending Repository Flags\n   * @template M - Type extending Model\n   * @param {OperationKeys.DELETE} operation - The operation type\n   * @param {Partial<F>} overrides - Flag overrides\n   * @param {Constructor<M>} model - The model constructor\n   * @param {any} args - Operation arguments\n   * @returns {Promise<C>} Promise resolving to new context\n   */\n  static async from<\n    M extends Model,\n    F extends RepositoryFlags,\n    C extends Context<F>,\n  >(\n    operation:\n      | OperationKeys.CREATE\n      | OperationKeys.READ\n      | OperationKeys.UPDATE\n      | OperationKeys.DELETE,\n    overrides: Partial<F>,\n    model: Constructor<M>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: any[]\n  ): Promise<C> {\n    return Context.factory(\n      Object.assign({}, DefaultRepositoryFlags, overrides, {\n        operation: operation,\n        model: model,\n      })\n    ) as C;\n  }\n\n  /**\n   * @description Prepares arguments for context operations\n   * @summary Creates a context args object with the specified operation parameters\n   *\n   * @template F - Type extending {@link RepositoryFlags}\n   * @template M - Type extending {@link Model}\n   * @param {OperationKeys.DELETE} operation - The operation type\n   * @param {Constructor<M>} model - The model constructor\n   * @param {any[]} args - Operation arguments\n   * @param {Contextual<F>} [contextual] - Optional contextual object\n   * @param {Partial<F>} [overrides] - Optional flag overrides\n   * @returns {Promise<ContextArgs>} Promise resolving to context arguments\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Context\n   *   participant M as Model\n   *   participant A as Args\n   *\n   *   C->>C: Receive operation request\n   *   C->>M: Validate model constructor\n   *   C->>C: Create child context\n   *   C->>A: Process operation args\n   *   A->>C: Return context args\n   *   C->>C: Apply overrides\n   *   C->>C: Return final context\n   */\n  static async args<\n    M extends Model,\n    C extends Context<F>,\n    F extends RepositoryFlags,\n  >(\n    operation:\n      | OperationKeys.CREATE\n      | OperationKeys.READ\n      | OperationKeys.UPDATE\n      | OperationKeys.DELETE,\n    model: Constructor<M>,\n    args: any[],\n    contextual?: Contextual<F>,\n    overrides?: Partial<F>\n  ): Promise<ContextArgs<F, C>> {\n    const last = args.pop();\n\n    async function getContext() {\n      if (contextual)\n        return contextual.context(operation, overrides || {}, model, ...args);\n      return Context.from(operation, overrides || {}, model, ...args);\n    }\n\n    let c: C;\n    if (last) {\n      if (last instanceof Context) {\n        c = last as C;\n        args.push(last);\n      } else {\n        c = (await getContext()) as C;\n        args.push(last, c);\n      }\n    } else {\n      c = (await getContext()) as C;\n      args.push(c);\n    }\n\n    return { context: c, args: args };\n  }\n}\n"]}