@pipeline-builder/pipeline-data 3.3.33 → 3.4.1

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.
@@ -89,6 +89,26 @@ interface BuildFailure {
89
89
  occurrences: number;
90
90
  lastSeen: string;
91
91
  }
92
+ /** Event payload accepted by `ReportingService.ingestEvents`. Mirrors the route's Zod shape. */
93
+ export interface IngestEvent {
94
+ pipelineArn: string;
95
+ eventSource: 'codepipeline' | 'codebuild' | 'plugin-build';
96
+ eventType: 'PIPELINE' | 'STAGE' | 'ACTION' | 'BUILD';
97
+ status: string;
98
+ executionId?: string;
99
+ stageName?: string;
100
+ actionName?: string;
101
+ startedAt?: string;
102
+ completedAt?: string;
103
+ durationMs?: number;
104
+ detail?: Record<string, unknown>;
105
+ }
106
+ /** Counts + the (possibly truncated) list of unregistered ARNs the caller can log. */
107
+ export interface IngestResult {
108
+ inserted: number;
109
+ skipped: number;
110
+ unregisteredArns: string[];
111
+ }
92
112
  /**
93
113
  * Read-only reporting service for pipeline execution and plugin inventory aggregations.
94
114
  * Does not extend CrudService — reports are aggregate queries, not entity CRUD.
@@ -100,6 +120,16 @@ interface BuildFailure {
100
120
  export declare class ReportingService {
101
121
  /** Invalidate all cached reports for an org (call after event ingest). */
102
122
  invalidateOrg(orgId: string): Promise<void>;
123
+ /**
124
+ * Resolve incoming events against the pipeline registry, batch-insert the
125
+ * matched ones, and invalidate reporting caches for affected orgs.
126
+ * Events for unregistered pipeline ARNs are dropped (and logged at WARN
127
+ * with sample ARNs so an operator can see when EventBridge is delivering
128
+ * events for pipelines that haven't called POST /pipelines/registry yet).
129
+ *
130
+ * Returns counts + a sample of unregistered ARNs for observability.
131
+ */
132
+ ingestEvents(events: IngestEvent[]): Promise<IngestResult>;
103
133
  /** 1.1 Execution count per pipeline with status breakdown. */
104
134
  getExecutionCount(orgId: string): Promise<ExecutionCount[]>;
105
135
  /** 1.2 Success rate over time for an org. */
@@ -8,8 +8,7 @@ const drizzle_orm_1 = require("drizzle-orm");
8
8
  const crud_service_1 = require("./crud-service");
9
9
  const drizzle_schema_1 = require("../database/drizzle-schema");
10
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);
11
+ const logger = (0, api_core_1.createLogger)('reporting-service');
13
12
  /**
14
13
  * Cache for reporting aggregations. Two tiers:
15
14
  * - Inventory queries (plugin summary/distribution/versions): 5 min TTL — changes on plugin CRUD
@@ -34,6 +33,66 @@ class ReportingService {
34
33
  timeseriesCache.invalidatePattern(`${orgId}:*`),
35
34
  ]);
36
35
  }
36
+ /**
37
+ * Resolve incoming events against the pipeline registry, batch-insert the
38
+ * matched ones, and invalidate reporting caches for affected orgs.
39
+ * Events for unregistered pipeline ARNs are dropped (and logged at WARN
40
+ * with sample ARNs so an operator can see when EventBridge is delivering
41
+ * events for pipelines that haven't called POST /pipelines/registry yet).
42
+ *
43
+ * Returns counts + a sample of unregistered ARNs for observability.
44
+ */
45
+ async ingestEvents(events) {
46
+ // Batch-resolve all unique ARNs in one query
47
+ const uniqueArns = [...new Set(events.map(e => e.pipelineArn))];
48
+ const registryRows = await postgres_connection_1.db
49
+ .select({
50
+ pipelineId: drizzle_schema_1.schema.pipelineRegistry.pipelineId,
51
+ orgId: drizzle_schema_1.schema.pipelineRegistry.orgId,
52
+ pipelineArn: drizzle_schema_1.schema.pipelineRegistry.pipelineArn,
53
+ })
54
+ .from(drizzle_schema_1.schema.pipelineRegistry)
55
+ .where((0, drizzle_orm_1.inArray)(drizzle_schema_1.schema.pipelineRegistry.pipelineArn, uniqueArns));
56
+ const arnMap = new Map(registryRows.map(r => [r.pipelineArn, r]));
57
+ // Build insert batch (skip unregistered ARNs)
58
+ const rows = [];
59
+ let skipped = 0;
60
+ const unregisteredArns = [];
61
+ for (const event of events) {
62
+ const registry = arnMap.get(event.pipelineArn);
63
+ if (!registry) {
64
+ skipped++;
65
+ unregisteredArns.push(event.pipelineArn);
66
+ continue;
67
+ }
68
+ rows.push({
69
+ pipelineId: registry.pipelineId,
70
+ orgId: registry.orgId,
71
+ eventSource: event.eventSource,
72
+ eventType: event.eventType,
73
+ status: event.status,
74
+ pipelineArn: (0, api_core_1.hashAccountInArn)(event.pipelineArn),
75
+ executionId: event.executionId,
76
+ stageName: event.stageName,
77
+ actionName: event.actionName,
78
+ startedAt: event.startedAt ? new Date(event.startedAt) : undefined,
79
+ completedAt: event.completedAt ? new Date(event.completedAt) : undefined,
80
+ durationMs: event.durationMs,
81
+ detail: event.detail,
82
+ });
83
+ }
84
+ if (rows.length > 0) {
85
+ await postgres_connection_1.db.insert(drizzle_schema_1.schema.pipelineEvent).values(rows);
86
+ // Per-org failures are logged but don't fail the batch — `Promise.all`
87
+ // so the caller doesn't return before invalidation completes,
88
+ // otherwise dashboards can serve stale data right after ingest.
89
+ const affectedOrgs = [...new Set(rows.map(r => r.orgId))];
90
+ await Promise.all(affectedOrgs.map((org) => this.invalidateOrg(org).catch((err) => {
91
+ logger.warn('Reporting cache invalidation failed', { orgId: org, error: (0, api_core_1.errorMessage)(err) });
92
+ })));
93
+ }
94
+ return { inserted: rows.length, skipped, unregisteredArns };
95
+ }
37
96
  // ── Category 1: Pipeline Execution & Performance ──
38
97
  /** 1.1 Execution count per pipeline with status breakdown. */
39
98
  async getExecutionCount(orgId) {
@@ -52,7 +111,7 @@ class ReportingService {
52
111
  WHERE p.org_id = ${orgId} AND p.is_active = true
53
112
  GROUP BY p.id
54
113
  ORDER BY total DESC
55
- `).then(r => sqlRows(r)));
114
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
56
115
  }
57
116
  /** 1.2 Success rate over time for an org. */
58
117
  async getSuccessRate(orgId, interval, from, to) {
@@ -70,7 +129,7 @@ class ReportingService {
70
129
  AND e.status IN ('SUCCEEDED', 'FAILED', 'CANCELED')
71
130
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
72
131
  GROUP BY period ORDER BY period
73
- `).then(r => sqlRows(r)));
132
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
74
133
  }
75
134
  /** 1.3 Average duration per pipeline. */
76
135
  async getAverageDuration(orgId, from, to) {
@@ -87,7 +146,7 @@ class ReportingService {
87
146
  WHERE p.org_id = ${orgId} AND e.event_type = 'PIPELINE' AND e.duration_ms IS NOT NULL
88
147
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
89
148
  GROUP BY p.id ORDER BY avg_ms DESC
90
- `).then(r => sqlRows(r)));
149
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
91
150
  }
92
151
  /** 1.5 Stage failure heatmap — which stages fail most. */
93
152
  async getStageFailures(orgId, from, to) {
@@ -103,7 +162,7 @@ class ReportingService {
103
162
  WHERE p.org_id = ${orgId} AND e.event_type = 'STAGE' AND e.stage_name IS NOT NULL
104
163
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
105
164
  GROUP BY e.stage_name ORDER BY failures DESC
106
- `).then(r => sqlRows(r)));
165
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
107
166
  }
108
167
  /** 1.6 Stage bottlenecks — slowest stages per pipeline. */
109
168
  async getStageBottlenecks(orgId, from, to) {
@@ -117,7 +176,7 @@ class ReportingService {
117
176
  WHERE p.org_id = ${orgId} AND e.event_type = 'STAGE' AND e.duration_ms IS NOT NULL
118
177
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
119
178
  GROUP BY p.id, e.stage_name ORDER BY avg_ms DESC
120
- `).then(r => sqlRows(r)));
179
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
121
180
  }
122
181
  /** 1.7 Action failure rate — which plugin steps fail most. */
123
182
  async getActionFailures(orgId, from, to) {
@@ -133,7 +192,7 @@ class ReportingService {
133
192
  WHERE p.org_id = ${orgId} AND e.event_type = 'ACTION' AND e.action_name IS NOT NULL
134
193
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
135
194
  GROUP BY e.action_name ORDER BY failures DESC
136
- `).then(r => sqlRows(r)));
195
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
137
196
  }
138
197
  /** 1.8 Error categorization — group failure messages. */
139
198
  async getErrors(orgId, from, to, limit = 20) {
@@ -149,7 +208,7 @@ class ReportingService {
149
208
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
150
209
  GROUP BY error_pattern ORDER BY occurrences DESC
151
210
  LIMIT ${limit}
152
- `).then(r => sqlRows(r)));
211
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
153
212
  }
154
213
  // ── Category 2: Plugin Inventory & Builds ──
155
214
  /** 2.1 Plugin summary — counts and breakdowns. */
@@ -166,7 +225,7 @@ class ReportingService {
166
225
  FROM ${drizzle_schema_1.schema.plugin}
167
226
  WHERE ${drizzle_schema_1.schema.plugin.orgId} = ${orgId}
168
227
  `);
169
- return (sqlRows(rows)[0] || { total: 0, active: 0, inactive: 0, public: 0, private: 0, uniqueNames: 0 });
228
+ return ((0, crud_service_1.drizzleRows)(rows.rows)[0] || { total: 0, active: 0, inactive: 0, public: 0, private: 0, uniqueNames: 0 });
170
229
  });
171
230
  }
172
231
  /** 2.2 Type & compute distribution. */
@@ -180,7 +239,7 @@ class ReportingService {
180
239
  WHERE ${drizzle_schema_1.schema.plugin.orgId} = ${orgId} AND ${drizzle_schema_1.schema.plugin.isActive} = true
181
240
  GROUP BY ${drizzle_schema_1.schema.plugin.pluginType}, ${drizzle_schema_1.schema.plugin.computeType}
182
241
  ORDER BY count DESC
183
- `).then(r => sqlRows(r)));
242
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
184
243
  }
185
244
  /** 2.3 Version counts per plugin name. */
186
245
  async getPluginVersions(orgId) {
@@ -194,7 +253,7 @@ class ReportingService {
194
253
  WHERE ${drizzle_schema_1.schema.plugin.orgId} = ${orgId} AND ${drizzle_schema_1.schema.plugin.isActive} = true
195
254
  GROUP BY ${drizzle_schema_1.schema.plugin.name}
196
255
  ORDER BY version_count DESC
197
- `).then(r => sqlRows(r)));
256
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
198
257
  }
199
258
  /** 2.4 Build success rate over time. */
200
259
  async getBuildSuccessRate(orgId, interval, from, to) {
@@ -210,7 +269,7 @@ class ReportingService {
210
269
  AND e.status IN ('completed', 'failed')
211
270
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
212
271
  GROUP BY period ORDER BY period
213
- `).then(r => sqlRows(r)));
272
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
214
273
  }
215
274
  /** 2.5 Build duration per plugin. */
216
275
  async getBuildDuration(orgId, from, to) {
@@ -224,7 +283,7 @@ class ReportingService {
224
283
  WHERE e.org_id = ${orgId} AND e.event_source = 'plugin-build' AND e.duration_ms IS NOT NULL
225
284
  AND e.started_at >= ${from}::timestamptz AND e.started_at <= ${to}::timestamptz
226
285
  GROUP BY plugin_name ORDER BY avg_ms DESC
227
- `).then(r => sqlRows(r)));
286
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
228
287
  }
229
288
  /** 2.6 Build failures — top error messages. */
230
289
  async getBuildFailures(orgId, from, to, limit = 20) {
@@ -240,9 +299,9 @@ class ReportingService {
240
299
  GROUP BY plugin_name, e.error_message
241
300
  ORDER BY occurrences DESC
242
301
  LIMIT ${limit}
243
- `).then(r => sqlRows(r)));
302
+ `).then(r => (0, crud_service_1.drizzleRows)(r.rows)));
244
303
  }
245
304
  }
246
305
  exports.ReportingService = ReportingService;
247
306
  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"]}
307
+ //# 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, createLogger, errorMessage, hashAccountInArn } from '@pipeline-builder/api-core';\nimport { inArray, sql } from 'drizzle-orm';\nimport { drizzleRows } from './crud-service';\nimport { schema } from '../database/drizzle-schema';\nimport { db } from '../database/postgres-connection';\n\nconst logger = createLogger('reporting-service');\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/** Event payload accepted by `ReportingService.ingestEvents`. Mirrors the route's Zod shape. */\nexport interface IngestEvent {\n  pipelineArn: string;\n  eventSource: 'codepipeline' | 'codebuild' | 'plugin-build';\n  eventType: 'PIPELINE' | 'STAGE' | 'ACTION' | 'BUILD';\n  status: string;\n  executionId?: string;\n  stageName?: string;\n  actionName?: string;\n  startedAt?: string;\n  completedAt?: string;\n  durationMs?: number;\n  detail?: Record<string, unknown>;\n}\n\n/** Counts + the (possibly truncated) list of unregistered ARNs the caller can log. */\nexport interface IngestResult {\n  inserted: number;\n  skipped: number;\n  unregisteredArns: 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  /**\n   * Resolve incoming events against the pipeline registry, batch-insert the\n   * matched ones, and invalidate reporting caches for affected orgs.\n   * Events for unregistered pipeline ARNs are dropped (and logged at WARN\n   * with sample ARNs so an operator can see when EventBridge is delivering\n   * events for pipelines that haven't called POST /pipelines/registry yet).\n   *\n   * Returns counts + a sample of unregistered ARNs for observability.\n   */\n  async ingestEvents(events: IngestEvent[]): Promise<IngestResult> {\n    // Batch-resolve all unique ARNs in one query\n    const uniqueArns = [...new Set(events.map(e => e.pipelineArn))];\n    const registryRows = await db\n      .select({\n        pipelineId: schema.pipelineRegistry.pipelineId,\n        orgId: schema.pipelineRegistry.orgId,\n        pipelineArn: schema.pipelineRegistry.pipelineArn,\n      })\n      .from(schema.pipelineRegistry)\n      .where(inArray(schema.pipelineRegistry.pipelineArn, uniqueArns));\n\n    const arnMap = new Map(registryRows.map(r => [r.pipelineArn, r]));\n\n    // Build insert batch (skip unregistered ARNs)\n    const rows: Array<typeof schema.pipelineEvent.$inferInsert> = [];\n    let skipped = 0;\n    const unregisteredArns: string[] = [];\n\n    for (const event of events) {\n      const registry = arnMap.get(event.pipelineArn);\n      if (!registry) {\n        skipped++;\n        unregisteredArns.push(event.pipelineArn);\n        continue;\n      }\n\n      rows.push({\n        pipelineId: registry.pipelineId,\n        orgId: registry.orgId,\n        eventSource: event.eventSource,\n        eventType: event.eventType,\n        status: event.status,\n        pipelineArn: hashAccountInArn(event.pipelineArn),\n        executionId: event.executionId,\n        stageName: event.stageName,\n        actionName: event.actionName,\n        startedAt: event.startedAt ? new Date(event.startedAt) : undefined,\n        completedAt: event.completedAt ? new Date(event.completedAt) : undefined,\n        durationMs: event.durationMs,\n        detail: event.detail,\n      });\n    }\n\n    if (rows.length > 0) {\n      await db.insert(schema.pipelineEvent).values(rows);\n\n      // Per-org failures are logged but don't fail the batch — `Promise.all`\n      // so the caller doesn't return before invalidation completes,\n      // otherwise dashboards can serve stale data right after ingest.\n      const affectedOrgs = [...new Set(rows.map(r => r.orgId))];\n      await Promise.all(affectedOrgs.map((org) =>\n        this.invalidateOrg(org).catch((err) => {\n          logger.warn('Reporting cache invalidation failed', { orgId: org, error: errorMessage(err) });\n        }),\n      ));\n    }\n\n    return { inserted: rows.length, skipped, unregisteredArns };\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 => drizzleRows<ExecutionCount>(r.rows)),\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 => drizzleRows<TimeSeriesEntry>(r.rows)),\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 => drizzleRows<DurationStats>(r.rows)),\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 => drizzleRows<StageFailure>(r.rows)),\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 => drizzleRows<StageBottleneck>(r.rows)),\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 => drizzleRows<ActionFailure>(r.rows)),\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 => drizzleRows<ErrorEntry>(r.rows)),\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 (drizzleRows<PluginSummary>(rows.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 => drizzleRows<TypeComputeDistribution>(r.rows)),\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 => drizzleRows<VersionCount>(r.rows)),\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 => drizzleRows<BuildTimeSeriesEntry>(r.rows)),\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 => drizzleRows<BuildDuration>(r.rows)),\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 => drizzleRows<BuildFailure>(r.rows)),\n    );\n  }\n}\n\nexport const reportingService = new ReportingService();\n"]}
package/package.json CHANGED
@@ -22,7 +22,7 @@
22
22
  "typescript": "5.9.3"
23
23
  },
24
24
  "dependencies": {
25
- "@pipeline-builder/api-core": "3.3.33",
25
+ "@pipeline-builder/api-core": "3.4.0",
26
26
  "drizzle-orm": "0.45.1",
27
27
  "pg": "8.18.0"
28
28
  },
@@ -64,7 +64,7 @@
64
64
  "main": "lib/index.js",
65
65
  "license": "Apache-2.0",
66
66
  "homepage": "https://mwashburn160.github.io/pipeline-builder/",
67
- "version": "3.3.33",
67
+ "version": "3.4.1",
68
68
  "bugs": {
69
69
  "url": "https://github.com/mwashburn160/pipeline-builder/issues"
70
70
  },