@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.
@@ -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"]}