@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,248 @@
|
|
|
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.reportingService = exports.ReportingService = void 0;
|
|
6
|
+
const api_core_1 = require("@pipeline-builder/api-core");
|
|
7
|
+
const drizzle_orm_1 = require("drizzle-orm");
|
|
8
|
+
const crud_service_1 = require("./crud-service");
|
|
9
|
+
const drizzle_schema_1 = require("../database/drizzle-schema");
|
|
10
|
+
const postgres_connection_1 = require("../database/postgres-connection");
|
|
11
|
+
/** Cast raw SQL result rows to a typed array. Alias for drizzleRows. */
|
|
12
|
+
const sqlRows = (result) => (0, crud_service_1.drizzleRows)(result.rows);
|
|
13
|
+
/**
|
|
14
|
+
* Cache for reporting aggregations. Two tiers:
|
|
15
|
+
* - Inventory queries (plugin summary/distribution/versions): 5 min TTL — changes on plugin CRUD
|
|
16
|
+
* - Execution/build queries with date ranges: 2 min TTL — new events arrive continuously
|
|
17
|
+
*/
|
|
18
|
+
const inventoryCache = (0, api_core_1.createCacheService)('report:inv:', parseInt(process.env.CACHE_TTL_REPORT_INVENTORY || '300', 10));
|
|
19
|
+
const timeseriesCache = (0, api_core_1.createCacheService)('report:ts:', parseInt(process.env.CACHE_TTL_REPORT_TIMESERIES || '120', 10));
|
|
20
|
+
// ─── Service ────────────────────────────────────────────
|
|
21
|
+
/**
|
|
22
|
+
* Read-only reporting service for pipeline execution and plugin inventory aggregations.
|
|
23
|
+
* Does not extend CrudService — reports are aggregate queries, not entity CRUD.
|
|
24
|
+
*
|
|
25
|
+
* All queries are cached in-memory to avoid repeated expensive SQL aggregations:
|
|
26
|
+
* - Inventory queries (plugin summary/distribution/versions): 5 min TTL
|
|
27
|
+
* - Timeseries queries (execution/build metrics with date ranges): 2 min TTL
|
|
28
|
+
*/
|
|
29
|
+
class ReportingService {
|
|
30
|
+
/** Invalidate all cached reports for an org (call after event ingest). */
|
|
31
|
+
async invalidateOrg(orgId) {
|
|
32
|
+
await Promise.all([
|
|
33
|
+
inventoryCache.invalidatePattern(`${orgId}:*`),
|
|
34
|
+
timeseriesCache.invalidatePattern(`${orgId}:*`),
|
|
35
|
+
]);
|
|
36
|
+
}
|
|
37
|
+
// ── Category 1: Pipeline Execution & Performance ──
|
|
38
|
+
/** 1.1 Execution count per pipeline with status breakdown. */
|
|
39
|
+
async getExecutionCount(orgId) {
|
|
40
|
+
return timeseriesCache.getOrSet(`${orgId}:exec-count`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
41
|
+
SELECT
|
|
42
|
+
p.id, p.project, p.organization, p.pipeline_name,
|
|
43
|
+
COUNT(*)::int AS total,
|
|
44
|
+
COUNT(*) FILTER (WHERE e.status = 'SUCCEEDED')::int AS succeeded,
|
|
45
|
+
COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failed,
|
|
46
|
+
COUNT(*) FILTER (WHERE e.status = 'CANCELED')::int AS canceled,
|
|
47
|
+
MIN(e.started_at)::text AS first_execution,
|
|
48
|
+
MAX(e.started_at)::text AS last_execution
|
|
49
|
+
FROM ${drizzle_schema_1.schema.pipeline} p
|
|
50
|
+
JOIN ${drizzle_schema_1.schema.pipelineEvent} e ON e.pipeline_id = p.id
|
|
51
|
+
AND e.event_type = 'PIPELINE' AND e.status != 'STARTED'
|
|
52
|
+
WHERE p.org_id = ${orgId} AND p.is_active = true
|
|
53
|
+
GROUP BY p.id
|
|
54
|
+
ORDER BY total DESC
|
|
55
|
+
`).then(r => sqlRows(r)));
|
|
56
|
+
}
|
|
57
|
+
/** 1.2 Success rate over time for an org. */
|
|
58
|
+
async getSuccessRate(orgId, interval, from, to) {
|
|
59
|
+
return timeseriesCache.getOrSet(`${orgId}:success-rate:${interval}:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
60
|
+
SELECT
|
|
61
|
+
DATE_TRUNC(${interval}, e.started_at)::text AS period,
|
|
62
|
+
COUNT(*) FILTER (WHERE e.status = 'SUCCEEDED')::int AS succeeded,
|
|
63
|
+
COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failed,
|
|
64
|
+
COUNT(*) FILTER (WHERE e.status = 'CANCELED')::int AS canceled,
|
|
65
|
+
ROUND(COUNT(*) FILTER (WHERE e.status = 'SUCCEEDED')::numeric
|
|
66
|
+
/ NULLIF(COUNT(*), 0) * 100, 1)::float AS success_pct
|
|
67
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
68
|
+
JOIN ${drizzle_schema_1.schema.pipeline} p ON p.id = e.pipeline_id
|
|
69
|
+
WHERE p.org_id = ${orgId} AND e.event_type = 'PIPELINE'
|
|
70
|
+
AND e.status IN ('SUCCEEDED', 'FAILED', 'CANCELED')
|
|
71
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
72
|
+
GROUP BY period ORDER BY period
|
|
73
|
+
`).then(r => sqlRows(r)));
|
|
74
|
+
}
|
|
75
|
+
/** 1.3 Average duration per pipeline. */
|
|
76
|
+
async getAverageDuration(orgId, from, to) {
|
|
77
|
+
return timeseriesCache.getOrSet(`${orgId}:avg-duration:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
78
|
+
SELECT
|
|
79
|
+
p.id, p.project, p.pipeline_name,
|
|
80
|
+
AVG(e.duration_ms)::int AS avg_ms,
|
|
81
|
+
MIN(e.duration_ms)::int AS min_ms,
|
|
82
|
+
MAX(e.duration_ms)::int AS max_ms,
|
|
83
|
+
PERCENTILE_CONT(0.95) WITHIN GROUP (ORDER BY e.duration_ms)::int AS p95_ms,
|
|
84
|
+
COUNT(*)::int AS executions
|
|
85
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
86
|
+
JOIN ${drizzle_schema_1.schema.pipeline} p ON p.id = e.pipeline_id
|
|
87
|
+
WHERE p.org_id = ${orgId} AND e.event_type = 'PIPELINE' AND e.duration_ms IS NOT NULL
|
|
88
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
89
|
+
GROUP BY p.id ORDER BY avg_ms DESC
|
|
90
|
+
`).then(r => sqlRows(r)));
|
|
91
|
+
}
|
|
92
|
+
/** 1.5 Stage failure heatmap — which stages fail most. */
|
|
93
|
+
async getStageFailures(orgId, from, to) {
|
|
94
|
+
return timeseriesCache.getOrSet(`${orgId}:stage-failures:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
95
|
+
SELECT
|
|
96
|
+
e.stage_name,
|
|
97
|
+
COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failures,
|
|
98
|
+
COUNT(*)::int AS total,
|
|
99
|
+
ROUND(COUNT(*) FILTER (WHERE e.status = 'FAILED')::numeric
|
|
100
|
+
/ NULLIF(COUNT(*), 0) * 100, 1)::float AS failure_pct
|
|
101
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
102
|
+
JOIN ${drizzle_schema_1.schema.pipeline} p ON p.id = e.pipeline_id
|
|
103
|
+
WHERE p.org_id = ${orgId} AND e.event_type = 'STAGE' AND e.stage_name IS NOT NULL
|
|
104
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
105
|
+
GROUP BY e.stage_name ORDER BY failures DESC
|
|
106
|
+
`).then(r => sqlRows(r)));
|
|
107
|
+
}
|
|
108
|
+
/** 1.6 Stage bottlenecks — slowest stages per pipeline. */
|
|
109
|
+
async getStageBottlenecks(orgId, from, to) {
|
|
110
|
+
return timeseriesCache.getOrSet(`${orgId}:stage-bottlenecks:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
111
|
+
SELECT
|
|
112
|
+
p.id, p.pipeline_name, e.stage_name,
|
|
113
|
+
AVG(e.duration_ms)::int AS avg_ms,
|
|
114
|
+
MAX(e.duration_ms)::int AS max_ms
|
|
115
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
116
|
+
JOIN ${drizzle_schema_1.schema.pipeline} p ON p.id = e.pipeline_id
|
|
117
|
+
WHERE p.org_id = ${orgId} AND e.event_type = 'STAGE' AND e.duration_ms IS NOT NULL
|
|
118
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
119
|
+
GROUP BY p.id, e.stage_name ORDER BY avg_ms DESC
|
|
120
|
+
`).then(r => sqlRows(r)));
|
|
121
|
+
}
|
|
122
|
+
/** 1.7 Action failure rate — which plugin steps fail most. */
|
|
123
|
+
async getActionFailures(orgId, from, to) {
|
|
124
|
+
return timeseriesCache.getOrSet(`${orgId}:action-failures:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
125
|
+
SELECT
|
|
126
|
+
e.action_name,
|
|
127
|
+
COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failures,
|
|
128
|
+
COUNT(*)::int AS total,
|
|
129
|
+
ROUND(COUNT(*) FILTER (WHERE e.status = 'FAILED')::numeric
|
|
130
|
+
/ NULLIF(COUNT(*), 0) * 100, 1)::float AS failure_pct
|
|
131
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
132
|
+
JOIN ${drizzle_schema_1.schema.pipeline} p ON p.id = e.pipeline_id
|
|
133
|
+
WHERE p.org_id = ${orgId} AND e.event_type = 'ACTION' AND e.action_name IS NOT NULL
|
|
134
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
135
|
+
GROUP BY e.action_name ORDER BY failures DESC
|
|
136
|
+
`).then(r => sqlRows(r)));
|
|
137
|
+
}
|
|
138
|
+
/** 1.8 Error categorization — group failure messages. */
|
|
139
|
+
async getErrors(orgId, from, to, limit = 20) {
|
|
140
|
+
return timeseriesCache.getOrSet(`${orgId}:errors:${from}:${to}:${limit}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
141
|
+
SELECT
|
|
142
|
+
SUBSTRING(e.error_message FROM 1 FOR 200) AS error_pattern,
|
|
143
|
+
COUNT(*)::int AS occurrences,
|
|
144
|
+
COUNT(DISTINCT e.pipeline_id)::int AS affected_pipelines,
|
|
145
|
+
MAX(e.started_at)::text AS last_seen
|
|
146
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
147
|
+
JOIN ${drizzle_schema_1.schema.pipeline} p ON p.id = e.pipeline_id
|
|
148
|
+
WHERE p.org_id = ${orgId} AND e.status = 'FAILED' AND e.error_message IS NOT NULL
|
|
149
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
150
|
+
GROUP BY error_pattern ORDER BY occurrences DESC
|
|
151
|
+
LIMIT ${limit}
|
|
152
|
+
`).then(r => sqlRows(r)));
|
|
153
|
+
}
|
|
154
|
+
// ── Category 2: Plugin Inventory & Builds ──
|
|
155
|
+
/** 2.1 Plugin summary — counts and breakdowns. */
|
|
156
|
+
async getPluginSummary(orgId) {
|
|
157
|
+
return inventoryCache.getOrSet(`${orgId}:plugin-summary`, async () => {
|
|
158
|
+
const rows = await postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
159
|
+
SELECT
|
|
160
|
+
COUNT(*)::int AS total,
|
|
161
|
+
COUNT(*) FILTER (WHERE ${drizzle_schema_1.schema.plugin.isActive})::int AS active,
|
|
162
|
+
COUNT(*) FILTER (WHERE NOT ${drizzle_schema_1.schema.plugin.isActive})::int AS inactive,
|
|
163
|
+
COUNT(*) FILTER (WHERE ${drizzle_schema_1.schema.plugin.accessModifier} = 'public')::int AS public,
|
|
164
|
+
COUNT(*) FILTER (WHERE ${drizzle_schema_1.schema.plugin.accessModifier} = 'private')::int AS private,
|
|
165
|
+
COUNT(DISTINCT ${drizzle_schema_1.schema.plugin.name})::int AS unique_names
|
|
166
|
+
FROM ${drizzle_schema_1.schema.plugin}
|
|
167
|
+
WHERE ${drizzle_schema_1.schema.plugin.orgId} = ${orgId}
|
|
168
|
+
`);
|
|
169
|
+
return (sqlRows(rows)[0] || { total: 0, active: 0, inactive: 0, public: 0, private: 0, uniqueNames: 0 });
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
/** 2.2 Type & compute distribution. */
|
|
173
|
+
async getPluginDistribution(orgId) {
|
|
174
|
+
return inventoryCache.getOrSet(`${orgId}:plugin-distribution`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
175
|
+
SELECT
|
|
176
|
+
${drizzle_schema_1.schema.plugin.pluginType} AS plugin_type,
|
|
177
|
+
${drizzle_schema_1.schema.plugin.computeType} AS compute_type,
|
|
178
|
+
COUNT(*)::int AS count
|
|
179
|
+
FROM ${drizzle_schema_1.schema.plugin}
|
|
180
|
+
WHERE ${drizzle_schema_1.schema.plugin.orgId} = ${orgId} AND ${drizzle_schema_1.schema.plugin.isActive} = true
|
|
181
|
+
GROUP BY ${drizzle_schema_1.schema.plugin.pluginType}, ${drizzle_schema_1.schema.plugin.computeType}
|
|
182
|
+
ORDER BY count DESC
|
|
183
|
+
`).then(r => sqlRows(r)));
|
|
184
|
+
}
|
|
185
|
+
/** 2.3 Version counts per plugin name. */
|
|
186
|
+
async getPluginVersions(orgId) {
|
|
187
|
+
return inventoryCache.getOrSet(`${orgId}:plugin-versions`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
188
|
+
SELECT
|
|
189
|
+
${drizzle_schema_1.schema.plugin.name},
|
|
190
|
+
COUNT(*)::int AS version_count,
|
|
191
|
+
MAX(${drizzle_schema_1.schema.plugin.version}) AS latest_version,
|
|
192
|
+
bool_or(${drizzle_schema_1.schema.plugin.isDefault}) AS has_default
|
|
193
|
+
FROM ${drizzle_schema_1.schema.plugin}
|
|
194
|
+
WHERE ${drizzle_schema_1.schema.plugin.orgId} = ${orgId} AND ${drizzle_schema_1.schema.plugin.isActive} = true
|
|
195
|
+
GROUP BY ${drizzle_schema_1.schema.plugin.name}
|
|
196
|
+
ORDER BY version_count DESC
|
|
197
|
+
`).then(r => sqlRows(r)));
|
|
198
|
+
}
|
|
199
|
+
/** 2.4 Build success rate over time. */
|
|
200
|
+
async getBuildSuccessRate(orgId, interval, from, to) {
|
|
201
|
+
return timeseriesCache.getOrSet(`${orgId}:build-success:${interval}:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
202
|
+
SELECT
|
|
203
|
+
DATE_TRUNC(${interval}, e.started_at)::text AS period,
|
|
204
|
+
COUNT(*) FILTER (WHERE e.status = 'completed')::int AS succeeded,
|
|
205
|
+
COUNT(*) FILTER (WHERE e.status = 'failed')::int AS failed,
|
|
206
|
+
ROUND(COUNT(*) FILTER (WHERE e.status = 'completed')::numeric
|
|
207
|
+
/ NULLIF(COUNT(*), 0) * 100, 1)::float AS success_pct
|
|
208
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
209
|
+
WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build'
|
|
210
|
+
AND e.status IN ('completed', 'failed')
|
|
211
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
212
|
+
GROUP BY period ORDER BY period
|
|
213
|
+
`).then(r => sqlRows(r)));
|
|
214
|
+
}
|
|
215
|
+
/** 2.5 Build duration per plugin. */
|
|
216
|
+
async getBuildDuration(orgId, from, to) {
|
|
217
|
+
return timeseriesCache.getOrSet(`${orgId}:build-duration:${from}:${to}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
218
|
+
SELECT
|
|
219
|
+
e.detail->>'pluginName' AS plugin_name,
|
|
220
|
+
AVG(e.duration_ms)::int AS avg_ms,
|
|
221
|
+
MAX(e.duration_ms)::int AS max_ms,
|
|
222
|
+
COUNT(*)::int AS builds
|
|
223
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
224
|
+
WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build' AND e.duration_ms IS NOT NULL
|
|
225
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
226
|
+
GROUP BY plugin_name ORDER BY avg_ms DESC
|
|
227
|
+
`).then(r => sqlRows(r)));
|
|
228
|
+
}
|
|
229
|
+
/** 2.6 Build failures — top error messages. */
|
|
230
|
+
async getBuildFailures(orgId, from, to, limit = 20) {
|
|
231
|
+
return timeseriesCache.getOrSet(`${orgId}:build-failures:${from}:${to}:${limit}`, () => postgres_connection_1.db.execute((0, drizzle_orm_1.sql) `
|
|
232
|
+
SELECT
|
|
233
|
+
e.detail->>'pluginName' AS plugin_name,
|
|
234
|
+
e.error_message,
|
|
235
|
+
COUNT(*)::int AS occurrences,
|
|
236
|
+
MAX(e.started_at)::text AS last_seen
|
|
237
|
+
FROM ${drizzle_schema_1.schema.pipelineEvent} e
|
|
238
|
+
WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build' AND e.status = 'failed'
|
|
239
|
+
AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
|
|
240
|
+
GROUP BY plugin_name, e.error_message
|
|
241
|
+
ORDER BY occurrences DESC
|
|
242
|
+
LIMIT ${limit}
|
|
243
|
+
`).then(r => sqlRows(r)));
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
exports.ReportingService = ReportingService;
|
|
247
|
+
exports.reportingService = new ReportingService();
|
|
248
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reporting-service.js","sourceRoot":"","sources":["../../src/api/reporting-service.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createCacheService } from '@pipeline-builder/api-core';\nimport { sql } from 'drizzle-orm';\nimport { drizzleRows } from './crud-service';\nimport { schema } from '../database/drizzle-schema';\nimport { db } from '../database/postgres-connection';\n\n/** Cast raw SQL result rows to a typed array. Alias for drizzleRows. */\nconst sqlRows = <T>(result: { rows: unknown[] }): T[] => drizzleRows<T>(result.rows);\n\n/**\n * Cache for reporting aggregations. Two tiers:\n * - Inventory queries (plugin summary/distribution/versions): 5 min TTL — changes on plugin CRUD\n * - Execution/build queries with date ranges: 2 min TTL — new events arrive continuously\n */\nconst inventoryCache = createCacheService('report:inv:', parseInt(process.env.CACHE_TTL_REPORT_INVENTORY || '300', 10));\nconst timeseriesCache = createCacheService('report:ts:', parseInt(process.env.CACHE_TTL_REPORT_TIMESERIES || '120', 10));\n\n// ─── Types ──────────────────────────────────────────────\n\ninterface ExecutionCount {\n  id: string;\n  project: string;\n  organization: string;\n  pipelineName: string | null;\n  total: number;\n  succeeded: number;\n  failed: number;\n  canceled: number;\n  firstExecution: string | null;\n  lastExecution: string | null;\n}\n\ninterface TimeSeriesEntry {\n  period: string;\n  succeeded: number;\n  failed: number;\n  canceled: number;\n  successPct: number;\n}\n\ninterface DurationStats {\n  id: string;\n  project: string;\n  pipelineName: string | null;\n  avgMs: number;\n  minMs: number;\n  maxMs: number;\n  p95Ms: number;\n  executions: number;\n}\n\ninterface StageFailure {\n  stageName: string;\n  failures: number;\n  total: number;\n  failurePct: number;\n}\n\ninterface StageBottleneck {\n  id: string;\n  pipelineName: string | null;\n  stageName: string;\n  avgMs: number;\n  maxMs: number;\n}\n\ninterface ActionFailure {\n  actionName: string;\n  failures: number;\n  total: number;\n  failurePct: number;\n}\n\ninterface ErrorEntry {\n  errorPattern: string;\n  occurrences: number;\n  affectedPipelines: number;\n  lastSeen: string;\n}\n\ninterface PluginSummary {\n  total: number;\n  active: number;\n  inactive: number;\n  public: number;\n  private: number;\n  uniqueNames: number;\n}\n\ninterface TypeComputeDistribution {\n  pluginType: string;\n  computeType: string;\n  count: number;\n}\n\ninterface VersionCount {\n  name: string;\n  versionCount: number;\n  latestVersion: string;\n  hasDefault: boolean;\n}\n\ninterface BuildTimeSeriesEntry {\n  period: string;\n  succeeded: number;\n  failed: number;\n  successPct: number;\n}\n\ninterface BuildDuration {\n  pluginName: string;\n  avgMs: number;\n  maxMs: number;\n  builds: number;\n}\n\ninterface BuildFailure {\n  pluginName: string;\n  errorMessage: string;\n  occurrences: number;\n  lastSeen: string;\n}\n\n// ─── Service ────────────────────────────────────────────\n\n/**\n * Read-only reporting service for pipeline execution and plugin inventory aggregations.\n * Does not extend CrudService — reports are aggregate queries, not entity CRUD.\n *\n * All queries are cached in-memory to avoid repeated expensive SQL aggregations:\n * - Inventory queries (plugin summary/distribution/versions): 5 min TTL\n * - Timeseries queries (execution/build metrics with date ranges): 2 min TTL\n */\nexport class ReportingService {\n\n  /** Invalidate all cached reports for an org (call after event ingest). */\n  async invalidateOrg(orgId: string): Promise<void> {\n    await Promise.all([\n      inventoryCache.invalidatePattern(`${orgId}:*`),\n      timeseriesCache.invalidatePattern(`${orgId}:*`),\n    ]);\n  }\n\n  // ── Category 1: Pipeline Execution & Performance ──\n\n  /** 1.1 Execution count per pipeline with status breakdown. */\n  async getExecutionCount(orgId: string): Promise<ExecutionCount[]> {\n    return timeseriesCache.getOrSet(`${orgId}:exec-count`, () =>\n      db.execute(sql`\n        SELECT\n          p.id, p.project, p.organization, p.pipeline_name,\n          COUNT(*)::int AS total,\n          COUNT(*) FILTER (WHERE e.status = 'SUCCEEDED')::int AS succeeded,\n          COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failed,\n          COUNT(*) FILTER (WHERE e.status = 'CANCELED')::int AS canceled,\n          MIN(e.started_at)::text AS first_execution,\n          MAX(e.started_at)::text AS last_execution\n        FROM ${schema.pipeline} p\n        JOIN ${schema.pipelineEvent} e ON e.pipeline_id = p.id\n          AND e.event_type = 'PIPELINE' AND e.status != 'STARTED'\n        WHERE p.org_id = ${orgId} AND p.is_active = true\n        GROUP BY p.id\n        ORDER BY total DESC\n      `).then(r => sqlRows<ExecutionCount>(r)),\n    );\n  }\n\n  /** 1.2 Success rate over time for an org. */\n  async getSuccessRate(orgId: string, interval: string, from: string, to: string): Promise<TimeSeriesEntry[]> {\n    return timeseriesCache.getOrSet(`${orgId}:success-rate:${interval}:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          DATE_TRUNC(${interval}, e.started_at)::text AS period,\n          COUNT(*) FILTER (WHERE e.status = 'SUCCEEDED')::int AS succeeded,\n          COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failed,\n          COUNT(*) FILTER (WHERE e.status = 'CANCELED')::int AS canceled,\n          ROUND(COUNT(*) FILTER (WHERE e.status = 'SUCCEEDED')::numeric\n            / NULLIF(COUNT(*), 0) * 100, 1)::float AS success_pct\n        FROM ${schema.pipelineEvent} e\n        JOIN ${schema.pipeline} p ON p.id = e.pipeline_id\n        WHERE p.org_id = ${orgId} AND e.event_type = 'PIPELINE'\n          AND e.status IN ('SUCCEEDED', 'FAILED', 'CANCELED')\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY period ORDER BY period\n      `).then(r => sqlRows<TimeSeriesEntry>(r)),\n    );\n  }\n\n  /** 1.3 Average duration per pipeline. */\n  async getAverageDuration(orgId: string, from: string, to: string): Promise<DurationStats[]> {\n    return timeseriesCache.getOrSet(`${orgId}:avg-duration:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          p.id, p.project, p.pipeline_name,\n          AVG(e.duration_ms)::int AS avg_ms,\n          MIN(e.duration_ms)::int AS min_ms,\n          MAX(e.duration_ms)::int AS max_ms,\n          PERCENTILE_CONT(0.95) WITHIN GROUP (ORDER BY e.duration_ms)::int AS p95_ms,\n          COUNT(*)::int AS executions\n        FROM ${schema.pipelineEvent} e\n        JOIN ${schema.pipeline} p ON p.id = e.pipeline_id\n        WHERE p.org_id = ${orgId} AND e.event_type = 'PIPELINE' AND e.duration_ms IS NOT NULL\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY p.id ORDER BY avg_ms DESC\n      `).then(r => sqlRows<DurationStats>(r)),\n    );\n  }\n\n  /** 1.5 Stage failure heatmap — which stages fail most. */\n  async getStageFailures(orgId: string, from: string, to: string): Promise<StageFailure[]> {\n    return timeseriesCache.getOrSet(`${orgId}:stage-failures:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          e.stage_name,\n          COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failures,\n          COUNT(*)::int AS total,\n          ROUND(COUNT(*) FILTER (WHERE e.status = 'FAILED')::numeric\n            / NULLIF(COUNT(*), 0) * 100, 1)::float AS failure_pct\n        FROM ${schema.pipelineEvent} e\n        JOIN ${schema.pipeline} p ON p.id = e.pipeline_id\n        WHERE p.org_id = ${orgId} AND e.event_type = 'STAGE' AND e.stage_name IS NOT NULL\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY e.stage_name ORDER BY failures DESC\n      `).then(r => sqlRows<StageFailure>(r)),\n    );\n  }\n\n  /** 1.6 Stage bottlenecks — slowest stages per pipeline. */\n  async getStageBottlenecks(orgId: string, from: string, to: string): Promise<StageBottleneck[]> {\n    return timeseriesCache.getOrSet(`${orgId}:stage-bottlenecks:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          p.id, p.pipeline_name, e.stage_name,\n          AVG(e.duration_ms)::int AS avg_ms,\n          MAX(e.duration_ms)::int AS max_ms\n        FROM ${schema.pipelineEvent} e\n        JOIN ${schema.pipeline} p ON p.id = e.pipeline_id\n        WHERE p.org_id = ${orgId} AND e.event_type = 'STAGE' AND e.duration_ms IS NOT NULL\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY p.id, e.stage_name ORDER BY avg_ms DESC\n      `).then(r => sqlRows<StageBottleneck>(r)),\n    );\n  }\n\n  /** 1.7 Action failure rate — which plugin steps fail most. */\n  async getActionFailures(orgId: string, from: string, to: string): Promise<ActionFailure[]> {\n    return timeseriesCache.getOrSet(`${orgId}:action-failures:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          e.action_name,\n          COUNT(*) FILTER (WHERE e.status = 'FAILED')::int AS failures,\n          COUNT(*)::int AS total,\n          ROUND(COUNT(*) FILTER (WHERE e.status = 'FAILED')::numeric\n            / NULLIF(COUNT(*), 0) * 100, 1)::float AS failure_pct\n        FROM ${schema.pipelineEvent} e\n        JOIN ${schema.pipeline} p ON p.id = e.pipeline_id\n        WHERE p.org_id = ${orgId} AND e.event_type = 'ACTION' AND e.action_name IS NOT NULL\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY e.action_name ORDER BY failures DESC\n      `).then(r => sqlRows<ActionFailure>(r)),\n    );\n  }\n\n  /** 1.8 Error categorization — group failure messages. */\n  async getErrors(orgId: string, from: string, to: string, limit: number = 20): Promise<ErrorEntry[]> {\n    return timeseriesCache.getOrSet(`${orgId}:errors:${from}:${to}:${limit}`, () =>\n      db.execute(sql`\n        SELECT\n          SUBSTRING(e.error_message FROM 1 FOR 200) AS error_pattern,\n          COUNT(*)::int AS occurrences,\n          COUNT(DISTINCT e.pipeline_id)::int AS affected_pipelines,\n          MAX(e.started_at)::text AS last_seen\n        FROM ${schema.pipelineEvent} e\n        JOIN ${schema.pipeline} p ON p.id = e.pipeline_id\n        WHERE p.org_id = ${orgId} AND e.status = 'FAILED' AND e.error_message IS NOT NULL\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY error_pattern ORDER BY occurrences DESC\n        LIMIT ${limit}\n      `).then(r => sqlRows<ErrorEntry>(r)),\n    );\n  }\n\n  // ── Category 2: Plugin Inventory & Builds ──\n\n  /** 2.1 Plugin summary — counts and breakdowns. */\n  async getPluginSummary(orgId: string): Promise<PluginSummary> {\n    return inventoryCache.getOrSet(`${orgId}:plugin-summary`, async () => {\n      const rows = await db.execute(sql`\n        SELECT\n          COUNT(*)::int AS total,\n          COUNT(*) FILTER (WHERE ${schema.plugin.isActive})::int AS active,\n          COUNT(*) FILTER (WHERE NOT ${schema.plugin.isActive})::int AS inactive,\n          COUNT(*) FILTER (WHERE ${schema.plugin.accessModifier} = 'public')::int AS public,\n          COUNT(*) FILTER (WHERE ${schema.plugin.accessModifier} = 'private')::int AS private,\n          COUNT(DISTINCT ${schema.plugin.name})::int AS unique_names\n        FROM ${schema.plugin}\n        WHERE ${schema.plugin.orgId} = ${orgId}\n      `);\n      return (sqlRows<PluginSummary>(rows)[0] || { total: 0, active: 0, inactive: 0, public: 0, private: 0, uniqueNames: 0 });\n    });\n  }\n\n  /** 2.2 Type & compute distribution. */\n  async getPluginDistribution(orgId: string): Promise<TypeComputeDistribution[]> {\n    return inventoryCache.getOrSet(`${orgId}:plugin-distribution`, () =>\n      db.execute(sql`\n        SELECT\n          ${schema.plugin.pluginType} AS plugin_type,\n          ${schema.plugin.computeType} AS compute_type,\n          COUNT(*)::int AS count\n        FROM ${schema.plugin}\n        WHERE ${schema.plugin.orgId} = ${orgId} AND ${schema.plugin.isActive} = true\n        GROUP BY ${schema.plugin.pluginType}, ${schema.plugin.computeType}\n        ORDER BY count DESC\n      `).then(r => sqlRows<TypeComputeDistribution>(r)),\n    );\n  }\n\n  /** 2.3 Version counts per plugin name. */\n  async getPluginVersions(orgId: string): Promise<VersionCount[]> {\n    return inventoryCache.getOrSet(`${orgId}:plugin-versions`, () =>\n      db.execute(sql`\n        SELECT\n          ${schema.plugin.name},\n          COUNT(*)::int AS version_count,\n          MAX(${schema.plugin.version}) AS latest_version,\n          bool_or(${schema.plugin.isDefault}) AS has_default\n        FROM ${schema.plugin}\n        WHERE ${schema.plugin.orgId} = ${orgId} AND ${schema.plugin.isActive} = true\n        GROUP BY ${schema.plugin.name}\n        ORDER BY version_count DESC\n      `).then(r => sqlRows<VersionCount>(r)),\n    );\n  }\n\n  /** 2.4 Build success rate over time. */\n  async getBuildSuccessRate(orgId: string, interval: string, from: string, to: string): Promise<BuildTimeSeriesEntry[]> {\n    return timeseriesCache.getOrSet(`${orgId}:build-success:${interval}:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          DATE_TRUNC(${interval}, e.started_at)::text AS period,\n          COUNT(*) FILTER (WHERE e.status = 'completed')::int AS succeeded,\n          COUNT(*) FILTER (WHERE e.status = 'failed')::int AS failed,\n          ROUND(COUNT(*) FILTER (WHERE e.status = 'completed')::numeric\n            / NULLIF(COUNT(*), 0) * 100, 1)::float AS success_pct\n        FROM ${schema.pipelineEvent} e\n        WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build'\n          AND e.status IN ('completed', 'failed')\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY period ORDER BY period\n      `).then(r => sqlRows<BuildTimeSeriesEntry>(r)),\n    );\n  }\n\n  /** 2.5 Build duration per plugin. */\n  async getBuildDuration(orgId: string, from: string, to: string): Promise<BuildDuration[]> {\n    return timeseriesCache.getOrSet(`${orgId}:build-duration:${from}:${to}`, () =>\n      db.execute(sql`\n        SELECT\n          e.detail->>'pluginName' AS plugin_name,\n          AVG(e.duration_ms)::int AS avg_ms,\n          MAX(e.duration_ms)::int AS max_ms,\n          COUNT(*)::int AS builds\n        FROM ${schema.pipelineEvent} e\n        WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build' AND e.duration_ms IS NOT NULL\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY plugin_name ORDER BY avg_ms DESC\n      `).then(r => sqlRows<BuildDuration>(r)),\n    );\n  }\n\n  /** 2.6 Build failures — top error messages. */\n  async getBuildFailures(orgId: string, from: string, to: string, limit: number = 20): Promise<BuildFailure[]> {\n    return timeseriesCache.getOrSet(`${orgId}:build-failures:${from}:${to}:${limit}`, () =>\n      db.execute(sql`\n        SELECT\n          e.detail->>'pluginName' AS plugin_name,\n          e.error_message,\n          COUNT(*)::int AS occurrences,\n          MAX(e.started_at)::text AS last_seen\n        FROM ${schema.pipelineEvent} e\n        WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build' AND e.status = 'failed'\n          AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz\n        GROUP BY plugin_name, e.error_message\n        ORDER BY occurrences DESC\n        LIMIT ${limit}\n      `).then(r => sqlRows<BuildFailure>(r)),\n    );\n  }\n}\n\nexport const reportingService = new ReportingService();\n"]}
|
|
@@ -0,0 +1,235 @@
|
|
|
1
|
+
import { AccessModifier } from '@pipeline-builder/api-core';
|
|
2
|
+
/**
|
|
3
|
+
* Base filter interface containing common filter properties shared across all entity types.
|
|
4
|
+
*
|
|
5
|
+
* @example
|
|
6
|
+
* ```typescript
|
|
7
|
+
* const filter: CommonFilter = {
|
|
8
|
+
* id: '123',
|
|
9
|
+
* orgId: 'my-org',
|
|
10
|
+
* accessModifier: AccessModifier.PUBLIC,
|
|
11
|
+
* isDefault: true,
|
|
12
|
+
* isActive: true
|
|
13
|
+
* };
|
|
14
|
+
* ```
|
|
15
|
+
*/
|
|
16
|
+
export interface CommonFilter {
|
|
17
|
+
/**
|
|
18
|
+
* Unique identifier for the entity
|
|
19
|
+
* Can be a single ID or array of IDs for batch filtering
|
|
20
|
+
*/
|
|
21
|
+
readonly id?: string | string[];
|
|
22
|
+
/**
|
|
23
|
+
* Organization identifier to filter entities by organization
|
|
24
|
+
* Can be a single org ID or array for multi-org filtering
|
|
25
|
+
*/
|
|
26
|
+
readonly orgId?: string | string[];
|
|
27
|
+
/**
|
|
28
|
+
* Access modifier to filter by visibility/permissions
|
|
29
|
+
* Use AccessModifier enum for type safety
|
|
30
|
+
* @see AccessModifier
|
|
31
|
+
*/
|
|
32
|
+
readonly accessModifier?: AccessModifier | string;
|
|
33
|
+
/**
|
|
34
|
+
* Filter by default status
|
|
35
|
+
* - true: Only default entities
|
|
36
|
+
* - false: Only non-default entities
|
|
37
|
+
* - undefined: All entities
|
|
38
|
+
*/
|
|
39
|
+
readonly isDefault?: boolean;
|
|
40
|
+
/**
|
|
41
|
+
* Filter by active/inactive status
|
|
42
|
+
* - true: Only active entities
|
|
43
|
+
* - false: Only inactive entities
|
|
44
|
+
* - undefined: All entities
|
|
45
|
+
*/
|
|
46
|
+
readonly isActive?: boolean;
|
|
47
|
+
/**
|
|
48
|
+
* Number of results to return
|
|
49
|
+
* @minimum 1
|
|
50
|
+
* @maximum 1000
|
|
51
|
+
*/
|
|
52
|
+
readonly limit?: number;
|
|
53
|
+
/**
|
|
54
|
+
* Number of results to skip (for pagination)
|
|
55
|
+
* @minimum 0
|
|
56
|
+
*/
|
|
57
|
+
readonly offset?: number;
|
|
58
|
+
/**
|
|
59
|
+
* Sort field and direction
|
|
60
|
+
* @example "name:asc", "createdAt:desc"
|
|
61
|
+
*/
|
|
62
|
+
readonly sort?: string;
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Filter interface for plugin-specific properties.
|
|
66
|
+
* Extends CommonFilter to include plugin-related filter options.
|
|
67
|
+
*
|
|
68
|
+
* @example
|
|
69
|
+
* ```typescript
|
|
70
|
+
* const filter: PluginFilter = {
|
|
71
|
+
* name: 'nodejs-build',
|
|
72
|
+
* version: '1.0.0',
|
|
73
|
+
* isActive: true
|
|
74
|
+
* };
|
|
75
|
+
* ```
|
|
76
|
+
*/
|
|
77
|
+
export interface PluginFilter extends CommonFilter {
|
|
78
|
+
/**
|
|
79
|
+
* Plugin name to filter by
|
|
80
|
+
*/
|
|
81
|
+
readonly name?: string;
|
|
82
|
+
/**
|
|
83
|
+
* Plugin version to filter by
|
|
84
|
+
* Supports semantic versioning
|
|
85
|
+
* @example "1.0.0", "^2.0.0", "~1.2.3"
|
|
86
|
+
*/
|
|
87
|
+
readonly version?: string;
|
|
88
|
+
/**
|
|
89
|
+
* Docker image tag associated with the plugin
|
|
90
|
+
*/
|
|
91
|
+
readonly imageTag?: string;
|
|
92
|
+
/**
|
|
93
|
+
* Keyword to search within the keywords JSONB array (case-insensitive contains)
|
|
94
|
+
*/
|
|
95
|
+
readonly keyword?: string;
|
|
96
|
+
/**
|
|
97
|
+
* Plugin category to filter by
|
|
98
|
+
* @example "language", "security", "testing"
|
|
99
|
+
*/
|
|
100
|
+
readonly category?: string;
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Filter interface for pipeline-specific properties.
|
|
104
|
+
* Extends CommonFilter to include pipeline-related filter options.
|
|
105
|
+
*
|
|
106
|
+
* @example
|
|
107
|
+
* ```typescript
|
|
108
|
+
* const filter: PipelineFilter = {
|
|
109
|
+
* project: 'my-app',
|
|
110
|
+
* organization: 'my-org',
|
|
111
|
+
* pipelineName: 'my-pipeline',
|
|
112
|
+
* isActive: true
|
|
113
|
+
* };
|
|
114
|
+
* ```
|
|
115
|
+
*/
|
|
116
|
+
export interface PipelineFilter extends CommonFilter {
|
|
117
|
+
/**
|
|
118
|
+
* Project name associated with the pipeline
|
|
119
|
+
*/
|
|
120
|
+
readonly project?: string;
|
|
121
|
+
/**
|
|
122
|
+
* Organization name associated with the pipeline
|
|
123
|
+
*/
|
|
124
|
+
readonly organization?: string;
|
|
125
|
+
/**
|
|
126
|
+
* Pipeline name to filter by
|
|
127
|
+
*/
|
|
128
|
+
readonly pipelineName?: string;
|
|
129
|
+
/**
|
|
130
|
+
* Keyword to search within the keywords JSONB array (case-insensitive contains)
|
|
131
|
+
*/
|
|
132
|
+
readonly keyword?: string;
|
|
133
|
+
}
|
|
134
|
+
/**
|
|
135
|
+
* Filter interface for message-specific properties.
|
|
136
|
+
* Extends CommonFilter to include message-related filter options.
|
|
137
|
+
*/
|
|
138
|
+
export interface MessageFilter extends CommonFilter {
|
|
139
|
+
/**
|
|
140
|
+
* Thread ID to filter by (for fetching thread replies).
|
|
141
|
+
* Pass `null` to filter for root messages only (threadId IS NULL).
|
|
142
|
+
*/
|
|
143
|
+
readonly threadId?: string | null;
|
|
144
|
+
/**
|
|
145
|
+
* Recipient organization ID to filter by
|
|
146
|
+
* Use '*' for broadcast announcements
|
|
147
|
+
*/
|
|
148
|
+
readonly recipientOrgId?: string;
|
|
149
|
+
/**
|
|
150
|
+
* Message type filter
|
|
151
|
+
*/
|
|
152
|
+
readonly messageType?: 'announcement' | 'conversation';
|
|
153
|
+
/**
|
|
154
|
+
* Filter by read status
|
|
155
|
+
*/
|
|
156
|
+
readonly isRead?: boolean;
|
|
157
|
+
/**
|
|
158
|
+
* Filter by priority level
|
|
159
|
+
*/
|
|
160
|
+
readonly priority?: 'normal' | 'high' | 'urgent';
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Filter for compliance policies.
|
|
164
|
+
*/
|
|
165
|
+
export interface CompliancePolicyFilter extends CommonFilter {
|
|
166
|
+
readonly name?: string;
|
|
167
|
+
readonly isTemplate?: boolean;
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Filter for compliance rules.
|
|
171
|
+
*/
|
|
172
|
+
export interface ComplianceRuleFilter extends CommonFilter {
|
|
173
|
+
readonly name?: string;
|
|
174
|
+
readonly policyId?: string;
|
|
175
|
+
readonly target?: 'plugin' | 'pipeline';
|
|
176
|
+
readonly field?: string;
|
|
177
|
+
readonly severity?: 'warning' | 'error' | 'critical';
|
|
178
|
+
readonly scope?: 'org' | 'published';
|
|
179
|
+
readonly tag?: string;
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Filter for compliance rule subscriptions.
|
|
183
|
+
*/
|
|
184
|
+
export interface ComplianceRuleSubscriptionFilter {
|
|
185
|
+
readonly orgId?: string;
|
|
186
|
+
readonly ruleId?: string;
|
|
187
|
+
readonly isActive?: boolean;
|
|
188
|
+
readonly limit?: number;
|
|
189
|
+
readonly offset?: number;
|
|
190
|
+
}
|
|
191
|
+
/**
|
|
192
|
+
* Filter for compliance exemptions.
|
|
193
|
+
*/
|
|
194
|
+
export interface ComplianceExemptionFilter {
|
|
195
|
+
readonly orgId?: string;
|
|
196
|
+
readonly ruleId?: string;
|
|
197
|
+
readonly entityType?: 'plugin' | 'pipeline';
|
|
198
|
+
readonly entityId?: string;
|
|
199
|
+
readonly status?: 'pending' | 'approved' | 'rejected' | 'expired';
|
|
200
|
+
readonly limit?: number;
|
|
201
|
+
readonly offset?: number;
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* Filter for compliance audit log entries.
|
|
205
|
+
*/
|
|
206
|
+
export interface ComplianceAuditFilter {
|
|
207
|
+
readonly orgId?: string;
|
|
208
|
+
readonly target?: 'plugin' | 'pipeline';
|
|
209
|
+
readonly action?: string;
|
|
210
|
+
readonly result?: 'pass' | 'warn' | 'block';
|
|
211
|
+
readonly scanId?: string;
|
|
212
|
+
readonly dateFrom?: string;
|
|
213
|
+
readonly dateTo?: string;
|
|
214
|
+
readonly limit?: number;
|
|
215
|
+
readonly offset?: number;
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Filter for compliance scans.
|
|
219
|
+
*/
|
|
220
|
+
export interface ComplianceScanFilter {
|
|
221
|
+
readonly orgId?: string;
|
|
222
|
+
readonly target?: 'plugin' | 'pipeline' | 'all';
|
|
223
|
+
readonly status?: 'pending' | 'running' | 'completed' | 'failed' | 'cancelled';
|
|
224
|
+
readonly triggeredBy?: 'manual' | 'scheduled' | 'rule-change' | 'rule-dry-run';
|
|
225
|
+
readonly limit?: number;
|
|
226
|
+
readonly offset?: number;
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* Validates message filter properties.
|
|
230
|
+
* Returns a result object with `valid` flag and `errors` array.
|
|
231
|
+
*/
|
|
232
|
+
export declare function validateMessageFilter(filter: Partial<MessageFilter>): {
|
|
233
|
+
valid: boolean;
|
|
234
|
+
errors: string[];
|
|
235
|
+
};
|
|
@@ -0,0 +1,23 @@
|
|
|
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.validateMessageFilter = validateMessageFilter;
|
|
6
|
+
/**
|
|
7
|
+
* Validates message filter properties.
|
|
8
|
+
* Returns a result object with `valid` flag and `errors` array.
|
|
9
|
+
*/
|
|
10
|
+
function validateMessageFilter(filter) {
|
|
11
|
+
const errors = [];
|
|
12
|
+
if (filter.messageType && !['announcement', 'conversation'].includes(filter.messageType)) {
|
|
13
|
+
errors.push(`Invalid messageType: "${filter.messageType}". Must be "announcement" or "conversation"`);
|
|
14
|
+
}
|
|
15
|
+
if (filter.priority && !['normal', 'high', 'urgent'].includes(filter.priority)) {
|
|
16
|
+
errors.push(`Invalid priority: "${filter.priority}". Must be "normal", "high", or "urgent"`);
|
|
17
|
+
}
|
|
18
|
+
if (filter.threadId !== undefined && filter.threadId !== null && typeof filter.threadId !== 'string') {
|
|
19
|
+
errors.push('threadId must be a string UUID or null');
|
|
20
|
+
}
|
|
21
|
+
return { valid: errors.length === 0, errors };
|
|
22
|
+
}
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-filters.js","sourceRoot":"","sources":["../../src/core/query-filters.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC;;AA0QtC,sDAiBC;AArBD;;;GAGG;AACH,SAAgB,qBAAqB,CAAC,MAA8B;IAIlE,MAAM,MAAM,GAAa,EAAE,CAAC;IAE5B,IAAI,MAAM,CAAC,WAAW,IAAI,CAAC,CAAC,cAAc,EAAE,cAAc,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;QACzF,MAAM,CAAC,IAAI,CAAC,yBAAyB,MAAM,CAAC,WAAW,6CAA6C,CAAC,CAAC;IACxG,CAAC;IACD,IAAI,MAAM,CAAC,QAAQ,IAAI,CAAC,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC/E,MAAM,CAAC,IAAI,CAAC,sBAAsB,MAAM,CAAC,QAAQ,0CAA0C,CAAC,CAAC;IAC/F,CAAC;IACD,IAAI,MAAM,CAAC,QAAQ,KAAK,SAAS,IAAI,MAAM,CAAC,QAAQ,KAAK,IAAI,IAAI,OAAO,MAAM,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;QACrG,MAAM,CAAC,IAAI,CAAC,wCAAwC,CAAC,CAAC;IACxD,CAAC;IAED,OAAO,EAAE,KAAK,EAAE,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;AAChD,CAAC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { AccessModifier } from '@pipeline-builder/api-core';\n\n/**\n * Base filter interface containing common filter properties shared across all entity types.\n *\n * @example\n * ```typescript\n * const filter: CommonFilter = {\n *   id: '123',\n *   orgId: 'my-org',\n *   accessModifier: AccessModifier.PUBLIC,\n *   isDefault: true,\n *   isActive: true\n * };\n * ```\n */\nexport interface CommonFilter {\n  /**\n   * Unique identifier for the entity\n   * Can be a single ID or array of IDs for batch filtering\n   */\n  readonly id?: string | string[];\n\n  /**\n   * Organization identifier to filter entities by organization\n   * Can be a single org ID or array for multi-org filtering\n   */\n  readonly orgId?: string | string[];\n\n  /**\n   * Access modifier to filter by visibility/permissions\n   * Use AccessModifier enum for type safety\n   * @see AccessModifier\n   */\n  readonly accessModifier?: AccessModifier | string;\n\n  /**\n   * Filter by default status\n   * - true: Only default entities\n   * - false: Only non-default entities\n   * - undefined: All entities\n   */\n  readonly isDefault?: boolean;\n\n  /**\n   * Filter by active/inactive status\n   * - true: Only active entities\n   * - false: Only inactive entities\n   * - undefined: All entities\n   */\n  readonly isActive?: boolean;\n\n  /**\n   * Number of results to return\n   * @minimum 1\n   * @maximum 1000\n   */\n  readonly limit?: number;\n\n  /**\n   * Number of results to skip (for pagination)\n   * @minimum 0\n   */\n  readonly offset?: number;\n\n  /**\n   * Sort field and direction\n   * @example \"name:asc\", \"createdAt:desc\"\n   */\n  readonly sort?: string;\n}\n\n/**\n * Filter interface for plugin-specific properties.\n * Extends CommonFilter to include plugin-related filter options.\n *\n * @example\n * ```typescript\n * const filter: PluginFilter = {\n *   name: 'nodejs-build',\n *   version: '1.0.0',\n *   isActive: true\n * };\n * ```\n */\nexport interface PluginFilter extends CommonFilter {\n  /**\n   * Plugin name to filter by\n   */\n  readonly name?: string;\n\n  /**\n   * Plugin version to filter by\n   * Supports semantic versioning\n   * @example \"1.0.0\", \"^2.0.0\", \"~1.2.3\"\n   */\n  readonly version?: string;\n\n  /**\n   * Docker image tag associated with the plugin\n   */\n  readonly imageTag?: string;\n\n  /**\n   * Keyword to search within the keywords JSONB array (case-insensitive contains)\n   */\n  readonly keyword?: string;\n\n  /**\n   * Plugin category to filter by\n   * @example \"language\", \"security\", \"testing\"\n   */\n  readonly category?: string;\n}\n\n/**\n * Filter interface for pipeline-specific properties.\n * Extends CommonFilter to include pipeline-related filter options.\n *\n * @example\n * ```typescript\n * const filter: PipelineFilter = {\n *   project: 'my-app',\n *   organization: 'my-org',\n *   pipelineName: 'my-pipeline',\n *   isActive: true\n * };\n * ```\n */\nexport interface PipelineFilter extends CommonFilter {\n  /**\n   * Project name associated with the pipeline\n   */\n  readonly project?: string;\n\n  /**\n   * Organization name associated with the pipeline\n   */\n  readonly organization?: string;\n\n  /**\n   * Pipeline name to filter by\n   */\n  readonly pipelineName?: string;\n\n  /**\n   * Keyword to search within the keywords JSONB array (case-insensitive contains)\n   */\n  readonly keyword?: string;\n}\n\n/**\n * Filter interface for message-specific properties.\n * Extends CommonFilter to include message-related filter options.\n */\nexport interface MessageFilter extends CommonFilter {\n  /**\n   * Thread ID to filter by (for fetching thread replies).\n   * Pass `null` to filter for root messages only (threadId IS NULL).\n   */\n  readonly threadId?: string | null;\n\n  /**\n   * Recipient organization ID to filter by\n   * Use '*' for broadcast announcements\n   */\n  readonly recipientOrgId?: string;\n\n  /**\n   * Message type filter\n   */\n  readonly messageType?: 'announcement' | 'conversation';\n\n  /**\n   * Filter by read status\n   */\n  readonly isRead?: boolean;\n\n  /**\n   * Filter by priority level\n   */\n  readonly priority?: 'normal' | 'high' | 'urgent';\n}\n\n// ========================================\n// Compliance Filters\n// ========================================\n\n/**\n * Filter for compliance policies.\n */\nexport interface CompliancePolicyFilter extends CommonFilter {\n  readonly name?: string;\n  readonly isTemplate?: boolean;\n}\n\n/**\n * Filter for compliance rules.\n */\nexport interface ComplianceRuleFilter extends CommonFilter {\n  readonly name?: string;\n  readonly policyId?: string;\n  readonly target?: 'plugin' | 'pipeline';\n  readonly field?: string;\n  readonly severity?: 'warning' | 'error' | 'critical';\n  readonly scope?: 'org' | 'published';\n  readonly tag?: string;\n}\n\n/**\n * Filter for compliance rule subscriptions.\n */\nexport interface ComplianceRuleSubscriptionFilter {\n  readonly orgId?: string;\n  readonly ruleId?: string;\n  readonly isActive?: boolean;\n  readonly limit?: number;\n  readonly offset?: number;\n}\n\n/**\n * Filter for compliance exemptions.\n */\nexport interface ComplianceExemptionFilter {\n  readonly orgId?: string;\n  readonly ruleId?: string;\n  readonly entityType?: 'plugin' | 'pipeline';\n  readonly entityId?: string;\n  readonly status?: 'pending' | 'approved' | 'rejected' | 'expired';\n  readonly limit?: number;\n  readonly offset?: number;\n}\n\n/**\n * Filter for compliance audit log entries.\n */\nexport interface ComplianceAuditFilter {\n  readonly orgId?: string;\n  readonly target?: 'plugin' | 'pipeline';\n  readonly action?: string;\n  readonly result?: 'pass' | 'warn' | 'block';\n  readonly scanId?: string;\n  readonly dateFrom?: string;\n  readonly dateTo?: string;\n  readonly limit?: number;\n  readonly offset?: number;\n}\n\n/**\n * Filter for compliance scans.\n */\nexport interface ComplianceScanFilter {\n  readonly orgId?: string;\n  readonly target?: 'plugin' | 'pipeline' | 'all';\n  readonly status?: 'pending' | 'running' | 'completed' | 'failed' | 'cancelled';\n  readonly triggeredBy?: 'manual' | 'scheduled' | 'rule-change' | 'rule-dry-run';\n  readonly limit?: number;\n  readonly offset?: number;\n}\n\n/**\n * Validates message filter properties.\n * Returns a result object with `valid` flag and `errors` array.\n */\nexport function validateMessageFilter(filter: Partial<MessageFilter>): {\n  valid: boolean;\n  errors: string[];\n} {\n  const errors: string[] = [];\n\n  if (filter.messageType && !['announcement', 'conversation'].includes(filter.messageType)) {\n    errors.push(`Invalid messageType: \"${filter.messageType}\". Must be \"announcement\" or \"conversation\"`);\n  }\n  if (filter.priority && !['normal', 'high', 'urgent'].includes(filter.priority)) {\n    errors.push(`Invalid priority: \"${filter.priority}\". Must be \"normal\", \"high\", or \"urgent\"`);\n  }\n  if (filter.threadId !== undefined && filter.threadId !== null && typeof filter.threadId !== 'string') {\n    errors.push('threadId must be a string UUID or null');\n  }\n\n  return { valid: errors.length === 0, errors };\n}\n"]}
|