@pipeline-builder/pipeline-data 3.1.0
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/LICENSE +202 -0
- package/README.md +34 -0
- package/drizzle.config.ts +17 -0
- package/lib/api/access-control-builder.d.ts +109 -0
- package/lib/api/access-control-builder.js +181 -0
- package/lib/api/crud-service.d.ts +170 -0
- package/lib/api/crud-service.js +387 -0
- package/lib/api/query-builders.d.ts +74 -0
- package/lib/api/query-builders.js +336 -0
- package/lib/api/reporting-service.d.ts +131 -0
- package/lib/api/reporting-service.js +248 -0
- package/lib/core/query-filters.d.ts +235 -0
- package/lib/core/query-filters.js +23 -0
- package/lib/database/drizzle-schema.d.ts +10043 -0
- package/lib/database/drizzle-schema.js +715 -0
- package/lib/database/index.d.ts +3 -0
- package/lib/database/index.js +22 -0
- package/lib/database/postgres-connection.d.ts +232 -0
- package/lib/database/postgres-connection.js +456 -0
- package/lib/database/retry-strategy.d.ts +68 -0
- package/lib/database/retry-strategy.js +126 -0
- package/lib/index.d.ts +30 -0
- package/lib/index.js +52 -0
- package/package.json +125 -0
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import { SQL } from 'drizzle-orm';
|
|
2
|
+
import type { AnyColumn } from 'drizzle-orm/column';
|
|
3
|
+
import type { PgTable } from 'drizzle-orm/pg-core';
|
|
4
|
+
/**
|
|
5
|
+
* Cast Drizzle query results to a typed array.
|
|
6
|
+
* Drizzle's generic return type (`PgSelectBase<...>`) doesn't narrow to our
|
|
7
|
+
* entity generics, so an explicit cast is needed. Centralised here so every
|
|
8
|
+
* call-site stays one-liner clean and the cast is documented in one place.
|
|
9
|
+
*/
|
|
10
|
+
export declare function drizzleRows<T>(rows: unknown): T[];
|
|
11
|
+
/** Cast a Drizzle aggregate result to extract `[{ count: number }]`. */
|
|
12
|
+
export declare function drizzleCount(rows: unknown): [{
|
|
13
|
+
count: number;
|
|
14
|
+
}];
|
|
15
|
+
/**
|
|
16
|
+
* Base interface for entities with common fields
|
|
17
|
+
*/
|
|
18
|
+
export interface BaseEntity {
|
|
19
|
+
id: string;
|
|
20
|
+
orgId: string;
|
|
21
|
+
isDefault: boolean;
|
|
22
|
+
createdAt: Date;
|
|
23
|
+
updatedAt: Date;
|
|
24
|
+
createdBy: string;
|
|
25
|
+
updatedBy: string;
|
|
26
|
+
[key: string]: unknown;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Pagination and sorting options
|
|
30
|
+
*/
|
|
31
|
+
export interface QueryOptions {
|
|
32
|
+
limit?: number;
|
|
33
|
+
offset?: number;
|
|
34
|
+
sortBy?: string;
|
|
35
|
+
sortOrder?: 'asc' | 'desc';
|
|
36
|
+
/** When true, runs a separate COUNT(*) query to include exact total. Default: false. */
|
|
37
|
+
includeTotal?: boolean;
|
|
38
|
+
/** Cursor-based pagination: fetch rows after this cursor value (uses sortBy column). */
|
|
39
|
+
cursor?: string;
|
|
40
|
+
/** Sparse fieldset: column names to select. Returns all columns when omitted. */
|
|
41
|
+
fields?: string[];
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Paginated result with metadata
|
|
45
|
+
*/
|
|
46
|
+
export interface PaginatedResult<T> {
|
|
47
|
+
data: T[];
|
|
48
|
+
/** Total count of matching entities. Only present when includeTotal is true. */
|
|
49
|
+
total?: number;
|
|
50
|
+
limit: number;
|
|
51
|
+
offset: number;
|
|
52
|
+
hasMore: boolean;
|
|
53
|
+
/** Cursor pointing to the last item, for cursor-based pagination. */
|
|
54
|
+
nextCursor?: string;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Abstract CRUD service with access control and common operations.
|
|
58
|
+
*
|
|
59
|
+
* @typeParam TEntity - Entity type extending BaseEntity
|
|
60
|
+
* @typeParam TFilter - Filter type for query parameters
|
|
61
|
+
* @typeParam TInsert - Insert DTO type
|
|
62
|
+
* @typeParam TUpdate - Update DTO type
|
|
63
|
+
*
|
|
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
|
+
*
|
|
68
|
+
* Errors are not caught here — they propagate to the route-level error handler (`withRoute`)
|
|
69
|
+
* which provides consistent logging with request context.
|
|
70
|
+
*
|
|
71
|
+
* @example
|
|
72
|
+
* ```typescript
|
|
73
|
+
* class PipelineService extends CrudService<Pipeline, PipelineFilter, PipelineInsert, PipelineUpdate> {
|
|
74
|
+
* protected get schema() { return schema.pipeline; }
|
|
75
|
+
* protected buildConditions(filter, orgId) { return buildPipelineConditions(filter, orgId); }
|
|
76
|
+
* protected getSortColumn(sortBy) { return sortColumnMap[sortBy] ?? null; }
|
|
77
|
+
* protected getProjectColumn() { return schema.pipeline.project; }
|
|
78
|
+
* protected getOrgColumn() { return schema.pipeline.organization; }
|
|
79
|
+
* }
|
|
80
|
+
* ```
|
|
81
|
+
*/
|
|
82
|
+
export declare abstract class CrudService<TEntity extends BaseEntity, TFilter, TInsert, TUpdate> {
|
|
83
|
+
/** Drizzle schema table for this entity */
|
|
84
|
+
protected abstract get schema(): PgTable;
|
|
85
|
+
/** Build SQL conditions for filtering entities */
|
|
86
|
+
protected abstract buildConditions(filter: Partial<TFilter>, orgId?: string): SQL[];
|
|
87
|
+
/** Get the schema column for sorting by field name */
|
|
88
|
+
protected abstract getSortColumn(sortBy: string): AnyColumn | null;
|
|
89
|
+
/** Get the project column for setDefault scoping (null if entity has no project scope) */
|
|
90
|
+
protected abstract getProjectColumn(): AnyColumn | null;
|
|
91
|
+
/** Get the organization column for setDefault scoping */
|
|
92
|
+
protected abstract getOrgColumn(): AnyColumn;
|
|
93
|
+
/** Get the unique constraint columns for onConflictDoUpdate */
|
|
94
|
+
protected abstract get conflictTarget(): AnyColumn[];
|
|
95
|
+
private readonly _logger;
|
|
96
|
+
/** Build conditions for a single entity by ID. */
|
|
97
|
+
private idConditions;
|
|
98
|
+
/** Called after a new entity is created */
|
|
99
|
+
protected onAfterCreate(_entity: TEntity, _userId: string): Promise<void>;
|
|
100
|
+
/** Called after an entity is updated */
|
|
101
|
+
protected onAfterUpdate(_id: string, _entity: TEntity, _userId: string): Promise<void>;
|
|
102
|
+
/** Called after an entity is soft-deleted */
|
|
103
|
+
protected onAfterDelete(_id: string, _entity: TEntity, _userId: string): Promise<void>;
|
|
104
|
+
/**
|
|
105
|
+
* Find entities matching filter criteria
|
|
106
|
+
*
|
|
107
|
+
* @param filter - Filter criteria
|
|
108
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
109
|
+
*/
|
|
110
|
+
find(filter: Partial<TFilter>, orgId?: string): Promise<TEntity[]>;
|
|
111
|
+
/**
|
|
112
|
+
* Find entities with pagination and sorting
|
|
113
|
+
*
|
|
114
|
+
* @param filter - Filter criteria
|
|
115
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
116
|
+
* @param options - Pagination and sorting options
|
|
117
|
+
*/
|
|
118
|
+
findPaginated(filter: Partial<TFilter>, orgId?: string, options?: QueryOptions): Promise<PaginatedResult<TEntity>>;
|
|
119
|
+
/**
|
|
120
|
+
* Build a column selection map for sparse fieldsets.
|
|
121
|
+
* Falls back to full select if no matching columns found.
|
|
122
|
+
*/
|
|
123
|
+
private buildFieldSelect;
|
|
124
|
+
/**
|
|
125
|
+
* Count entities matching filter criteria
|
|
126
|
+
*
|
|
127
|
+
* @param filter - Filter criteria
|
|
128
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
129
|
+
*/
|
|
130
|
+
count(filter: Partial<TFilter>, orgId?: string): Promise<number>;
|
|
131
|
+
/**
|
|
132
|
+
* Find a single entity by ID
|
|
133
|
+
*
|
|
134
|
+
* @param id - Entity ID
|
|
135
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
136
|
+
*/
|
|
137
|
+
findById(id: string, orgId?: string): Promise<TEntity | null>;
|
|
138
|
+
/**
|
|
139
|
+
* Create a new entity
|
|
140
|
+
*/
|
|
141
|
+
create(data: TInsert, userId: string): Promise<TEntity>;
|
|
142
|
+
/**
|
|
143
|
+
* Update an existing entity
|
|
144
|
+
*/
|
|
145
|
+
update(id: string, data: Partial<TUpdate>, orgId: string, userId: string): Promise<TEntity | null>;
|
|
146
|
+
/**
|
|
147
|
+
* Delete an entity (soft delete by setting isActive = false)
|
|
148
|
+
*/
|
|
149
|
+
delete(id: string, orgId: string, userId: string): Promise<TEntity | null>;
|
|
150
|
+
/**
|
|
151
|
+
* Set an entity as the default for a project/organization scope.
|
|
152
|
+
* Marks all other entities as non-default, then sets the specified entity.
|
|
153
|
+
* Uses a transaction to ensure atomicity.
|
|
154
|
+
*/
|
|
155
|
+
setDefault(project: string, org: string, id: string, userId: string): Promise<TEntity>;
|
|
156
|
+
/**
|
|
157
|
+
* Update multiple entities matching filter
|
|
158
|
+
*/
|
|
159
|
+
updateMany(filter: Partial<TFilter>, data: Partial<TUpdate>, orgId: string, userId: string): Promise<TEntity[]>;
|
|
160
|
+
/**
|
|
161
|
+
* Create multiple entities in a single batch insert.
|
|
162
|
+
* Uses upsert (onConflictDoUpdate) — all rows are inserted in one query per chunk.
|
|
163
|
+
* Chunks of 100 to stay within PostgreSQL parameter limits.
|
|
164
|
+
*/
|
|
165
|
+
bulkCreate(items: TInsert[], userId: string): Promise<TEntity[]>;
|
|
166
|
+
/**
|
|
167
|
+
* Soft-delete multiple entities by IDs in a single batch operation.
|
|
168
|
+
*/
|
|
169
|
+
bulkDelete(ids: string[], orgId: string, userId: string): Promise<TEntity[]>;
|
|
170
|
+
}
|
|
@@ -0,0 +1,387 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright 2026 Pipeline Builder Contributors
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
5
|
+
exports.CrudService = void 0;
|
|
6
|
+
exports.drizzleRows = drizzleRows;
|
|
7
|
+
exports.drizzleCount = drizzleCount;
|
|
8
|
+
const api_core_1 = require("@pipeline-builder/api-core");
|
|
9
|
+
const drizzle_orm_1 = require("drizzle-orm");
|
|
10
|
+
const postgres_connection_1 = require("../database/postgres-connection");
|
|
11
|
+
/** Pagination defaults — read from env to match CoreConstants in pipeline-core. */
|
|
12
|
+
const DEFAULT_PAGE_LIMIT = parseInt(process.env.DEFAULT_PAGE_LIMIT || '100', 10);
|
|
13
|
+
const MAX_PAGE_LIMIT = parseInt(process.env.MAX_PAGE_LIMIT || '1000', 10);
|
|
14
|
+
/**
|
|
15
|
+
* Cast Drizzle query results to a typed array.
|
|
16
|
+
* Drizzle's generic return type (`PgSelectBase<...>`) doesn't narrow to our
|
|
17
|
+
* entity generics, so an explicit cast is needed. Centralised here so every
|
|
18
|
+
* call-site stays one-liner clean and the cast is documented in one place.
|
|
19
|
+
*/
|
|
20
|
+
function drizzleRows(rows) {
|
|
21
|
+
return rows;
|
|
22
|
+
}
|
|
23
|
+
/** Cast a Drizzle aggregate result to extract `[{ count: number }]`. */
|
|
24
|
+
function drizzleCount(rows) {
|
|
25
|
+
return rows;
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Abstract CRUD service with access control and common operations.
|
|
29
|
+
*
|
|
30
|
+
* @typeParam TEntity - Entity type extending BaseEntity
|
|
31
|
+
* @typeParam TFilter - Filter type for query parameters
|
|
32
|
+
* @typeParam TInsert - Insert DTO type
|
|
33
|
+
* @typeParam TUpdate - Update DTO type
|
|
34
|
+
*
|
|
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
|
+
*
|
|
39
|
+
* Errors are not caught here — they propagate to the route-level error handler (`withRoute`)
|
|
40
|
+
* which provides consistent logging with request context.
|
|
41
|
+
*
|
|
42
|
+
* @example
|
|
43
|
+
* ```typescript
|
|
44
|
+
* class PipelineService extends CrudService<Pipeline, PipelineFilter, PipelineInsert, PipelineUpdate> {
|
|
45
|
+
* protected get schema() { return schema.pipeline; }
|
|
46
|
+
* protected buildConditions(filter, orgId) { return buildPipelineConditions(filter, orgId); }
|
|
47
|
+
* protected getSortColumn(sortBy) { return sortColumnMap[sortBy] ?? null; }
|
|
48
|
+
* protected getProjectColumn() { return schema.pipeline.project; }
|
|
49
|
+
* protected getOrgColumn() { return schema.pipeline.organization; }
|
|
50
|
+
* }
|
|
51
|
+
* ```
|
|
52
|
+
*/
|
|
53
|
+
class CrudService {
|
|
54
|
+
_logger = (0, api_core_1.createLogger)('CrudService');
|
|
55
|
+
/** Build conditions for a single entity by ID. */
|
|
56
|
+
idConditions(id, orgId) {
|
|
57
|
+
return this.buildConditions({ id }, orgId);
|
|
58
|
+
}
|
|
59
|
+
// Lifecycle hooks — override in subclasses to react to mutations
|
|
60
|
+
// These are fire-and-forget: errors are logged but never block the caller.
|
|
61
|
+
/** Called after a new entity is created */
|
|
62
|
+
async onAfterCreate(_entity, _userId) { }
|
|
63
|
+
/** Called after an entity is updated */
|
|
64
|
+
async onAfterUpdate(_id, _entity, _userId) { }
|
|
65
|
+
/** Called after an entity is soft-deleted */
|
|
66
|
+
async onAfterDelete(_id, _entity, _userId) { }
|
|
67
|
+
/**
|
|
68
|
+
* Find entities matching filter criteria
|
|
69
|
+
*
|
|
70
|
+
* @param filter - Filter criteria
|
|
71
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
72
|
+
*/
|
|
73
|
+
async find(filter, orgId) {
|
|
74
|
+
const conditions = this.buildConditions(filter, orgId);
|
|
75
|
+
return postgres_connection_1.db
|
|
76
|
+
.select()
|
|
77
|
+
.from(this.schema)
|
|
78
|
+
.where((0, drizzle_orm_1.and)(...conditions)).then(r => drizzleRows(r));
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Find entities with pagination and sorting
|
|
82
|
+
*
|
|
83
|
+
* @param filter - Filter criteria
|
|
84
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
85
|
+
* @param options - Pagination and sorting options
|
|
86
|
+
*/
|
|
87
|
+
async findPaginated(filter, orgId, options = {}) {
|
|
88
|
+
const { limit: rawLimit = DEFAULT_PAGE_LIMIT, offset = 0, sortBy, sortOrder = 'asc', includeTotal = false, cursor, fields } = options;
|
|
89
|
+
const limit = Math.min(Math.max(1, rawLimit), MAX_PAGE_LIMIT);
|
|
90
|
+
// Cursor and offset are mutually exclusive — cursor takes precedence
|
|
91
|
+
const useCursor = !!(cursor && sortBy);
|
|
92
|
+
const conditions = this.buildConditions(filter, orgId);
|
|
93
|
+
// Cursor-based pagination: add WHERE clause for keyset pagination
|
|
94
|
+
if (useCursor) {
|
|
95
|
+
const sortColumn = this.getSortColumn(sortBy);
|
|
96
|
+
if (sortColumn) {
|
|
97
|
+
const op = sortOrder === 'desc'
|
|
98
|
+
? (0, drizzle_orm_1.sql) `${sortColumn} < ${cursor}`
|
|
99
|
+
: (0, drizzle_orm_1.sql) `${sortColumn} > ${cursor}`;
|
|
100
|
+
conditions.push(op);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
// Build SELECT — sparse fieldset when fields are specified
|
|
104
|
+
const selectSpec = fields ? this.buildFieldSelect(fields) : undefined;
|
|
105
|
+
let query = selectSpec
|
|
106
|
+
? postgres_connection_1.db.select(selectSpec).from(this.schema).where((0, drizzle_orm_1.and)(...conditions))
|
|
107
|
+
: postgres_connection_1.db.select().from(this.schema).where((0, drizzle_orm_1.and)(...conditions));
|
|
108
|
+
if (sortBy) {
|
|
109
|
+
const sortColumn = this.getSortColumn(sortBy);
|
|
110
|
+
if (sortColumn) {
|
|
111
|
+
query = query.orderBy(sortOrder === 'desc' ? (0, drizzle_orm_1.desc)(sortColumn) : (0, drizzle_orm_1.asc)(sortColumn));
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
// Fetch limit+1 to detect hasMore without COUNT(*)
|
|
115
|
+
const effectiveOffset = useCursor ? 0 : offset;
|
|
116
|
+
const rows = await query
|
|
117
|
+
.limit(limit + 1)
|
|
118
|
+
.offset(effectiveOffset).then(r => drizzleRows(r));
|
|
119
|
+
const hasMore = rows.length > limit;
|
|
120
|
+
const data = hasMore ? rows.slice(0, limit) : rows;
|
|
121
|
+
const result = { data, limit, offset: effectiveOffset, hasMore };
|
|
122
|
+
// Provide next cursor from last item's sort column value
|
|
123
|
+
if (data.length > 0 && sortBy) {
|
|
124
|
+
const lastItem = data[data.length - 1];
|
|
125
|
+
const cursorValue = lastItem[sortBy];
|
|
126
|
+
if (cursorValue !== undefined) {
|
|
127
|
+
result.nextCursor = cursorValue instanceof Date ? cursorValue.toISOString() : String(cursorValue);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
// Only run the COUNT(*) query when the caller explicitly needs the total
|
|
131
|
+
if (includeTotal) {
|
|
132
|
+
const baseConditions = this.buildConditions(filter, orgId);
|
|
133
|
+
const [countResult] = await postgres_connection_1.db
|
|
134
|
+
.select({ count: (0, drizzle_orm_1.sql) `count(*)::int` })
|
|
135
|
+
.from(this.schema)
|
|
136
|
+
.where((0, drizzle_orm_1.and)(...baseConditions)).then(r => drizzleCount(r));
|
|
137
|
+
result.total = countResult?.count || 0;
|
|
138
|
+
}
|
|
139
|
+
return result;
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Build a column selection map for sparse fieldsets.
|
|
143
|
+
* Falls back to full select if no matching columns found.
|
|
144
|
+
*/
|
|
145
|
+
buildFieldSelect(fields) {
|
|
146
|
+
if (fields.length === 0)
|
|
147
|
+
return undefined;
|
|
148
|
+
const columns = {};
|
|
149
|
+
// Always include id for entity identity
|
|
150
|
+
columns.id = this.schema.id;
|
|
151
|
+
for (const field of fields) {
|
|
152
|
+
if (field === 'id')
|
|
153
|
+
continue; // Already included
|
|
154
|
+
const col = this.schema[field];
|
|
155
|
+
if (col)
|
|
156
|
+
columns[field] = col;
|
|
157
|
+
}
|
|
158
|
+
// At minimum we'll have { id }, which is valid
|
|
159
|
+
return columns;
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Count entities matching filter criteria
|
|
163
|
+
*
|
|
164
|
+
* @param filter - Filter criteria
|
|
165
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
166
|
+
*/
|
|
167
|
+
async count(filter, orgId) {
|
|
168
|
+
const conditions = this.buildConditions(filter, orgId);
|
|
169
|
+
const [result] = await postgres_connection_1.db
|
|
170
|
+
.select({ count: (0, drizzle_orm_1.sql) `count(*)::int` })
|
|
171
|
+
.from(this.schema)
|
|
172
|
+
.where((0, drizzle_orm_1.and)(...conditions)).then(r => drizzleCount(r));
|
|
173
|
+
return result?.count || 0;
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Find a single entity by ID
|
|
177
|
+
*
|
|
178
|
+
* @param id - Entity ID
|
|
179
|
+
* @param orgId - User's organization ID (optional — omit for anonymous/system-public-only access)
|
|
180
|
+
*/
|
|
181
|
+
async findById(id, orgId) {
|
|
182
|
+
const conditions = this.idConditions(id, orgId);
|
|
183
|
+
const results = await postgres_connection_1.db
|
|
184
|
+
.select()
|
|
185
|
+
.from(this.schema)
|
|
186
|
+
.where((0, drizzle_orm_1.and)(...conditions))
|
|
187
|
+
.limit(1).then(r => drizzleRows(r));
|
|
188
|
+
return results[0] || null;
|
|
189
|
+
}
|
|
190
|
+
// Mutation operations
|
|
191
|
+
/**
|
|
192
|
+
* Create a new entity
|
|
193
|
+
*/
|
|
194
|
+
async create(data, userId) {
|
|
195
|
+
const [created] = await postgres_connection_1.db
|
|
196
|
+
.insert(this.schema)
|
|
197
|
+
.values({
|
|
198
|
+
...data,
|
|
199
|
+
createdBy: userId || 'system',
|
|
200
|
+
updatedBy: userId || 'system',
|
|
201
|
+
})
|
|
202
|
+
.onConflictDoUpdate({
|
|
203
|
+
target: this.conflictTarget,
|
|
204
|
+
set: {
|
|
205
|
+
...data,
|
|
206
|
+
updatedAt: new Date(),
|
|
207
|
+
updatedBy: userId || 'system',
|
|
208
|
+
},
|
|
209
|
+
})
|
|
210
|
+
.returning().then(r => drizzleRows(r));
|
|
211
|
+
this.onAfterCreate(created, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));
|
|
212
|
+
return created;
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Update an existing entity
|
|
216
|
+
*/
|
|
217
|
+
async update(id, data, orgId, userId) {
|
|
218
|
+
const conditions = this.idConditions(id, orgId);
|
|
219
|
+
const [updated] = await postgres_connection_1.db
|
|
220
|
+
.update(this.schema)
|
|
221
|
+
.set({
|
|
222
|
+
...data,
|
|
223
|
+
updatedAt: new Date(),
|
|
224
|
+
updatedBy: userId || 'system',
|
|
225
|
+
})
|
|
226
|
+
.where((0, drizzle_orm_1.and)(...conditions))
|
|
227
|
+
.returning().then(r => drizzleRows(r));
|
|
228
|
+
if (updated) {
|
|
229
|
+
this.onAfterUpdate(id, updated, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));
|
|
230
|
+
}
|
|
231
|
+
return updated || null;
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Delete an entity (soft delete by setting isActive = false)
|
|
235
|
+
*/
|
|
236
|
+
async delete(id, orgId, userId) {
|
|
237
|
+
const conditions = this.idConditions(id, orgId);
|
|
238
|
+
const [deleted] = await postgres_connection_1.db
|
|
239
|
+
.update(this.schema)
|
|
240
|
+
.set({
|
|
241
|
+
isActive: false,
|
|
242
|
+
updatedAt: new Date(),
|
|
243
|
+
updatedBy: userId || 'system',
|
|
244
|
+
deletedAt: new Date(),
|
|
245
|
+
deletedBy: userId || 'system',
|
|
246
|
+
})
|
|
247
|
+
.where((0, drizzle_orm_1.and)(...conditions))
|
|
248
|
+
.returning().then(r => drizzleRows(r));
|
|
249
|
+
if (deleted) {
|
|
250
|
+
this.onAfterDelete(id, deleted, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));
|
|
251
|
+
}
|
|
252
|
+
return deleted || null;
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Set an entity as the default for a project/organization scope.
|
|
256
|
+
* Marks all other entities as non-default, then sets the specified entity.
|
|
257
|
+
* Uses a transaction to ensure atomicity.
|
|
258
|
+
*/
|
|
259
|
+
async setDefault(project, org, id, userId) {
|
|
260
|
+
return postgres_connection_1.db.transaction(async (tx) => {
|
|
261
|
+
const orgColumn = this.getOrgColumn();
|
|
262
|
+
const projectColumn = this.getProjectColumn();
|
|
263
|
+
// Build scoping conditions for clearing defaults
|
|
264
|
+
const scopeConditions = [
|
|
265
|
+
(0, drizzle_orm_1.eq)(orgColumn, org),
|
|
266
|
+
(0, drizzle_orm_1.eq)(this.schema.isDefault, true),
|
|
267
|
+
];
|
|
268
|
+
if (projectColumn) {
|
|
269
|
+
scopeConditions.push((0, drizzle_orm_1.eq)(projectColumn, project));
|
|
270
|
+
}
|
|
271
|
+
// Lock existing defaults with FOR UPDATE to prevent concurrent setDefault races
|
|
272
|
+
await tx.execute((0, drizzle_orm_1.sql) `SELECT id FROM ${this.schema}
|
|
273
|
+
WHERE ${orgColumn} = ${org}
|
|
274
|
+
AND ${this.schema.isDefault} = true
|
|
275
|
+
${projectColumn ? (0, drizzle_orm_1.sql) `AND ${projectColumn} = ${project}` : (0, drizzle_orm_1.sql) ``}
|
|
276
|
+
FOR UPDATE`);
|
|
277
|
+
// Mark all entities in scope as non-default
|
|
278
|
+
await tx
|
|
279
|
+
.update(this.schema)
|
|
280
|
+
.set({
|
|
281
|
+
isDefault: false,
|
|
282
|
+
updatedAt: new Date(),
|
|
283
|
+
updatedBy: userId || 'system',
|
|
284
|
+
})
|
|
285
|
+
.where((0, drizzle_orm_1.and)(...scopeConditions));
|
|
286
|
+
// Set the specified entity as default
|
|
287
|
+
const [updated] = await tx
|
|
288
|
+
.update(this.schema)
|
|
289
|
+
.set({
|
|
290
|
+
isDefault: true,
|
|
291
|
+
updatedAt: new Date(),
|
|
292
|
+
updatedBy: userId || 'system',
|
|
293
|
+
})
|
|
294
|
+
.where((0, drizzle_orm_1.and)((0, drizzle_orm_1.eq)(this.schema.id, id), (0, drizzle_orm_1.eq)(orgColumn, org)))
|
|
295
|
+
.returning().then(r => drizzleRows(r));
|
|
296
|
+
if (!updated) {
|
|
297
|
+
throw new api_core_1.NotFoundError(`Entity with id ${id} not found`);
|
|
298
|
+
}
|
|
299
|
+
return updated;
|
|
300
|
+
});
|
|
301
|
+
}
|
|
302
|
+
/**
|
|
303
|
+
* Update multiple entities matching filter
|
|
304
|
+
*/
|
|
305
|
+
async updateMany(filter, data, orgId, userId) {
|
|
306
|
+
const conditions = this.buildConditions(filter, orgId);
|
|
307
|
+
return postgres_connection_1.db
|
|
308
|
+
.update(this.schema)
|
|
309
|
+
.set({
|
|
310
|
+
...data,
|
|
311
|
+
updatedAt: new Date(),
|
|
312
|
+
updatedBy: userId || 'system',
|
|
313
|
+
})
|
|
314
|
+
.where((0, drizzle_orm_1.and)(...conditions))
|
|
315
|
+
.returning().then(r => drizzleRows(r));
|
|
316
|
+
}
|
|
317
|
+
/**
|
|
318
|
+
* Create multiple entities in a single batch insert.
|
|
319
|
+
* Uses upsert (onConflictDoUpdate) — all rows are inserted in one query per chunk.
|
|
320
|
+
* Chunks of 100 to stay within PostgreSQL parameter limits.
|
|
321
|
+
*/
|
|
322
|
+
async bulkCreate(items, userId) {
|
|
323
|
+
if (items.length === 0)
|
|
324
|
+
return [];
|
|
325
|
+
const CHUNK_SIZE = 100;
|
|
326
|
+
const now = new Date();
|
|
327
|
+
const user = userId || 'system';
|
|
328
|
+
const results = await postgres_connection_1.db.transaction(async (tx) => {
|
|
329
|
+
const allCreated = [];
|
|
330
|
+
for (let i = 0; i < items.length; i += CHUNK_SIZE) {
|
|
331
|
+
const chunk = items.slice(i, i + CHUNK_SIZE);
|
|
332
|
+
const values = chunk.map(data => ({
|
|
333
|
+
...data,
|
|
334
|
+
createdBy: user,
|
|
335
|
+
updatedBy: user,
|
|
336
|
+
}));
|
|
337
|
+
const created = await tx
|
|
338
|
+
.insert(this.schema)
|
|
339
|
+
.values(values)
|
|
340
|
+
.onConflictDoUpdate({
|
|
341
|
+
target: this.conflictTarget,
|
|
342
|
+
set: {
|
|
343
|
+
updatedAt: now,
|
|
344
|
+
updatedBy: user,
|
|
345
|
+
},
|
|
346
|
+
})
|
|
347
|
+
.returning().then(r => drizzleRows(r));
|
|
348
|
+
allCreated.push(...created);
|
|
349
|
+
}
|
|
350
|
+
return allCreated;
|
|
351
|
+
});
|
|
352
|
+
for (const entity of results) {
|
|
353
|
+
this.onAfterCreate(entity, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));
|
|
354
|
+
}
|
|
355
|
+
return results;
|
|
356
|
+
}
|
|
357
|
+
/**
|
|
358
|
+
* Soft-delete multiple entities by IDs in a single batch operation.
|
|
359
|
+
*/
|
|
360
|
+
async bulkDelete(ids, orgId, userId) {
|
|
361
|
+
if (ids.length === 0)
|
|
362
|
+
return [];
|
|
363
|
+
const now = new Date();
|
|
364
|
+
const user = userId || 'system';
|
|
365
|
+
const conditions = [
|
|
366
|
+
(0, drizzle_orm_1.inArray)(this.schema.id, ids),
|
|
367
|
+
...this.buildConditions({}, orgId),
|
|
368
|
+
];
|
|
369
|
+
const deleted = await postgres_connection_1.db
|
|
370
|
+
.update(this.schema)
|
|
371
|
+
.set({
|
|
372
|
+
isActive: false,
|
|
373
|
+
updatedAt: now,
|
|
374
|
+
updatedBy: user,
|
|
375
|
+
deletedAt: now,
|
|
376
|
+
deletedBy: user,
|
|
377
|
+
})
|
|
378
|
+
.where((0, drizzle_orm_1.and)(...conditions))
|
|
379
|
+
.returning().then(r => drizzleRows(r));
|
|
380
|
+
for (const entity of deleted) {
|
|
381
|
+
this.onAfterDelete(entity.id, entity, userId).catch(err => this._logger.warn('Lifecycle hook failed', { error: String(err) }));
|
|
382
|
+
}
|
|
383
|
+
return deleted;
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
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"]}
|