@pipeline-builder/pipeline-data 3.4.18 → 3.4.19

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.
@@ -54,19 +54,26 @@ export interface PaginatedResult<T> {
54
54
  nextCursor?: string;
55
55
  }
56
56
  /**
57
- * Abstract CRUD service with access control and common operations.
57
+ * Abstract CRUD service with multi-tenant access control and pagination.
58
+ *
59
+ * Subclasses bind to a specific Drizzle table by implementing `schema`,
60
+ * `buildConditions`, `getSortColumn`, and the org/project-column accessors.
58
61
  *
59
62
  * @typeParam TEntity - Entity type extending BaseEntity
60
63
  * @typeParam TFilter - Filter type for query parameters
61
64
  * @typeParam TInsert - Insert DTO type
62
65
  * @typeParam TUpdate - Update DTO type
63
66
  *
64
- * Type assertions (`as any`, `as unknown as T`) are used throughout for Drizzle ORM compatibility.
65
- * This is safe because: access control filters by orgId, schema validation is at the DB level,
66
- * and each subclass is tested for type correctness.
67
+ * **A note on the type casts.** Drizzle's row types are inferred from
68
+ * `pgTable(...)` and don't generically narrow through the abstract `schema`
69
+ * getter, so the base class casts query results to `TEntity` and back. The
70
+ * cast is *not* a runtime safety guarantee — it relies on each subclass
71
+ * passing the matching entity type. Org-scoping is enforced by every
72
+ * subclass's `buildConditions` injecting `WHERE org_id = $1`; this class
73
+ * does not add that filter itself.
67
74
  *
68
- * Errors are not caught here — they propagate to the route-level error handler (`withRoute`)
69
- * which provides consistent logging with request context.
75
+ * **Error policy.** Errors propagate up to the route-level handler
76
+ * (`withRoute`); no catch-and-swallow here.
70
77
  *
71
78
  * @example
72
79
  * ```typescript
@@ -25,19 +25,26 @@ function drizzleCount(rows) {
25
25
  return rows;
26
26
  }
27
27
  /**
28
- * Abstract CRUD service with access control and common operations.
28
+ * Abstract CRUD service with multi-tenant access control and pagination.
29
+ *
30
+ * Subclasses bind to a specific Drizzle table by implementing `schema`,
31
+ * `buildConditions`, `getSortColumn`, and the org/project-column accessors.
29
32
  *
30
33
  * @typeParam TEntity - Entity type extending BaseEntity
31
34
  * @typeParam TFilter - Filter type for query parameters
32
35
  * @typeParam TInsert - Insert DTO type
33
36
  * @typeParam TUpdate - Update DTO type
34
37
  *
35
- * Type assertions (`as any`, `as unknown as T`) are used throughout for Drizzle ORM compatibility.
36
- * This is safe because: access control filters by orgId, schema validation is at the DB level,
37
- * and each subclass is tested for type correctness.
38
+ * **A note on the type casts.** Drizzle's row types are inferred from
39
+ * `pgTable(...)` and don't generically narrow through the abstract `schema`
40
+ * getter, so the base class casts query results to `TEntity` and back. The
41
+ * cast is *not* a runtime safety guarantee — it relies on each subclass
42
+ * passing the matching entity type. Org-scoping is enforced by every
43
+ * subclass's `buildConditions` injecting `WHERE org_id = $1`; this class
44
+ * does not add that filter itself.
38
45
  *
39
- * Errors are not caught here — they propagate to the route-level error handler (`withRoute`)
40
- * which provides consistent logging with request context.
46
+ * **Error policy.** Errors propagate up to the route-level handler
47
+ * (`withRoute`); no catch-and-swallow here.
41
48
  *
42
49
  * @example
43
50
  * ```typescript
@@ -384,4 +391,4 @@ class CrudService {
384
391
  }
385
392
  }
386
393
  exports.CrudService = CrudService;
387
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crud-service.js","sourceRoot":"","sources":["../../src/api/crud-service.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { NotFoundError, createLogger } from '@pipeline-builder/api-core';\nimport { SQL, eq, and, asc, desc, sql, inArray } from 'drizzle-orm';\nimport type { AnyColumn } from 'drizzle-orm/column';\nimport type { PgTable } from 'drizzle-orm/pg-core';\nimport { db } from '../database/postgres-connection';\n\n/** Pagination defaults — read from env to match CoreConstants in pipeline-core. */\nconst DEFAULT_PAGE_LIMIT = parseInt(process.env.DEFAULT_PAGE_LIMIT || '100', 10);\nconst MAX_PAGE_LIMIT = parseInt(process.env.MAX_PAGE_LIMIT || '1000', 10);\n\n/**\n * Cast Drizzle query results to a typed array.\n * Drizzle's generic return type (`PgSelectBase<...>`) doesn't narrow to our\n * entity generics, so an explicit cast is needed. Centralised here so every\n * call-site stays one-liner clean and the cast is documented in one place.\n */\nexport function drizzleRows<T>(rows: unknown): T[] {\n  return rows as T[];\n}\n\n/** Cast a Drizzle aggregate result to extract `[{ count: number }]`. */\nexport function drizzleCount(rows: unknown): [{ count: number }] {\n  return rows as [{ count: number }];\n}\n\n/**\n * Base interface for entities with common fields\n */\nexport interface BaseEntity {\n  id: string;\n  orgId: string;\n  isDefault: boolean;\n  createdAt: Date;\n  updatedAt: Date;\n  createdBy: string;\n  updatedBy: string;\n  [key: string]: unknown;\n}\n\n/**\n * Pagination and sorting options\n */\nexport interface QueryOptions {\n  limit?: number;\n  offset?: number;\n  sortBy?: string;\n  sortOrder?: 'asc' | 'desc';\n  /** When true, runs a separate COUNT(*) query to include exact total. Default: false. */\n  includeTotal?: boolean;\n  /** Cursor-based pagination: fetch rows after this cursor value (uses sortBy column). */\n  cursor?: string;\n  /** Sparse fieldset: column names to select. Returns all columns when omitted. */\n  fields?: string[];\n}\n\n/**\n * Paginated result with metadata\n */\nexport interface PaginatedResult<T> {\n  data: T[];\n  /** Total count of matching entities. Only present when includeTotal is true. */\n  total?: number;\n  limit: number;\n  offset: number;\n  hasMore: boolean;\n  /** Cursor pointing to the last item, for cursor-based pagination. */\n  nextCursor?: string;\n}\n\n/**\n * Abstract CRUD service with access control and common operations.\n *\n * @typeParam TEntity - Entity type extending BaseEntity\n * @typeParam TFilter - Filter type for query parameters\n * @typeParam TInsert - Insert DTO type\n * @typeParam TUpdate - Update DTO type\n *\n * Type assertions (`as any`, `as unknown as T`) are used throughout for Drizzle ORM compatibility.\n * This is safe because: access control filters by orgId, schema validation is at the DB level,\n * and each subclass is tested for type correctness.\n *\n * Errors are not caught here — they propagate to the route-level error handler (`withRoute`)\n * which provides consistent logging with request context.\n *\n * @example\n * ```typescript\n * class PipelineService extends CrudService<Pipeline, PipelineFilter, PipelineInsert, PipelineUpdate> {\n *   protected get schema() { return schema.pipeline; }\n *   protected buildConditions(filter, orgId) { return buildPipelineConditions(filter, orgId); }\n *   protected getSortColumn(sortBy) { return sortColumnMap[sortBy] ?? null; }\n *   protected getProjectColumn() { return schema.pipeline.project; }\n *   protected getOrgColumn() { return schema.pipeline.organization; }\n * }\n * ```\n */\nexport abstract class CrudService<\n  TEntity extends BaseEntity,\n  TFilter,\n  TInsert,\n  TUpdate,\n> {\n  /** Drizzle schema table for this entity */\n  protected abstract get schema(): PgTable;\n\n  /** Build SQL conditions for filtering entities */\n  protected abstract buildConditions(filter: Partial<TFilter>, orgId?: string): SQL[];\n\n  /** Get the schema column for sorting by field name */\n  protected abstract getSortColumn(sortBy: string): AnyColumn | null;\n\n  /** Get the project column for setDefault scoping (null if entity has no project scope) */\n  protected abstract getProjectColumn(): AnyColumn | null;\n\n  /** Get the organization column for setDefault scoping */\n  protected abstract getOrgColumn(): AnyColumn;\n\n  /** Get the unique constraint columns for onConflictDoUpdate */\n  protected abstract get conflictTarget(): AnyColumn[];\n\n  private readonly _logger = createLogger('CrudService');\n\n  /** Build conditions for a single entity by ID. */\n  private idConditions(id: string, orgId?: string): SQL[] {\n    return this.buildConditions({ id } as unknown as Partial<TFilter>, orgId);\n  }\n\n  // Lifecycle hooks — override in subclasses to react to mutations\n  // These are fire-and-forget: errors are logged but never block the caller.\n\n  /** Called after a new entity is created */\n  protected async onAfterCreate(_entity: TEntity, _userId: string): Promise<void> {}\n\n  /** Called after an entity is updated */\n  protected async onAfterUpdate(_id: string, _entity: TEntity, _userId: string): Promise<void> {}\n\n  /** Called after an entity is soft-deleted */\n  protected async onAfterDelete(_id: string, _entity: TEntity, _userId: string): Promise<void> {}\n\n  /**\n   * Find entities matching filter criteria\n   *\n   * @param filter - Filter criteria\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   */\n  async find(filter: Partial<TFilter>, orgId?: string): Promise<TEntity[]> {\n    const conditions = this.buildConditions(filter, orgId);\n\n    return db\n      .select()\n      .from(this.schema)\n      .where(and(...conditions)).then(r => drizzleRows<TEntity>(r));\n  }\n\n  /**\n   * Find entities with pagination and sorting\n   *\n   * @param filter - Filter criteria\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   * @param options - Pagination and sorting options\n   */\n  async findPaginated(\n    filter: Partial<TFilter>,\n    orgId?: string,\n    options: QueryOptions = {},\n  ): Promise<PaginatedResult<TEntity>> {\n    const { limit: rawLimit = DEFAULT_PAGE_LIMIT, offset = 0, sortBy, sortOrder = 'asc', includeTotal = false, cursor, fields } = options;\n    const limit = Math.min(Math.max(1, rawLimit), MAX_PAGE_LIMIT);\n\n    // Cursor and offset are mutually exclusive — cursor takes precedence\n    const useCursor = !!(cursor && sortBy);\n\n    const conditions = this.buildConditions(filter, orgId);\n\n    // Cursor-based pagination: add WHERE clause for keyset pagination\n    if (useCursor) {\n      const sortColumn = this.getSortColumn(sortBy);\n      if (sortColumn) {\n        const op = sortOrder === 'desc'\n          ? sql`${sortColumn} < ${cursor}`\n          : sql`${sortColumn} > ${cursor}`;\n        conditions.push(op);\n      }\n    }\n\n    // Build SELECT — sparse fieldset when fields are specified\n    const selectSpec = fields ? this.buildFieldSelect(fields) : undefined;\n    let query = selectSpec\n      ? db.select(selectSpec as any).from(this.schema).where(and(...conditions))\n      : db.select().from(this.schema).where(and(...conditions));\n\n    if (sortBy) {\n      const sortColumn = this.getSortColumn(sortBy);\n      if (sortColumn) {\n        query = query.orderBy(sortOrder === 'desc' ? desc(sortColumn) : asc(sortColumn)) as any;\n      }\n    }\n\n    // Fetch limit+1 to detect hasMore without COUNT(*)\n    const effectiveOffset = useCursor ? 0 : offset;\n    const rows = await query\n      .limit(limit + 1)\n      .offset(effectiveOffset).then(r => drizzleRows<TEntity>(r));\n\n    const hasMore = rows.length > limit;\n    const data = hasMore ? rows.slice(0, limit) : rows;\n\n    const result: PaginatedResult<TEntity> = { data, limit, offset: effectiveOffset, hasMore };\n\n    // Provide next cursor from last item's sort column value\n    if (data.length > 0 && sortBy) {\n      const lastItem = data[data.length - 1] as Record<string, unknown>;\n      const cursorValue = lastItem[sortBy];\n      if (cursorValue !== undefined) {\n        result.nextCursor = cursorValue instanceof Date ? cursorValue.toISOString() : String(cursorValue);\n      }\n    }\n\n    // Only run the COUNT(*) query when the caller explicitly needs the total\n    if (includeTotal) {\n      const baseConditions = this.buildConditions(filter, orgId);\n      const [countResult] = await db\n        .select({ count: sql<number>`count(*)::int` })\n        .from(this.schema)\n        .where(and(...baseConditions)).then(r => drizzleCount(r));\n      result.total = countResult?.count || 0;\n    }\n\n    return result;\n  }\n\n  /**\n   * Build a column selection map for sparse fieldsets.\n   * Falls back to full select if no matching columns found.\n   */\n  private buildFieldSelect(fields: string[]): Record<string, unknown> | undefined {\n    if (fields.length === 0) return undefined;\n\n    const columns: Record<string, unknown> = {};\n    // Always include id for entity identity\n    columns.id = (this.schema as any).id;\n\n    for (const field of fields) {\n      if (field === 'id') continue; // Already included\n      const col = (this.schema as any)[field];\n      if (col) columns[field] = col;\n    }\n\n    // At minimum we'll have { id }, which is valid\n    return columns;\n  }\n\n  /**\n   * Count entities matching filter criteria\n   *\n   * @param filter - Filter criteria\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   */\n  async count(filter: Partial<TFilter>, orgId?: string): Promise<number> {\n    const conditions = this.buildConditions(filter, orgId);\n\n    const [result] = await db\n      .select({ count: sql<number>`count(*)::int` })\n      .from(this.schema)\n      .where(and(...conditions)).then(r => drizzleCount(r));\n\n    return result?.count || 0;\n  }\n\n  /**\n   * Find a single entity by ID\n   *\n   * @param id - Entity ID\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   */\n  async findById(id: string, orgId?: string): Promise<TEntity | null> {\n    const conditions = this.idConditions(id, orgId);\n\n    const results = await db\n      .select()\n      .from(this.schema)\n      .where(and(...conditions))\n      .limit(1).then(r => drizzleRows<TEntity>(r));\n\n    return results[0] || null;\n  }\n\n  // Mutation operations\n\n  /**\n   * Create a new entity\n   */\n  async create(data: TInsert, userId: string): Promise<TEntity> {\n    const [created] = await db\n      .insert(this.schema)\n      .values({\n        ...data,\n        createdBy: userId || 'system',\n        updatedBy: userId || 'system',\n      } as any)\n      .onConflictDoUpdate({\n        target: this.conflictTarget as any,\n        set: {\n          ...data,\n          updatedAt: new Date(),\n          updatedBy: userId || 'system',\n        } as any,\n      })\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    this.onAfterCreate(created, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n\n    return created;\n  }\n\n  /**\n   * Update an existing entity\n   */\n  async update(\n    id: string,\n    data: Partial<TUpdate>,\n    orgId: string,\n    userId: string,\n  ): Promise<TEntity | null> {\n    const conditions = this.idConditions(id, orgId);\n\n    const [updated] = await db\n      .update(this.schema)\n      .set({\n        ...data,\n        updatedAt: new Date(),\n        updatedBy: userId || 'system',\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    if (updated) {\n      this.onAfterUpdate(id, updated, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return updated || null;\n  }\n\n  /**\n   * Delete an entity (soft delete by setting isActive = false)\n   */\n  async delete(id: string, orgId: string, userId: string): Promise<TEntity | null> {\n    const conditions = this.idConditions(id, orgId);\n\n    const [deleted] = await db\n      .update(this.schema)\n      .set({\n        isActive: false,\n        updatedAt: new Date(),\n        updatedBy: userId || 'system',\n        deletedAt: new Date(),\n        deletedBy: userId || 'system',\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    if (deleted) {\n      this.onAfterDelete(id, deleted, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return deleted || null;\n  }\n\n  /**\n   * Set an entity as the default for a project/organization scope.\n   * Marks all other entities as non-default, then sets the specified entity.\n   * Uses a transaction to ensure atomicity.\n   */\n  async setDefault(\n    project: string,\n    org: string,\n    id: string,\n    userId: string,\n  ): Promise<TEntity> {\n    return db.transaction(async (tx) => {\n      const orgColumn = this.getOrgColumn();\n      const projectColumn = this.getProjectColumn();\n\n      // Build scoping conditions for clearing defaults\n      const scopeConditions = [\n        eq(orgColumn, org),\n        eq((this.schema as any).isDefault, true),\n      ];\n      if (projectColumn) {\n        scopeConditions.push(eq(projectColumn, project));\n      }\n\n      // Lock existing defaults with FOR UPDATE to prevent concurrent setDefault races\n      await tx.execute(\n        sql`SELECT id FROM ${this.schema}\n            WHERE ${orgColumn} = ${org}\n              AND ${(this.schema as any).isDefault} = true\n            ${projectColumn ? sql`AND ${projectColumn} = ${project}` : sql``}\n            FOR UPDATE`,\n      );\n\n      // Mark all entities in scope as non-default\n      await tx\n        .update(this.schema)\n        .set({\n          isDefault: false,\n          updatedAt: new Date(),\n          updatedBy: userId || 'system',\n        } as any)\n        .where(and(...scopeConditions));\n\n      // Set the specified entity as default\n      const [updated] = await tx\n        .update(this.schema)\n        .set({\n          isDefault: true,\n          updatedAt: new Date(),\n          updatedBy: userId || 'system',\n        } as any)\n        .where(\n          and(\n            eq((this.schema as any).id, id),\n            eq(orgColumn, org),\n          ),\n        )\n        .returning().then(r => drizzleRows<TEntity>(r));\n\n      if (!updated) {\n        throw new NotFoundError(`Entity with id ${id} not found`);\n      }\n\n      return updated;\n    });\n  }\n\n  /**\n   * Update multiple entities matching filter\n   */\n  async updateMany(\n    filter: Partial<TFilter>,\n    data: Partial<TUpdate>,\n    orgId: string,\n    userId: string,\n  ): Promise<TEntity[]> {\n    const conditions = this.buildConditions(filter, orgId);\n\n    return db\n      .update(this.schema)\n      .set({\n        ...data,\n        updatedAt: new Date(),\n        updatedBy: userId || 'system',\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n  }\n\n  /**\n   * Create multiple entities in a single batch insert.\n   * Uses upsert (onConflictDoUpdate) — all rows are inserted in one query per chunk.\n   * Chunks of 100 to stay within PostgreSQL parameter limits.\n   */\n  async bulkCreate(items: TInsert[], userId: string): Promise<TEntity[]> {\n    if (items.length === 0) return [];\n\n    const CHUNK_SIZE = 100;\n    const now = new Date();\n    const user = userId || 'system';\n\n    const results = await db.transaction(async (tx) => {\n      const allCreated: TEntity[] = [];\n\n      for (let i = 0; i < items.length; i += CHUNK_SIZE) {\n        const chunk = items.slice(i, i + CHUNK_SIZE);\n        const values = chunk.map(data => ({\n          ...data,\n          createdBy: user,\n          updatedBy: user,\n        } as any));\n\n        const created = await tx\n          .insert(this.schema)\n          .values(values)\n          .onConflictDoUpdate({\n            target: this.conflictTarget as any,\n            set: {\n              updatedAt: now,\n              updatedBy: user,\n            } as any,\n          })\n          .returning().then(r => drizzleRows<TEntity>(r));\n\n        allCreated.push(...created);\n      }\n\n      return allCreated;\n    });\n\n    for (const entity of results) {\n      this.onAfterCreate(entity, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return results;\n  }\n\n  /**\n   * Soft-delete multiple entities by IDs in a single batch operation.\n   */\n  async bulkDelete(\n    ids: string[],\n    orgId: string,\n    userId: string,\n  ): Promise<TEntity[]> {\n    if (ids.length === 0) return [];\n\n    const now = new Date();\n    const user = userId || 'system';\n    const conditions = [\n      inArray((this.schema as any).id, ids),\n      ...this.buildConditions({} as Partial<TFilter>, orgId),\n    ];\n\n    const deleted = await db\n      .update(this.schema)\n      .set({\n        isActive: false,\n        updatedAt: now,\n        updatedBy: user,\n        deletedAt: now,\n        deletedBy: user,\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    for (const entity of deleted) {\n      this.onAfterDelete(entity.id, entity, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return deleted;\n  }\n}\n"]}
394
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crud-service.js","sourceRoot":"","sources":["../../src/api/crud-service.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC;;;AAkBtC,kCAEC;AAGD,oCAEC;AAvBD,yDAAyE;AACzE,6CAAoE;AAGpE,yEAAqD;AAErD,mFAAmF;AACnF,MAAM,kBAAkB,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,IAAI,KAAK,EAAE,EAAE,CAAC,CAAC;AACjF,MAAM,cAAc,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,MAAM,EAAE,EAAE,CAAC,CAAC;AAE1E;;;;;GAKG;AACH,SAAgB,WAAW,CAAI,IAAa;IAC1C,OAAO,IAAW,CAAC;AACrB,CAAC;AAED,wEAAwE;AACxE,SAAgB,YAAY,CAAC,IAAa;IACxC,OAAO,IAA2B,CAAC;AACrC,CAAC;AA8CD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCG;AACH,MAAsB,WAAW;IAwBd,OAAO,GAAG,IAAA,uBAAY,EAAC,aAAa,CAAC,CAAC;IAEvD,kDAAkD;IAC1C,YAAY,CAAC,EAAU,EAAE,KAAc;QAC7C,OAAO,IAAI,CAAC,eAAe,CAAC,EAAE,EAAE,EAAiC,EAAE,KAAK,CAAC,CAAC;IAC5E,CAAC;IAED,iEAAiE;IACjE,2EAA2E;IAE3E,2CAA2C;IACjC,KAAK,CAAC,aAAa,CAAC,OAAgB,EAAE,OAAe,IAAkB,CAAC;IAElF,wCAAwC;IAC9B,KAAK,CAAC,aAAa,CAAC,GAAW,EAAE,OAAgB,EAAE,OAAe,IAAkB,CAAC;IAE/F,6CAA6C;IACnC,KAAK,CAAC,aAAa,CAAC,GAAW,EAAE,OAAgB,EAAE,OAAe,IAAkB,CAAC;IAE/F;;;;;OAKG;IACH,KAAK,CAAC,IAAI,CAAC,MAAwB,EAAE,KAAc;QACjD,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAEvD,OAAO,wBAAE;aACN,MAAM,EAAE;aACR,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC;aACjB,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,aAAa,CACjB,MAAwB,EACxB,KAAc,EACd,UAAwB,EAAE;QAE1B,MAAM,EAAE,KAAK,EAAE,QAAQ,GAAG,kBAAkB,EAAE,MAAM,GAAG,CAAC,EAAE,MAAM,EAAE,SAAS,GAAG,KAAK,EAAE,YAAY,GAAG,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC;QACtI,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,CAAC,EAAE,cAAc,CAAC,CAAC;QAE9D,qEAAqE;QACrE,MAAM,SAAS,GAAG,CAAC,CAAC,CAAC,MAAM,IAAI,MAAM,CAAC,CAAC;QAEvC,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAEvD,kEAAkE;QAClE,IAAI,SAAS,EAAE,CAAC;YACd,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;YAC9C,IAAI,UAAU,EAAE,CAAC;gBACf,MAAM,EAAE,GAAG,SAAS,KAAK,MAAM;oBAC7B,CAAC,CAAC,IAAA,iBAAG,EAAA,GAAG,UAAU,MAAM,MAAM,EAAE;oBAChC,CAAC,CAAC,IAAA,iBAAG,EAAA,GAAG,UAAU,MAAM,MAAM,EAAE,CAAC;gBACnC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACtB,CAAC;QACH,CAAC;QAED,2DAA2D;QAC3D,MAAM,UAAU,GAAG,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QACtE,IAAI,KAAK,GAAG,UAAU;YACpB,CAAC,CAAC,wBAAE,CAAC,MAAM,CAAC,UAAiB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC;YAC1E,CAAC,CAAC,wBAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC,CAAC;QAE5D,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;YAC9C,IAAI,UAAU,EAAE,CAAC;gBACf,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,SAAS,KAAK,MAAM,CAAC,CAAC,CAAC,IAAA,kBAAI,EAAC,UAAU,CAAC,CAAC,CAAC,CAAC,IAAA,iBAAG,EAAC,UAAU,CAAC,CAAQ,CAAC;YAC1F,CAAC;QACH,CAAC;QAED,mDAAmD;QACnD,MAAM,eAAe,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;QAC/C,MAAM,IAAI,GAAG,MAAM,KAAK;aACrB,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC;aAChB,MAAM,CAAC,eAAe,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;QAE9D,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QACpC,MAAM,IAAI,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAEnD,MAAM,MAAM,GAA6B,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,EAAE,eAAe,EAAE,OAAO,EAAE,CAAC;QAE3F,yDAAyD;QACzD,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,MAAM,EAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAA4B,CAAC;YAClE,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC;YACrC,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;gBAC9B,MAAM,CAAC,UAAU,GAAG,WAAW,YAAY,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;YACpG,CAAC;QACH,CAAC;QAED,yEAAyE;QACzE,IAAI,YAAY,EAAE,CAAC;YACjB,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;YAC3D,MAAM,CAAC,WAAW,CAAC,GAAG,MAAM,wBAAE;iBAC3B,MAAM,CAAC,EAAE,KAAK,EAAE,IAAA,iBAAG,EAAQ,eAAe,EAAE,CAAC;iBAC7C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC;iBACjB,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,cAAc,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;YAC5D,MAAM,CAAC,KAAK,GAAG,WAAW,EAAE,KAAK,IAAI,CAAC,CAAC;QACzC,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;OAGG;IACK,gBAAgB,CAAC,MAAgB;QACvC,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,SAAS,CAAC;QAE1C,MAAM,OAAO,GAA4B,EAAE,CAAC;QAC5C,wCAAwC;QACxC,OAAO,CAAC,EAAE,GAAI,IAAI,CAAC,MAAc,CAAC,EAAE,CAAC;QAErC,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,IAAI,KAAK,KAAK,IAAI;gBAAE,SAAS,CAAC,mBAAmB;YACjD,MAAM,GAAG,GAAI,IAAI,CAAC,MAAc,CAAC,KAAK,CAAC,CAAC;YACxC,IAAI,GAAG;gBAAE,OAAO,CAAC,KAAK,CAAC,GAAG,GAAG,CAAC;QAChC,CAAC;QAED,+CAA+C;QAC/C,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,KAAK,CAAC,MAAwB,EAAE,KAAc;QAClD,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAEvD,MAAM,CAAC,MAAM,CAAC,GAAG,MAAM,wBAAE;aACtB,MAAM,CAAC,EAAE,KAAK,EAAE,IAAA,iBAAG,EAAQ,eAAe,EAAE,CAAC;aAC7C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC;aACjB,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;QAExD,OAAO,MAAM,EAAE,KAAK,IAAI,CAAC,CAAC;IAC5B,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,QAAQ,CAAC,EAAU,EAAE,KAAc;QACvC,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QAEhD,MAAM,OAAO,GAAG,MAAM,wBAAE;aACrB,MAAM,EAAE;aACR,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC;aACjB,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC;aACzB,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;QAE/C,OAAO,OAAO,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;IAC5B,CAAC;IAED,sBAAsB;IAEtB;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,IAAa,EAAE,MAAc;QACxC,MAAM,CAAC,OAAO,CAAC,GAAG,MAAM,wBAAE;aACvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;aACnB,MAAM,CAAC;YACN,GAAG,IAAI;YACP,SAAS,EAAE,MAAM,IAAI,QAAQ;YAC7B,SAAS,EAAE,MAAM,IAAI,QAAQ;SACvB,CAAC;aACR,kBAAkB,CAAC;YAClB,MAAM,EAAE,IAAI,CAAC,cAAqB;YAClC,GAAG,EAAE;gBACH,GAAG,IAAI;gBACP,SAAS,EAAE,IAAI,IAAI,EAAE;gBACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;aACvB;SACT,CAAC;aACD,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;QAElD,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QAErH,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CACV,EAAU,EACV,IAAsB,EACtB,KAAa,EACb,MAAc;QAEd,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QAEhD,MAAM,CAAC,OAAO,CAAC,GAAG,MAAM,wBAAE;aACvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;aACnB,GAAG,CAAC;YACH,GAAG,IAAI;YACP,SAAS,EAAE,IAAI,IAAI,EAAE;YACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;SACvB,CAAC;aACR,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC;aACzB,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;QAElD,IAAI,OAAO,EAAE,CAAC;YACZ,IAAI,CAAC,aAAa,CAAC,EAAE,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3H,CAAC;QAED,OAAO,OAAO,IAAI,IAAI,CAAC;IACzB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU,EAAE,KAAa,EAAE,MAAc;QACpD,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QAEhD,MAAM,CAAC,OAAO,CAAC,GAAG,MAAM,wBAAE;aACvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;aACnB,GAAG,CAAC;YACH,QAAQ,EAAE,KAAK;YACf,SAAS,EAAE,IAAI,IAAI,EAAE;YACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;YAC7B,SAAS,EAAE,IAAI,IAAI,EAAE;YACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;SACvB,CAAC;aACR,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC;aACzB,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;QAElD,IAAI,OAAO,EAAE,CAAC;YACZ,IAAI,CAAC,aAAa,CAAC,EAAE,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3H,CAAC;QAED,OAAO,OAAO,IAAI,IAAI,CAAC;IACzB,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,UAAU,CACd,OAAe,EACf,GAAW,EACX,EAAU,EACV,MAAc;QAEd,OAAO,wBAAE,CAAC,WAAW,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YACjC,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACtC,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAE9C,iDAAiD;YACjD,MAAM,eAAe,GAAG;gBACtB,IAAA,gBAAE,EAAC,SAAS,EAAE,GAAG,CAAC;gBAClB,IAAA,gBAAE,EAAE,IAAI,CAAC,MAAc,CAAC,SAAS,EAAE,IAAI,CAAC;aACzC,CAAC;YACF,IAAI,aAAa,EAAE,CAAC;gBAClB,eAAe,CAAC,IAAI,CAAC,IAAA,gBAAE,EAAC,aAAa,EAAE,OAAO,CAAC,CAAC,CAAC;YACnD,CAAC;YAED,gFAAgF;YAChF,MAAM,EAAE,CAAC,OAAO,CACd,IAAA,iBAAG,EAAA,kBAAkB,IAAI,CAAC,MAAM;oBACpB,SAAS,MAAM,GAAG;oBACjB,IAAI,CAAC,MAAc,CAAC,SAAS;cACpC,aAAa,CAAC,CAAC,CAAC,IAAA,iBAAG,EAAA,OAAO,aAAa,MAAM,OAAO,EAAE,CAAC,CAAC,CAAC,IAAA,iBAAG,EAAA,EAAE;uBACrD,CAChB,CAAC;YAEF,4CAA4C;YAC5C,MAAM,EAAE;iBACL,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;iBACnB,GAAG,CAAC;gBACH,SAAS,EAAE,KAAK;gBAChB,SAAS,EAAE,IAAI,IAAI,EAAE;gBACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;aACvB,CAAC;iBACR,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,eAAe,CAAC,CAAC,CAAC;YAElC,sCAAsC;YACtC,MAAM,CAAC,OAAO,CAAC,GAAG,MAAM,EAAE;iBACvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;iBACnB,GAAG,CAAC;gBACH,SAAS,EAAE,IAAI;gBACf,SAAS,EAAE,IAAI,IAAI,EAAE;gBACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;aACvB,CAAC;iBACR,KAAK,CACJ,IAAA,iBAAG,EACD,IAAA,gBAAE,EAAE,IAAI,CAAC,MAAc,CAAC,EAAE,EAAE,EAAE,CAAC,EAC/B,IAAA,gBAAE,EAAC,SAAS,EAAE,GAAG,CAAC,CACnB,CACF;iBACA,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;YAElD,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,MAAM,IAAI,wBAAa,CAAC,kBAAkB,EAAE,YAAY,CAAC,CAAC;YAC5D,CAAC;YAED,OAAO,OAAO,CAAC;QACjB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,UAAU,CACd,MAAwB,EACxB,IAAsB,EACtB,KAAa,EACb,MAAc;QAEd,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAEvD,OAAO,wBAAE;aACN,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;aACnB,GAAG,CAAC;YACH,GAAG,IAAI;YACP,SAAS,EAAE,IAAI,IAAI,EAAE;YACrB,SAAS,EAAE,MAAM,IAAI,QAAQ;SACvB,CAAC;aACR,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC;aACzB,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;IACpD,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,UAAU,CAAC,KAAgB,EAAE,MAAc;QAC/C,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,EAAE,CAAC;QAElC,MAAM,UAAU,GAAG,GAAG,CAAC;QACvB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,IAAI,GAAG,MAAM,IAAI,QAAQ,CAAC;QAEhC,MAAM,OAAO,GAAG,MAAM,wBAAE,CAAC,WAAW,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YAChD,MAAM,UAAU,GAAc,EAAE,CAAC;YAEjC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,IAAI,UAAU,EAAE,CAAC;gBAClD,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,CAAC;gBAC7C,MAAM,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;oBAChC,GAAG,IAAI;oBACP,SAAS,EAAE,IAAI;oBACf,SAAS,EAAE,IAAI;iBACR,CAAA,CAAC,CAAC;gBAEX,MAAM,OAAO,GAAG,MAAM,EAAE;qBACrB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;qBACnB,MAAM,CAAC,MAAM,CAAC;qBACd,kBAAkB,CAAC;oBAClB,MAAM,EAAE,IAAI,CAAC,cAAqB;oBAClC,GAAG,EAAE;wBACH,SAAS,EAAE,GAAG;wBACd,SAAS,EAAE,IAAI;qBACT;iBACT,CAAC;qBACD,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;gBAElD,UAAU,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,CAAC;YAC9B,CAAC;YAED,OAAO,UAAU,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QACtH,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,UAAU,CACd,GAAa,EACb,KAAa,EACb,MAAc;QAEd,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,EAAE,CAAC;QAEhC,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,IAAI,GAAG,MAAM,IAAI,QAAQ,CAAC;QAChC,MAAM,UAAU,GAAG;YACjB,IAAA,qBAAO,EAAE,IAAI,CAAC,MAAc,CAAC,EAAE,EAAE,GAAG,CAAC;YACrC,GAAG,IAAI,CAAC,eAAe,CAAC,EAAsB,EAAE,KAAK,CAAC;SACvD,CAAC;QAEF,MAAM,OAAO,GAAG,MAAM,wBAAE;aACrB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;aACnB,GAAG,CAAC;YACH,QAAQ,EAAE,KAAK;YACf,SAAS,EAAE,GAAG;YACd,SAAS,EAAE,IAAI;YACf,SAAS,EAAE,GAAG;YACd,SAAS,EAAE,IAAI;SACT,CAAC;aACR,KAAK,CAAC,IAAA,iBAAG,EAAC,GAAG,UAAU,CAAC,CAAC;aACzB,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,WAAW,CAAU,CAAC,CAAC,CAAC,CAAC;QAElD,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QACjI,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;CACF;AA5bD,kCA4bC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { NotFoundError, createLogger } from '@pipeline-builder/api-core';\nimport { SQL, eq, and, asc, desc, sql, inArray } from 'drizzle-orm';\nimport type { AnyColumn } from 'drizzle-orm/column';\nimport type { PgTable } from 'drizzle-orm/pg-core';\nimport { db } from '../database/postgres-connection';\n\n/** Pagination defaults — read from env to match CoreConstants in pipeline-core. */\nconst DEFAULT_PAGE_LIMIT = parseInt(process.env.DEFAULT_PAGE_LIMIT || '100', 10);\nconst MAX_PAGE_LIMIT = parseInt(process.env.MAX_PAGE_LIMIT || '1000', 10);\n\n/**\n * Cast Drizzle query results to a typed array.\n * Drizzle's generic return type (`PgSelectBase<...>`) doesn't narrow to our\n * entity generics, so an explicit cast is needed. Centralised here so every\n * call-site stays one-liner clean and the cast is documented in one place.\n */\nexport function drizzleRows<T>(rows: unknown): T[] {\n  return rows as T[];\n}\n\n/** Cast a Drizzle aggregate result to extract `[{ count: number }]`. */\nexport function drizzleCount(rows: unknown): [{ count: number }] {\n  return rows as [{ count: number }];\n}\n\n/**\n * Base interface for entities with common fields\n */\nexport interface BaseEntity {\n  id: string;\n  orgId: string;\n  isDefault: boolean;\n  createdAt: Date;\n  updatedAt: Date;\n  createdBy: string;\n  updatedBy: string;\n  [key: string]: unknown;\n}\n\n/**\n * Pagination and sorting options\n */\nexport interface QueryOptions {\n  limit?: number;\n  offset?: number;\n  sortBy?: string;\n  sortOrder?: 'asc' | 'desc';\n  /** When true, runs a separate COUNT(*) query to include exact total. Default: false. */\n  includeTotal?: boolean;\n  /** Cursor-based pagination: fetch rows after this cursor value (uses sortBy column). */\n  cursor?: string;\n  /** Sparse fieldset: column names to select. Returns all columns when omitted. */\n  fields?: string[];\n}\n\n/**\n * Paginated result with metadata\n */\nexport interface PaginatedResult<T> {\n  data: T[];\n  /** Total count of matching entities. Only present when includeTotal is true. */\n  total?: number;\n  limit: number;\n  offset: number;\n  hasMore: boolean;\n  /** Cursor pointing to the last item, for cursor-based pagination. */\n  nextCursor?: string;\n}\n\n/**\n * Abstract CRUD service with multi-tenant access control and pagination.\n *\n * Subclasses bind to a specific Drizzle table by implementing `schema`,\n * `buildConditions`, `getSortColumn`, and the org/project-column accessors.\n *\n * @typeParam TEntity - Entity type extending BaseEntity\n * @typeParam TFilter - Filter type for query parameters\n * @typeParam TInsert - Insert DTO type\n * @typeParam TUpdate - Update DTO type\n *\n * **A note on the type casts.** Drizzle's row types are inferred from\n * `pgTable(...)` and don't generically narrow through the abstract `schema`\n * getter, so the base class casts query results to `TEntity` and back. The\n * cast is *not* a runtime safety guarantee — it relies on each subclass\n * passing the matching entity type. Org-scoping is enforced by every\n * subclass's `buildConditions` injecting `WHERE org_id = $1`; this class\n * does not add that filter itself.\n *\n * **Error policy.** Errors propagate up to the route-level handler\n * (`withRoute`); no catch-and-swallow here.\n *\n * @example\n * ```typescript\n * class PipelineService extends CrudService<Pipeline, PipelineFilter, PipelineInsert, PipelineUpdate> {\n *   protected get schema() { return schema.pipeline; }\n *   protected buildConditions(filter, orgId) { return buildPipelineConditions(filter, orgId); }\n *   protected getSortColumn(sortBy) { return sortColumnMap[sortBy] ?? null; }\n *   protected getProjectColumn() { return schema.pipeline.project; }\n *   protected getOrgColumn() { return schema.pipeline.organization; }\n * }\n * ```\n */\nexport abstract class CrudService<\n  TEntity extends BaseEntity,\n  TFilter,\n  TInsert,\n  TUpdate,\n> {\n  /** Drizzle schema table for this entity */\n  protected abstract get schema(): PgTable;\n\n  /** Build SQL conditions for filtering entities */\n  protected abstract buildConditions(filter: Partial<TFilter>, orgId?: string): SQL[];\n\n  /** Get the schema column for sorting by field name */\n  protected abstract getSortColumn(sortBy: string): AnyColumn | null;\n\n  /** Get the project column for setDefault scoping (null if entity has no project scope) */\n  protected abstract getProjectColumn(): AnyColumn | null;\n\n  /** Get the organization column for setDefault scoping */\n  protected abstract getOrgColumn(): AnyColumn;\n\n  /** Get the unique constraint columns for onConflictDoUpdate */\n  protected abstract get conflictTarget(): AnyColumn[];\n\n  private readonly _logger = createLogger('CrudService');\n\n  /** Build conditions for a single entity by ID. */\n  private idConditions(id: string, orgId?: string): SQL[] {\n    return this.buildConditions({ id } as unknown as Partial<TFilter>, orgId);\n  }\n\n  // Lifecycle hooks — override in subclasses to react to mutations\n  // These are fire-and-forget: errors are logged but never block the caller.\n\n  /** Called after a new entity is created */\n  protected async onAfterCreate(_entity: TEntity, _userId: string): Promise<void> {}\n\n  /** Called after an entity is updated */\n  protected async onAfterUpdate(_id: string, _entity: TEntity, _userId: string): Promise<void> {}\n\n  /** Called after an entity is soft-deleted */\n  protected async onAfterDelete(_id: string, _entity: TEntity, _userId: string): Promise<void> {}\n\n  /**\n   * Find entities matching filter criteria\n   *\n   * @param filter - Filter criteria\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   */\n  async find(filter: Partial<TFilter>, orgId?: string): Promise<TEntity[]> {\n    const conditions = this.buildConditions(filter, orgId);\n\n    return db\n      .select()\n      .from(this.schema)\n      .where(and(...conditions)).then(r => drizzleRows<TEntity>(r));\n  }\n\n  /**\n   * Find entities with pagination and sorting\n   *\n   * @param filter - Filter criteria\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   * @param options - Pagination and sorting options\n   */\n  async findPaginated(\n    filter: Partial<TFilter>,\n    orgId?: string,\n    options: QueryOptions = {},\n  ): Promise<PaginatedResult<TEntity>> {\n    const { limit: rawLimit = DEFAULT_PAGE_LIMIT, offset = 0, sortBy, sortOrder = 'asc', includeTotal = false, cursor, fields } = options;\n    const limit = Math.min(Math.max(1, rawLimit), MAX_PAGE_LIMIT);\n\n    // Cursor and offset are mutually exclusive — cursor takes precedence\n    const useCursor = !!(cursor && sortBy);\n\n    const conditions = this.buildConditions(filter, orgId);\n\n    // Cursor-based pagination: add WHERE clause for keyset pagination\n    if (useCursor) {\n      const sortColumn = this.getSortColumn(sortBy);\n      if (sortColumn) {\n        const op = sortOrder === 'desc'\n          ? sql`${sortColumn} < ${cursor}`\n          : sql`${sortColumn} > ${cursor}`;\n        conditions.push(op);\n      }\n    }\n\n    // Build SELECT — sparse fieldset when fields are specified\n    const selectSpec = fields ? this.buildFieldSelect(fields) : undefined;\n    let query = selectSpec\n      ? db.select(selectSpec as any).from(this.schema).where(and(...conditions))\n      : db.select().from(this.schema).where(and(...conditions));\n\n    if (sortBy) {\n      const sortColumn = this.getSortColumn(sortBy);\n      if (sortColumn) {\n        query = query.orderBy(sortOrder === 'desc' ? desc(sortColumn) : asc(sortColumn)) as any;\n      }\n    }\n\n    // Fetch limit+1 to detect hasMore without COUNT(*)\n    const effectiveOffset = useCursor ? 0 : offset;\n    const rows = await query\n      .limit(limit + 1)\n      .offset(effectiveOffset).then(r => drizzleRows<TEntity>(r));\n\n    const hasMore = rows.length > limit;\n    const data = hasMore ? rows.slice(0, limit) : rows;\n\n    const result: PaginatedResult<TEntity> = { data, limit, offset: effectiveOffset, hasMore };\n\n    // Provide next cursor from last item's sort column value\n    if (data.length > 0 && sortBy) {\n      const lastItem = data[data.length - 1] as Record<string, unknown>;\n      const cursorValue = lastItem[sortBy];\n      if (cursorValue !== undefined) {\n        result.nextCursor = cursorValue instanceof Date ? cursorValue.toISOString() : String(cursorValue);\n      }\n    }\n\n    // Only run the COUNT(*) query when the caller explicitly needs the total\n    if (includeTotal) {\n      const baseConditions = this.buildConditions(filter, orgId);\n      const [countResult] = await db\n        .select({ count: sql<number>`count(*)::int` })\n        .from(this.schema)\n        .where(and(...baseConditions)).then(r => drizzleCount(r));\n      result.total = countResult?.count || 0;\n    }\n\n    return result;\n  }\n\n  /**\n   * Build a column selection map for sparse fieldsets.\n   * Falls back to full select if no matching columns found.\n   */\n  private buildFieldSelect(fields: string[]): Record<string, unknown> | undefined {\n    if (fields.length === 0) return undefined;\n\n    const columns: Record<string, unknown> = {};\n    // Always include id for entity identity\n    columns.id = (this.schema as any).id;\n\n    for (const field of fields) {\n      if (field === 'id') continue; // Already included\n      const col = (this.schema as any)[field];\n      if (col) columns[field] = col;\n    }\n\n    // At minimum we'll have { id }, which is valid\n    return columns;\n  }\n\n  /**\n   * Count entities matching filter criteria\n   *\n   * @param filter - Filter criteria\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   */\n  async count(filter: Partial<TFilter>, orgId?: string): Promise<number> {\n    const conditions = this.buildConditions(filter, orgId);\n\n    const [result] = await db\n      .select({ count: sql<number>`count(*)::int` })\n      .from(this.schema)\n      .where(and(...conditions)).then(r => drizzleCount(r));\n\n    return result?.count || 0;\n  }\n\n  /**\n   * Find a single entity by ID\n   *\n   * @param id - Entity ID\n   * @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)\n   */\n  async findById(id: string, orgId?: string): Promise<TEntity | null> {\n    const conditions = this.idConditions(id, orgId);\n\n    const results = await db\n      .select()\n      .from(this.schema)\n      .where(and(...conditions))\n      .limit(1).then(r => drizzleRows<TEntity>(r));\n\n    return results[0] || null;\n  }\n\n  // Mutation operations\n\n  /**\n   * Create a new entity\n   */\n  async create(data: TInsert, userId: string): Promise<TEntity> {\n    const [created] = await db\n      .insert(this.schema)\n      .values({\n        ...data,\n        createdBy: userId || 'system',\n        updatedBy: userId || 'system',\n      } as any)\n      .onConflictDoUpdate({\n        target: this.conflictTarget as any,\n        set: {\n          ...data,\n          updatedAt: new Date(),\n          updatedBy: userId || 'system',\n        } as any,\n      })\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    this.onAfterCreate(created, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n\n    return created;\n  }\n\n  /**\n   * Update an existing entity\n   */\n  async update(\n    id: string,\n    data: Partial<TUpdate>,\n    orgId: string,\n    userId: string,\n  ): Promise<TEntity | null> {\n    const conditions = this.idConditions(id, orgId);\n\n    const [updated] = await db\n      .update(this.schema)\n      .set({\n        ...data,\n        updatedAt: new Date(),\n        updatedBy: userId || 'system',\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    if (updated) {\n      this.onAfterUpdate(id, updated, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return updated || null;\n  }\n\n  /**\n   * Delete an entity (soft delete by setting isActive = false)\n   */\n  async delete(id: string, orgId: string, userId: string): Promise<TEntity | null> {\n    const conditions = this.idConditions(id, orgId);\n\n    const [deleted] = await db\n      .update(this.schema)\n      .set({\n        isActive: false,\n        updatedAt: new Date(),\n        updatedBy: userId || 'system',\n        deletedAt: new Date(),\n        deletedBy: userId || 'system',\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    if (deleted) {\n      this.onAfterDelete(id, deleted, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return deleted || null;\n  }\n\n  /**\n   * Set an entity as the default for a project/organization scope.\n   * Marks all other entities as non-default, then sets the specified entity.\n   * Uses a transaction to ensure atomicity.\n   */\n  async setDefault(\n    project: string,\n    org: string,\n    id: string,\n    userId: string,\n  ): Promise<TEntity> {\n    return db.transaction(async (tx) => {\n      const orgColumn = this.getOrgColumn();\n      const projectColumn = this.getProjectColumn();\n\n      // Build scoping conditions for clearing defaults\n      const scopeConditions = [\n        eq(orgColumn, org),\n        eq((this.schema as any).isDefault, true),\n      ];\n      if (projectColumn) {\n        scopeConditions.push(eq(projectColumn, project));\n      }\n\n      // Lock existing defaults with FOR UPDATE to prevent concurrent setDefault races\n      await tx.execute(\n        sql`SELECT id FROM ${this.schema}\n            WHERE ${orgColumn} = ${org}\n              AND ${(this.schema as any).isDefault} = true\n            ${projectColumn ? sql`AND ${projectColumn} = ${project}` : sql``}\n            FOR UPDATE`,\n      );\n\n      // Mark all entities in scope as non-default\n      await tx\n        .update(this.schema)\n        .set({\n          isDefault: false,\n          updatedAt: new Date(),\n          updatedBy: userId || 'system',\n        } as any)\n        .where(and(...scopeConditions));\n\n      // Set the specified entity as default\n      const [updated] = await tx\n        .update(this.schema)\n        .set({\n          isDefault: true,\n          updatedAt: new Date(),\n          updatedBy: userId || 'system',\n        } as any)\n        .where(\n          and(\n            eq((this.schema as any).id, id),\n            eq(orgColumn, org),\n          ),\n        )\n        .returning().then(r => drizzleRows<TEntity>(r));\n\n      if (!updated) {\n        throw new NotFoundError(`Entity with id ${id} not found`);\n      }\n\n      return updated;\n    });\n  }\n\n  /**\n   * Update multiple entities matching filter\n   */\n  async updateMany(\n    filter: Partial<TFilter>,\n    data: Partial<TUpdate>,\n    orgId: string,\n    userId: string,\n  ): Promise<TEntity[]> {\n    const conditions = this.buildConditions(filter, orgId);\n\n    return db\n      .update(this.schema)\n      .set({\n        ...data,\n        updatedAt: new Date(),\n        updatedBy: userId || 'system',\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n  }\n\n  /**\n   * Create multiple entities in a single batch insert.\n   * Uses upsert (onConflictDoUpdate) — all rows are inserted in one query per chunk.\n   * Chunks of 100 to stay within PostgreSQL parameter limits.\n   */\n  async bulkCreate(items: TInsert[], userId: string): Promise<TEntity[]> {\n    if (items.length === 0) return [];\n\n    const CHUNK_SIZE = 100;\n    const now = new Date();\n    const user = userId || 'system';\n\n    const results = await db.transaction(async (tx) => {\n      const allCreated: TEntity[] = [];\n\n      for (let i = 0; i < items.length; i += CHUNK_SIZE) {\n        const chunk = items.slice(i, i + CHUNK_SIZE);\n        const values = chunk.map(data => ({\n          ...data,\n          createdBy: user,\n          updatedBy: user,\n        } as any));\n\n        const created = await tx\n          .insert(this.schema)\n          .values(values)\n          .onConflictDoUpdate({\n            target: this.conflictTarget as any,\n            set: {\n              updatedAt: now,\n              updatedBy: user,\n            } as any,\n          })\n          .returning().then(r => drizzleRows<TEntity>(r));\n\n        allCreated.push(...created);\n      }\n\n      return allCreated;\n    });\n\n    for (const entity of results) {\n      this.onAfterCreate(entity, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return results;\n  }\n\n  /**\n   * Soft-delete multiple entities by IDs in a single batch operation.\n   */\n  async bulkDelete(\n    ids: string[],\n    orgId: string,\n    userId: string,\n  ): Promise<TEntity[]> {\n    if (ids.length === 0) return [];\n\n    const now = new Date();\n    const user = userId || 'system';\n    const conditions = [\n      inArray((this.schema as any).id, ids),\n      ...this.buildConditions({} as Partial<TFilter>, orgId),\n    ];\n\n    const deleted = await db\n      .update(this.schema)\n      .set({\n        isActive: false,\n        updatedAt: now,\n        updatedBy: user,\n        deletedAt: now,\n        deletedBy: user,\n      } as any)\n      .where(and(...conditions))\n      .returning().then(r => drizzleRows<TEntity>(r));\n\n    for (const entity of deleted) {\n      this.onAfterDelete(entity.id, entity, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));\n    }\n\n    return deleted;\n  }\n}\n"]}
package/package.json CHANGED
@@ -22,7 +22,7 @@
22
22
  "typescript": "5.9.3"
23
23
  },
24
24
  "dependencies": {
25
- "@pipeline-builder/api-core": "3.4.17",
25
+ "@pipeline-builder/api-core": "3.4.18",
26
26
  "drizzle-orm": "0.45.1",
27
27
  "pg": "8.18.0"
28
28
  },
@@ -64,7 +64,7 @@
64
64
  "main": "lib/index.js",
65
65
  "license": "Apache-2.0",
66
66
  "homepage": "https://mwashburn160.github.io/pipeline-builder/",
67
- "version": "3.4.18",
67
+ "version": "3.4.19",
68
68
  "bugs": {
69
69
  "url": "https://github.com/mwashburn160/pipeline-builder/issues"
70
70
  },