@pipeline-builder/api-core 3.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +202 -0
- package/README.md +51 -0
- package/lib/constants/ai-providers.d.ts +41 -0
- package/lib/constants/ai-providers.js +88 -0
- package/lib/constants/http-status.d.ts +24 -0
- package/lib/constants/http-status.js +29 -0
- package/lib/constants/index.d.ts +3 -0
- package/lib/constants/index.js +22 -0
- package/lib/constants/time.d.ts +10 -0
- package/lib/constants/time.js +16 -0
- package/lib/errors/app-errors.d.ts +30 -0
- package/lib/errors/app-errors.js +62 -0
- package/lib/errors/index.d.ts +1 -0
- package/lib/errors/index.js +20 -0
- package/lib/helpers/access-helpers.d.ts +40 -0
- package/lib/helpers/access-helpers.js +56 -0
- package/lib/helpers/crud-helpers.d.ts +16 -0
- package/lib/helpers/crud-helpers.js +34 -0
- package/lib/helpers/index.d.ts +4 -0
- package/lib/helpers/index.js +23 -0
- package/lib/helpers/mask-helpers.d.ts +33 -0
- package/lib/helpers/mask-helpers.js +54 -0
- package/lib/helpers/sse-helpers.d.ts +13 -0
- package/lib/helpers/sse-helpers.js +40 -0
- package/lib/index.d.ts +57 -0
- package/lib/index.js +86 -0
- package/lib/middleware/auth.d.ts +50 -0
- package/lib/middleware/auth.js +171 -0
- package/lib/middleware/index.d.ts +1 -0
- package/lib/middleware/index.js +20 -0
- package/lib/openapi/extend-zod.d.ts +1 -0
- package/lib/openapi/extend-zod.js +8 -0
- package/lib/openapi/index.d.ts +2 -0
- package/lib/openapi/index.js +10 -0
- package/lib/openapi/registry.d.ts +17 -0
- package/lib/openapi/registry.js +42 -0
- package/lib/openapi/routes/billing-routes.d.ts +1 -0
- package/lib/openapi/routes/billing-routes.js +69 -0
- package/lib/openapi/routes/index.d.ts +5 -0
- package/lib/openapi/routes/index.js +22 -0
- package/lib/openapi/routes/message-routes.d.ts +1 -0
- package/lib/openapi/routes/message-routes.js +108 -0
- package/lib/openapi/routes/pipeline-routes.d.ts +1 -0
- package/lib/openapi/routes/pipeline-routes.js +90 -0
- package/lib/openapi/routes/plugin-routes.d.ts +1 -0
- package/lib/openapi/routes/plugin-routes.js +99 -0
- package/lib/openapi/routes/quota-routes.d.ts +1 -0
- package/lib/openapi/routes/quota-routes.js +65 -0
- package/lib/openapi/schema-registry.d.ts +25 -0
- package/lib/openapi/schema-registry.js +95 -0
- package/lib/routes/health.d.ts +47 -0
- package/lib/routes/health.js +81 -0
- package/lib/routes/index.d.ts +1 -0
- package/lib/routes/index.js +20 -0
- package/lib/services/admin-audit.d.ts +13 -0
- package/lib/services/admin-audit.js +31 -0
- package/lib/services/cache-service.d.ts +108 -0
- package/lib/services/cache-service.js +212 -0
- package/lib/services/compliance-client.d.ts +46 -0
- package/lib/services/compliance-client.js +102 -0
- package/lib/services/compliance-event-subscriber.d.ts +11 -0
- package/lib/services/compliance-event-subscriber.js +60 -0
- package/lib/services/compliance-queue.d.ts +11 -0
- package/lib/services/compliance-queue.js +38 -0
- package/lib/services/entity-events.d.ts +44 -0
- package/lib/services/entity-events.js +63 -0
- package/lib/services/http-client.d.ts +108 -0
- package/lib/services/http-client.js +285 -0
- package/lib/services/index.d.ts +10 -0
- package/lib/services/index.js +40 -0
- package/lib/services/quota.d.ts +59 -0
- package/lib/services/quota.js +137 -0
- package/lib/services/retry-strategy.d.ts +74 -0
- package/lib/services/retry-strategy.js +127 -0
- package/lib/types/billing.d.ts +47 -0
- package/lib/types/billing.js +5 -0
- package/lib/types/common.d.ts +161 -0
- package/lib/types/common.js +53 -0
- package/lib/types/error-codes.d.ts +38 -0
- package/lib/types/error-codes.js +77 -0
- package/lib/types/feature-flags.d.ts +38 -0
- package/lib/types/feature-flags.js +107 -0
- package/lib/types/http.d.ts +37 -0
- package/lib/types/http.js +5 -0
- package/lib/types/index.d.ts +7 -0
- package/lib/types/index.js +26 -0
- package/lib/types/pipeline.d.ts +70 -0
- package/lib/types/pipeline.js +44 -0
- package/lib/types/quota-tiers.d.ts +23 -0
- package/lib/types/quota-tiers.js +26 -0
- package/lib/utils/alias-resolver.d.ts +16 -0
- package/lib/utils/alias-resolver.js +49 -0
- package/lib/utils/headers.d.ts +18 -0
- package/lib/utils/headers.js +24 -0
- package/lib/utils/identity.d.ts +61 -0
- package/lib/utils/identity.js +75 -0
- package/lib/utils/index.d.ts +7 -0
- package/lib/utils/index.js +26 -0
- package/lib/utils/logger.d.ts +28 -0
- package/lib/utils/logger.js +77 -0
- package/lib/utils/object.d.ts +13 -0
- package/lib/utils/object.js +21 -0
- package/lib/utils/params.d.ts +89 -0
- package/lib/utils/params.js +148 -0
- package/lib/utils/response.d.ts +142 -0
- package/lib/utils/response.js +237 -0
- package/lib/validation/ai-schemas.d.ts +61 -0
- package/lib/validation/ai-schemas.js +81 -0
- package/lib/validation/common-schemas.d.ts +72 -0
- package/lib/validation/common-schemas.js +58 -0
- package/lib/validation/index.d.ts +6 -0
- package/lib/validation/index.js +25 -0
- package/lib/validation/message-schemas.d.ts +79 -0
- package/lib/validation/message-schemas.js +42 -0
- package/lib/validation/middleware.d.ts +60 -0
- package/lib/validation/middleware.js +77 -0
- package/lib/validation/pipeline-schemas.d.ts +135 -0
- package/lib/validation/pipeline-schemas.js +85 -0
- package/lib/validation/plugin-schemas.d.ts +127 -0
- package/lib/validation/plugin-schemas.js +84 -0
- package/openapi.yaml +292 -0
- package/package.json +127 -0
|
@@ -0,0 +1,31 @@
|
|
|
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.logAdminAction = logAdminAction;
|
|
6
|
+
exports.getAuditQueue = getAuditQueue;
|
|
7
|
+
const logger_1 = require("../utils/logger");
|
|
8
|
+
const logger = (0, logger_1.createLogger)('admin-audit');
|
|
9
|
+
const auditQueue = [];
|
|
10
|
+
let flushTimer = null;
|
|
11
|
+
function logAdminAction(entry) {
|
|
12
|
+
logger.info('Admin action', entry);
|
|
13
|
+
auditQueue.push(entry);
|
|
14
|
+
if (!flushTimer) {
|
|
15
|
+
flushTimer = setTimeout(flushAuditQueue, 5000);
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
async function flushAuditQueue() {
|
|
19
|
+
flushTimer = null;
|
|
20
|
+
const batch = auditQueue.splice(0, auditQueue.length);
|
|
21
|
+
if (batch.length === 0)
|
|
22
|
+
return;
|
|
23
|
+
// Batch will be persisted when a DB writer is registered
|
|
24
|
+
for (const entry of batch) {
|
|
25
|
+
logger.debug('Audit entry queued', { action: entry.action, target: entry.targetType });
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
function getAuditQueue() {
|
|
29
|
+
return [...auditQueue];
|
|
30
|
+
}
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWRtaW4tYXVkaXQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VydmljZXMvYWRtaW4tYXVkaXQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7O0FBcUJ0Qyx3Q0FNQztBQVlELHNDQUVDO0FBdkNELDRDQUErQztBQUUvQyxNQUFNLE1BQU0sR0FBRyxJQUFBLHFCQUFZLEVBQUMsYUFBYSxDQUFDLENBQUM7QUFjM0MsTUFBTSxVQUFVLEdBQWlCLEVBQUUsQ0FBQztBQUNwQyxJQUFJLFVBQVUsR0FBeUMsSUFBSSxDQUFDO0FBRTVELFNBQWdCLGNBQWMsQ0FBQyxLQUFpQjtJQUM5QyxNQUFNLENBQUMsSUFBSSxDQUFDLGNBQWMsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNuQyxVQUFVLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3ZCLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztRQUNoQixVQUFVLEdBQUcsVUFBVSxDQUFDLGVBQWUsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUNqRCxDQUFDO0FBQ0gsQ0FBQztBQUVELEtBQUssVUFBVSxlQUFlO0lBQzVCLFVBQVUsR0FBRyxJQUFJLENBQUM7SUFDbEIsTUFBTSxLQUFLLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEVBQUUsVUFBVSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3RELElBQUksS0FBSyxDQUFDLE1BQU0sS0FBSyxDQUFDO1FBQUUsT0FBTztJQUMvQix5REFBeUQ7SUFDekQsS0FBSyxNQUFNLEtBQUssSUFBSSxLQUFLLEVBQUUsQ0FBQztRQUMxQixNQUFNLENBQUMsS0FBSyxDQUFDLG9CQUFvQixFQUFFLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxVQUFVLEVBQUUsQ0FBQyxDQUFDO0lBQ3pGLENBQUM7QUFDSCxDQUFDO0FBRUQsU0FBZ0IsYUFBYTtJQUMzQixPQUFPLENBQUMsR0FBRyxVQUFVLENBQUMsQ0FBQztBQUN6QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IDIwMjYgUGlwZWxpbmUgQnVpbGRlciBDb250cmlidXRvcnNcbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5cbmltcG9ydCB7IGNyZWF0ZUxvZ2dlciB9IGZyb20gJy4uL3V0aWxzL2xvZ2dlcic7XG5cbmNvbnN0IGxvZ2dlciA9IGNyZWF0ZUxvZ2dlcignYWRtaW4tYXVkaXQnKTtcblxuZXhwb3J0IGludGVyZmFjZSBBdWRpdEVudHJ5IHtcbiAgdXNlcklkOiBzdHJpbmc7XG4gIHVzZXJFbWFpbD86IHN0cmluZztcbiAgb3JnSWQ/OiBzdHJpbmc7XG4gIGFjdGlvbjogc3RyaW5nO1xuICB0YXJnZXRUeXBlOiBzdHJpbmc7XG4gIHRhcmdldElkPzogc3RyaW5nO1xuICB0YXJnZXROYW1lPzogc3RyaW5nO1xuICBkZXRhaWw/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgaXBBZGRyZXNzPzogc3RyaW5nO1xufVxuXG5jb25zdCBhdWRpdFF1ZXVlOiBBdWRpdEVudHJ5W10gPSBbXTtcbmxldCBmbHVzaFRpbWVyOiBSZXR1cm5UeXBlPHR5cGVvZiBzZXRUaW1lb3V0PiB8IG51bGwgPSBudWxsO1xuXG5leHBvcnQgZnVuY3Rpb24gbG9nQWRtaW5BY3Rpb24oZW50cnk6IEF1ZGl0RW50cnkpOiB2b2lkIHtcbiAgbG9nZ2VyLmluZm8oJ0FkbWluIGFjdGlvbicsIGVudHJ5KTtcbiAgYXVkaXRRdWV1ZS5wdXNoKGVudHJ5KTtcbiAgaWYgKCFmbHVzaFRpbWVyKSB7XG4gICAgZmx1c2hUaW1lciA9IHNldFRpbWVvdXQoZmx1c2hBdWRpdFF1ZXVlLCA1MDAwKTtcbiAgfVxufVxuXG5hc3luYyBmdW5jdGlvbiBmbHVzaEF1ZGl0UXVldWUoKTogUHJvbWlzZTx2b2lkPiB7XG4gIGZsdXNoVGltZXIgPSBudWxsO1xuICBjb25zdCBiYXRjaCA9IGF1ZGl0UXVldWUuc3BsaWNlKDAsIGF1ZGl0UXVldWUubGVuZ3RoKTtcbiAgaWYgKGJhdGNoLmxlbmd0aCA9PT0gMCkgcmV0dXJuO1xuICAvLyBCYXRjaCB3aWxsIGJlIHBlcnNpc3RlZCB3aGVuIGEgREIgd3JpdGVyIGlzIHJlZ2lzdGVyZWRcbiAgZm9yIChjb25zdCBlbnRyeSBvZiBiYXRjaCkge1xuICAgIGxvZ2dlci5kZWJ1ZygnQXVkaXQgZW50cnkgcXVldWVkJywgeyBhY3Rpb246IGVudHJ5LmFjdGlvbiwgdGFyZ2V0OiBlbnRyeS50YXJnZXRUeXBlIH0pO1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRBdWRpdFF1ZXVlKCk6IEF1ZGl0RW50cnlbXSB7XG4gIHJldHVybiBbLi4uYXVkaXRRdWV1ZV07XG59XG4iXX0=
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Minimal Redis-like client interface (subset of ioredis).
|
|
3
|
+
* Services pass their own Redis client instance.
|
|
4
|
+
*/
|
|
5
|
+
export interface RedisCacheClient {
|
|
6
|
+
get(key: string): Promise<string | null>;
|
|
7
|
+
set(key: string, value: string, ...args: unknown[]): Promise<unknown>;
|
|
8
|
+
del(...keys: string[]): Promise<number>;
|
|
9
|
+
keys(pattern: string): Promise<string[]>;
|
|
10
|
+
/** SCAN-based iteration (preferred over KEYS for production). */
|
|
11
|
+
scanStream?(options: {
|
|
12
|
+
match: string;
|
|
13
|
+
count?: number;
|
|
14
|
+
}): NodeJS.ReadableStream;
|
|
15
|
+
}
|
|
16
|
+
export interface CacheConfig {
|
|
17
|
+
/** Key prefix for namespace isolation (e.g., 'plugin:', 'compliance:') */
|
|
18
|
+
prefix: string;
|
|
19
|
+
/** Default TTL in seconds */
|
|
20
|
+
defaultTtlSeconds: number;
|
|
21
|
+
/** Max entries for in-memory cache (default 1000) */
|
|
22
|
+
maxEntries?: number;
|
|
23
|
+
/** Optional Redis client — uses in-memory cache if not provided */
|
|
24
|
+
redis?: RedisCacheClient;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Cache service with get/set/del operations and automatic TTL expiry.
|
|
28
|
+
*
|
|
29
|
+
* @example
|
|
30
|
+
* ```typescript
|
|
31
|
+
* const cache = new CacheService({ prefix: 'plugin:', defaultTtlSeconds: 300 });
|
|
32
|
+
*
|
|
33
|
+
* // Set with default TTL
|
|
34
|
+
* await cache.set('org123:list', plugins);
|
|
35
|
+
*
|
|
36
|
+
* // Get (returns null on miss)
|
|
37
|
+
* const cached = await cache.get<Plugin[]>('org123:list');
|
|
38
|
+
*
|
|
39
|
+
* // Invalidate
|
|
40
|
+
* await cache.del('org123:list');
|
|
41
|
+
*
|
|
42
|
+
* // Invalidate by pattern
|
|
43
|
+
* await cache.invalidatePattern('org123:*');
|
|
44
|
+
* ```
|
|
45
|
+
*/
|
|
46
|
+
export declare class CacheService {
|
|
47
|
+
private memory;
|
|
48
|
+
private readonly prefix;
|
|
49
|
+
private readonly defaultTtlMs;
|
|
50
|
+
private readonly maxEntries;
|
|
51
|
+
private readonly redis?;
|
|
52
|
+
/** Cache metrics — tracks hits, misses, and invalidations. */
|
|
53
|
+
readonly metrics: {
|
|
54
|
+
hits: number;
|
|
55
|
+
misses: number;
|
|
56
|
+
sets: number;
|
|
57
|
+
invalidations: number;
|
|
58
|
+
};
|
|
59
|
+
constructor(config: CacheConfig);
|
|
60
|
+
private fullKey;
|
|
61
|
+
/**
|
|
62
|
+
* Get a cached value. Returns null on miss or error.
|
|
63
|
+
*/
|
|
64
|
+
get<T>(key: string): Promise<T | null>;
|
|
65
|
+
/**
|
|
66
|
+
* Set a cached value with optional TTL override.
|
|
67
|
+
*
|
|
68
|
+
* @param key - Cache key (prefix is added automatically)
|
|
69
|
+
* @param value - Value to cache (must be JSON-serializable for Redis)
|
|
70
|
+
* @param ttlSeconds - TTL override (uses default if not provided)
|
|
71
|
+
*/
|
|
72
|
+
set<T>(key: string, value: T, ttlSeconds?: number): Promise<void>;
|
|
73
|
+
/**
|
|
74
|
+
* Delete a cached value.
|
|
75
|
+
*/
|
|
76
|
+
del(key: string): Promise<void>;
|
|
77
|
+
/**
|
|
78
|
+
* Invalidate all keys matching a pattern (e.g., 'org123:*').
|
|
79
|
+
* Uses SCAN for Redis (non-blocking) with KEYS fallback, or regex for in-memory.
|
|
80
|
+
*/
|
|
81
|
+
invalidatePattern(pattern: string): Promise<number>;
|
|
82
|
+
/** Collect keys via Redis SCAN (non-blocking alternative to KEYS). */
|
|
83
|
+
private scanKeys;
|
|
84
|
+
/**
|
|
85
|
+
* Get or compute: returns cached value if available, otherwise calls the
|
|
86
|
+
* factory function, caches the result, and returns it.
|
|
87
|
+
*
|
|
88
|
+
* @param key - Cache key
|
|
89
|
+
* @param factory - Async function to compute the value on cache miss
|
|
90
|
+
* @param ttlSeconds - Optional TTL override
|
|
91
|
+
* @returns The cached or computed value
|
|
92
|
+
*/
|
|
93
|
+
getOrSet<T>(key: string, factory: () => Promise<T>, ttlSeconds?: number): Promise<T>;
|
|
94
|
+
/**
|
|
95
|
+
* Clear all entries (useful for testing).
|
|
96
|
+
*/
|
|
97
|
+
clear(): Promise<void>;
|
|
98
|
+
/** Current in-memory cache size (for diagnostics). */
|
|
99
|
+
get size(): number;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Create a cache service instance.
|
|
103
|
+
*
|
|
104
|
+
* @param prefix - Namespace prefix (e.g., 'compliance:', 'plugin:')
|
|
105
|
+
* @param defaultTtlSeconds - Default TTL in seconds (default 300 = 5 min)
|
|
106
|
+
* @param redis - Optional Redis client for cross-process caching
|
|
107
|
+
*/
|
|
108
|
+
export declare function createCacheService(prefix: string, defaultTtlSeconds?: number, redis?: RedisCacheClient): CacheService;
|
|
@@ -0,0 +1,212 @@
|
|
|
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.CacheService = void 0;
|
|
6
|
+
exports.createCacheService = createCacheService;
|
|
7
|
+
/**
|
|
8
|
+
* Cache service with get/set/del operations and automatic TTL expiry.
|
|
9
|
+
*
|
|
10
|
+
* @example
|
|
11
|
+
* ```typescript
|
|
12
|
+
* const cache = new CacheService({ prefix: 'plugin:', defaultTtlSeconds: 300 });
|
|
13
|
+
*
|
|
14
|
+
* // Set with default TTL
|
|
15
|
+
* await cache.set('org123:list', plugins);
|
|
16
|
+
*
|
|
17
|
+
* // Get (returns null on miss)
|
|
18
|
+
* const cached = await cache.get<Plugin[]>('org123:list');
|
|
19
|
+
*
|
|
20
|
+
* // Invalidate
|
|
21
|
+
* await cache.del('org123:list');
|
|
22
|
+
*
|
|
23
|
+
* // Invalidate by pattern
|
|
24
|
+
* await cache.invalidatePattern('org123:*');
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
class CacheService {
|
|
28
|
+
memory = new Map();
|
|
29
|
+
prefix;
|
|
30
|
+
defaultTtlMs;
|
|
31
|
+
maxEntries;
|
|
32
|
+
redis;
|
|
33
|
+
/** Cache metrics — tracks hits, misses, and invalidations. */
|
|
34
|
+
metrics = { hits: 0, misses: 0, sets: 0, invalidations: 0 };
|
|
35
|
+
constructor(config) {
|
|
36
|
+
this.prefix = config.prefix;
|
|
37
|
+
this.defaultTtlMs = config.defaultTtlSeconds * 1000;
|
|
38
|
+
this.maxEntries = config.maxEntries ?? 1000;
|
|
39
|
+
this.redis = config.redis;
|
|
40
|
+
}
|
|
41
|
+
fullKey(key) {
|
|
42
|
+
return `${this.prefix}${key}`;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Get a cached value. Returns null on miss or error.
|
|
46
|
+
*/
|
|
47
|
+
async get(key) {
|
|
48
|
+
const fk = this.fullKey(key);
|
|
49
|
+
try {
|
|
50
|
+
if (this.redis) {
|
|
51
|
+
const raw = await this.redis.get(fk);
|
|
52
|
+
if (!raw) {
|
|
53
|
+
this.metrics.misses++;
|
|
54
|
+
return null;
|
|
55
|
+
}
|
|
56
|
+
this.metrics.hits++;
|
|
57
|
+
return JSON.parse(raw);
|
|
58
|
+
}
|
|
59
|
+
// In-memory
|
|
60
|
+
const entry = this.memory.get(fk);
|
|
61
|
+
if (!entry) {
|
|
62
|
+
this.metrics.misses++;
|
|
63
|
+
return null;
|
|
64
|
+
}
|
|
65
|
+
if (Date.now() > entry.expiresAt) {
|
|
66
|
+
this.memory.delete(fk);
|
|
67
|
+
this.metrics.misses++;
|
|
68
|
+
return null;
|
|
69
|
+
}
|
|
70
|
+
this.metrics.hits++;
|
|
71
|
+
return entry.value;
|
|
72
|
+
}
|
|
73
|
+
catch {
|
|
74
|
+
this.metrics.misses++;
|
|
75
|
+
return null;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Set a cached value with optional TTL override.
|
|
80
|
+
*
|
|
81
|
+
* @param key - Cache key (prefix is added automatically)
|
|
82
|
+
* @param value - Value to cache (must be JSON-serializable for Redis)
|
|
83
|
+
* @param ttlSeconds - TTL override (uses default if not provided)
|
|
84
|
+
*/
|
|
85
|
+
async set(key, value, ttlSeconds) {
|
|
86
|
+
const fk = this.fullKey(key);
|
|
87
|
+
const ttl = ttlSeconds ?? this.defaultTtlMs / 1000;
|
|
88
|
+
try {
|
|
89
|
+
this.metrics.sets++;
|
|
90
|
+
if (this.redis) {
|
|
91
|
+
await this.redis.set(fk, JSON.stringify(value), 'EX', ttl);
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
// In-memory — evict oldest if at capacity
|
|
95
|
+
if (this.memory.size >= this.maxEntries) {
|
|
96
|
+
const firstKey = this.memory.keys().next().value;
|
|
97
|
+
if (firstKey)
|
|
98
|
+
this.memory.delete(firstKey);
|
|
99
|
+
}
|
|
100
|
+
this.memory.set(fk, {
|
|
101
|
+
value,
|
|
102
|
+
expiresAt: Date.now() + ttl * 1000,
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
catch {
|
|
106
|
+
// Cache set failure is non-fatal
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Delete a cached value.
|
|
111
|
+
*/
|
|
112
|
+
async del(key) {
|
|
113
|
+
const fk = this.fullKey(key);
|
|
114
|
+
try {
|
|
115
|
+
if (this.redis) {
|
|
116
|
+
await this.redis.del(fk);
|
|
117
|
+
return;
|
|
118
|
+
}
|
|
119
|
+
this.memory.delete(fk);
|
|
120
|
+
}
|
|
121
|
+
catch {
|
|
122
|
+
// Cache delete failure is non-fatal
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Invalidate all keys matching a pattern (e.g., 'org123:*').
|
|
127
|
+
* Uses SCAN for Redis (non-blocking) with KEYS fallback, or regex for in-memory.
|
|
128
|
+
*/
|
|
129
|
+
async invalidatePattern(pattern) {
|
|
130
|
+
const fp = this.fullKey(pattern);
|
|
131
|
+
try {
|
|
132
|
+
if (this.redis) {
|
|
133
|
+
const keys = this.redis.scanStream
|
|
134
|
+
? await this.scanKeys(fp)
|
|
135
|
+
: await this.redis.keys(fp);
|
|
136
|
+
if (keys.length > 0) {
|
|
137
|
+
await this.redis.del(...keys);
|
|
138
|
+
this.metrics.invalidations += keys.length;
|
|
139
|
+
}
|
|
140
|
+
return keys.length;
|
|
141
|
+
}
|
|
142
|
+
// In-memory — match glob-style pattern
|
|
143
|
+
const regex = new RegExp('^' + fp.replace(/\*/g, '.*') + '$');
|
|
144
|
+
let deleted = 0;
|
|
145
|
+
for (const key of this.memory.keys()) {
|
|
146
|
+
if (regex.test(key)) {
|
|
147
|
+
this.memory.delete(key);
|
|
148
|
+
deleted++;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
this.metrics.invalidations += deleted;
|
|
152
|
+
return deleted;
|
|
153
|
+
}
|
|
154
|
+
catch {
|
|
155
|
+
return 0;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
/** Collect keys via Redis SCAN (non-blocking alternative to KEYS). */
|
|
159
|
+
scanKeys(pattern) {
|
|
160
|
+
return new Promise((resolve, reject) => {
|
|
161
|
+
const stream = this.redis.scanStream({ match: pattern, count: 100 });
|
|
162
|
+
const keys = [];
|
|
163
|
+
stream.on('data', (batch) => keys.push(...batch));
|
|
164
|
+
stream.once('end', () => resolve(keys));
|
|
165
|
+
stream.once('error', reject);
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Get or compute: returns cached value if available, otherwise calls the
|
|
170
|
+
* factory function, caches the result, and returns it.
|
|
171
|
+
*
|
|
172
|
+
* @param key - Cache key
|
|
173
|
+
* @param factory - Async function to compute the value on cache miss
|
|
174
|
+
* @param ttlSeconds - Optional TTL override
|
|
175
|
+
* @returns The cached or computed value
|
|
176
|
+
*/
|
|
177
|
+
async getOrSet(key, factory, ttlSeconds) {
|
|
178
|
+
const cached = await this.get(key);
|
|
179
|
+
if (cached !== null)
|
|
180
|
+
return cached;
|
|
181
|
+
const value = await factory();
|
|
182
|
+
await this.set(key, value, ttlSeconds);
|
|
183
|
+
return value;
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Clear all entries (useful for testing).
|
|
187
|
+
*/
|
|
188
|
+
async clear() {
|
|
189
|
+
if (this.redis) {
|
|
190
|
+
await this.invalidatePattern('*');
|
|
191
|
+
}
|
|
192
|
+
else {
|
|
193
|
+
this.memory.clear();
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
/** Current in-memory cache size (for diagnostics). */
|
|
197
|
+
get size() {
|
|
198
|
+
return this.memory.size;
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
exports.CacheService = CacheService;
|
|
202
|
+
/**
|
|
203
|
+
* Create a cache service instance.
|
|
204
|
+
*
|
|
205
|
+
* @param prefix - Namespace prefix (e.g., 'compliance:', 'plugin:')
|
|
206
|
+
* @param defaultTtlSeconds - Default TTL in seconds (default 300 = 5 min)
|
|
207
|
+
* @param redis - Optional Redis client for cross-process caching
|
|
208
|
+
*/
|
|
209
|
+
function createCacheService(prefix, defaultTtlSeconds = 300, redis) {
|
|
210
|
+
return new CacheService({ prefix, defaultTtlSeconds, redis });
|
|
211
|
+
}
|
|
212
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cache-service.js","sourceRoot":"","sources":["../../src/services/cache-service.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC;;;AAiQtC,gDAMC;AAvND;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAa,YAAY;IACf,MAAM,GAAG,IAAI,GAAG,EAA+B,CAAC;IACvC,MAAM,CAAS;IACf,YAAY,CAAS;IACrB,UAAU,CAAS;IACnB,KAAK,CAAoB;IAE1C,8DAA8D;IACrD,OAAO,GAAG,EAAE,IAAI,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,aAAa,EAAE,CAAC,EAAE,CAAC;IAErE,YAAY,MAAmB;QAC7B,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC;QAC5B,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,iBAAiB,GAAG,IAAI,CAAC;QACpD,IAAI,CAAC,UAAU,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,CAAC;QAC5C,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC;IAC5B,CAAC;IAEO,OAAO,CAAC,GAAW;QACzB,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;IAChC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,GAAG,CAAI,GAAW;QACtB,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAE7B,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBACrC,IAAI,CAAC,GAAG,EAAE,CAAC;oBAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;oBAAC,OAAO,IAAI,CAAC;gBAAC,CAAC;gBACjD,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;gBACpB,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAM,CAAC;YAC9B,CAAC;YAED,YAAY;YACZ,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAClC,IAAI,CAAC,KAAK,EAAE,CAAC;gBAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;gBAAC,OAAO,IAAI,CAAC;YAAC,CAAC;YACnD,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC,SAAS,EAAE,CAAC;gBACjC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;gBACvB,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;gBACtB,OAAO,IAAI,CAAC;YACd,CAAC;YACD,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YACpB,OAAO,KAAK,CAAC,KAAU,CAAC;QAC1B,CAAC;QAAC,MAAM,CAAC;YACP,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,GAAG,CAAI,GAAW,EAAE,KAAQ,EAAE,UAAmB;QACrD,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAC7B,MAAM,GAAG,GAAG,UAAU,IAAI,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;QAEnD,IAAI,CAAC;YACH,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YACpB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,IAAI,EAAE,GAAG,CAAC,CAAC;gBAC3D,OAAO;YACT,CAAC;YAED,0CAA0C;YAC1C,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACxC,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC;gBACjD,IAAI,QAAQ;oBAAE,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;YAC7C,CAAC;YAED,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,EAAE;gBAClB,KAAK;gBACL,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,IAAI;aACnC,CAAC,CAAC;QACL,CAAC;QAAC,MAAM,CAAC;YACP,iCAAiC;QACnC,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,GAAG,CAAC,GAAW;QACnB,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAE7B,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBACzB,OAAO;YACT,CAAC;YACD,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACzB,CAAC;QAAC,MAAM,CAAC;YACP,oCAAoC;QACtC,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,iBAAiB,CAAC,OAAe;QACrC,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAEjC,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU;oBAChC,CAAC,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;oBACzB,CAAC,CAAC,MAAM,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBAC9B,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACpB,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,CAAC;oBAC9B,IAAI,CAAC,OAAO,CAAC,aAAa,IAAI,IAAI,CAAC,MAAM,CAAC;gBAC5C,CAAC;gBACD,OAAO,IAAI,CAAC,MAAM,CAAC;YACrB,CAAC;YAED,uCAAuC;YACvC,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC;YAC9D,IAAI,OAAO,GAAG,CAAC,CAAC;YAChB,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;gBACrC,IAAI,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;oBACpB,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;oBACxB,OAAO,EAAE,CAAC;gBACZ,CAAC;YACH,CAAC;YACD,IAAI,CAAC,OAAO,CAAC,aAAa,IAAI,OAAO,CAAC;YACtC,OAAO,OAAO,CAAC;QACjB,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,CAAC;QACX,CAAC;IACH,CAAC;IAED,sEAAsE;IAC9D,QAAQ,CAAC,OAAe;QAC9B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,MAAM,GAAG,IAAI,CAAC,KAAM,CAAC,UAAW,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;YACvE,MAAM,IAAI,GAAa,EAAE,CAAC;YAC1B,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAe,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC;YAC5D,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;YACxC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,QAAQ,CAAI,GAAW,EAAE,OAAyB,EAAE,UAAmB;QAC3E,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,GAAG,CAAI,GAAG,CAAC,CAAC;QACtC,IAAI,MAAM,KAAK,IAAI;YAAE,OAAO,MAAM,CAAC;QAEnC,MAAM,KAAK,GAAG,MAAM,OAAO,EAAE,CAAC;QAC9B,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,UAAU,CAAC,CAAC;QACvC,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAK;QACT,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,MAAM,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC;QACpC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACtB,CAAC;IACH,CAAC;IAED,sDAAsD;IACtD,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;IAC1B,CAAC;CACF;AApLD,oCAoLC;AAED;;;;;;GAMG;AACH,SAAgB,kBAAkB,CAChC,MAAc,EACd,iBAAiB,GAAG,GAAG,EACvB,KAAwB;IAExB,OAAO,IAAI,YAAY,CAAC,EAAE,MAAM,EAAE,iBAAiB,EAAE,KAAK,EAAE,CAAC,CAAC;AAChE,CAAC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\n/**\n * Lightweight caching service with TTL support.\n *\n * Two implementations:\n * - In-memory (default): LRU-style Map cache, no external dependencies\n * - Redis: When a Redis client is provided, uses Redis for cross-process caching\n *\n * Design:\n * - All operations are fail-safe: cache misses/errors return null, never throw\n * - JSON serialization for Redis; direct reference for in-memory\n * - Key namespace prefixing to avoid collisions between services\n */\n\n\n/**\n * Cache entry with value and expiration time.\n */\ninterface CacheEntry<T> {\n  value: T;\n  expiresAt: number; // Unix timestamp in ms\n}\n\n/**\n * Minimal Redis-like client interface (subset of ioredis).\n * Services pass their own Redis client instance.\n */\nexport interface RedisCacheClient {\n  get(key: string): Promise<string | null>;\n  set(key: string, value: string, ...args: unknown[]): Promise<unknown>;\n  del(...keys: string[]): Promise<number>;\n  keys(pattern: string): Promise<string[]>;\n  /** SCAN-based iteration (preferred over KEYS for production). */\n  scanStream?(options: { match: string; count?: number }): NodeJS.ReadableStream;\n}\n\nexport interface CacheConfig {\n  /** Key prefix for namespace isolation (e.g., 'plugin:', 'compliance:') */\n  prefix: string;\n  /** Default TTL in seconds */\n  defaultTtlSeconds: number;\n  /** Max entries for in-memory cache (default 1000) */\n  maxEntries?: number;\n  /** Optional Redis client — uses in-memory cache if not provided */\n  redis?: RedisCacheClient;\n}\n\n/**\n * Cache service with get/set/del operations and automatic TTL expiry.\n *\n * @example\n * ```typescript\n * const cache = new CacheService({ prefix: 'plugin:', defaultTtlSeconds: 300 });\n *\n * // Set with default TTL\n * await cache.set('org123:list', plugins);\n *\n * // Get (returns null on miss)\n * const cached = await cache.get<Plugin[]>('org123:list');\n *\n * // Invalidate\n * await cache.del('org123:list');\n *\n * // Invalidate by pattern\n * await cache.invalidatePattern('org123:*');\n * ```\n */\nexport class CacheService {\n  private memory = new Map<string, CacheEntry<unknown>>();\n  private readonly prefix: string;\n  private readonly defaultTtlMs: number;\n  private readonly maxEntries: number;\n  private readonly redis?: RedisCacheClient;\n\n  /** Cache metrics — tracks hits, misses, and invalidations. */\n  readonly metrics = { hits: 0, misses: 0, sets: 0, invalidations: 0 };\n\n  constructor(config: CacheConfig) {\n    this.prefix = config.prefix;\n    this.defaultTtlMs = config.defaultTtlSeconds * 1000;\n    this.maxEntries = config.maxEntries ?? 1000;\n    this.redis = config.redis;\n  }\n\n  private fullKey(key: string): string {\n    return `${this.prefix}${key}`;\n  }\n\n  /**\n   * Get a cached value. Returns null on miss or error.\n   */\n  async get<T>(key: string): Promise<T | null> {\n    const fk = this.fullKey(key);\n\n    try {\n      if (this.redis) {\n        const raw = await this.redis.get(fk);\n        if (!raw) { this.metrics.misses++; return null; }\n        this.metrics.hits++;\n        return JSON.parse(raw) as T;\n      }\n\n      // In-memory\n      const entry = this.memory.get(fk);\n      if (!entry) { this.metrics.misses++; return null; }\n      if (Date.now() > entry.expiresAt) {\n        this.memory.delete(fk);\n        this.metrics.misses++;\n        return null;\n      }\n      this.metrics.hits++;\n      return entry.value as T;\n    } catch {\n      this.metrics.misses++;\n      return null;\n    }\n  }\n\n  /**\n   * Set a cached value with optional TTL override.\n   *\n   * @param key - Cache key (prefix is added automatically)\n   * @param value - Value to cache (must be JSON-serializable for Redis)\n   * @param ttlSeconds - TTL override (uses default if not provided)\n   */\n  async set<T>(key: string, value: T, ttlSeconds?: number): Promise<void> {\n    const fk = this.fullKey(key);\n    const ttl = ttlSeconds ?? this.defaultTtlMs / 1000;\n\n    try {\n      this.metrics.sets++;\n      if (this.redis) {\n        await this.redis.set(fk, JSON.stringify(value), 'EX', ttl);\n        return;\n      }\n\n      // In-memory — evict oldest if at capacity\n      if (this.memory.size >= this.maxEntries) {\n        const firstKey = this.memory.keys().next().value;\n        if (firstKey) this.memory.delete(firstKey);\n      }\n\n      this.memory.set(fk, {\n        value,\n        expiresAt: Date.now() + ttl * 1000,\n      });\n    } catch {\n      // Cache set failure is non-fatal\n    }\n  }\n\n  /**\n   * Delete a cached value.\n   */\n  async del(key: string): Promise<void> {\n    const fk = this.fullKey(key);\n\n    try {\n      if (this.redis) {\n        await this.redis.del(fk);\n        return;\n      }\n      this.memory.delete(fk);\n    } catch {\n      // Cache delete failure is non-fatal\n    }\n  }\n\n  /**\n   * Invalidate all keys matching a pattern (e.g., 'org123:*').\n   * Uses SCAN for Redis (non-blocking) with KEYS fallback, or regex for in-memory.\n   */\n  async invalidatePattern(pattern: string): Promise<number> {\n    const fp = this.fullKey(pattern);\n\n    try {\n      if (this.redis) {\n        const keys = this.redis.scanStream\n          ? await this.scanKeys(fp)\n          : await this.redis.keys(fp);\n        if (keys.length > 0) {\n          await this.redis.del(...keys);\n          this.metrics.invalidations += keys.length;\n        }\n        return keys.length;\n      }\n\n      // In-memory — match glob-style pattern\n      const regex = new RegExp('^' + fp.replace(/\\*/g, '.*') + '$');\n      let deleted = 0;\n      for (const key of this.memory.keys()) {\n        if (regex.test(key)) {\n          this.memory.delete(key);\n          deleted++;\n        }\n      }\n      this.metrics.invalidations += deleted;\n      return deleted;\n    } catch {\n      return 0;\n    }\n  }\n\n  /** Collect keys via Redis SCAN (non-blocking alternative to KEYS). */\n  private scanKeys(pattern: string): Promise<string[]> {\n    return new Promise((resolve, reject) => {\n      const stream = this.redis!.scanStream!({ match: pattern, count: 100 });\n      const keys: string[] = [];\n      stream.on('data', (batch: string[]) => keys.push(...batch));\n      stream.once('end', () => resolve(keys));\n      stream.once('error', reject);\n    });\n  }\n\n  /**\n   * Get or compute: returns cached value if available, otherwise calls the\n   * factory function, caches the result, and returns it.\n   *\n   * @param key - Cache key\n   * @param factory - Async function to compute the value on cache miss\n   * @param ttlSeconds - Optional TTL override\n   * @returns The cached or computed value\n   */\n  async getOrSet<T>(key: string, factory: () => Promise<T>, ttlSeconds?: number): Promise<T> {\n    const cached = await this.get<T>(key);\n    if (cached !== null) return cached;\n\n    const value = await factory();\n    await this.set(key, value, ttlSeconds);\n    return value;\n  }\n\n  /**\n   * Clear all entries (useful for testing).\n   */\n  async clear(): Promise<void> {\n    if (this.redis) {\n      await this.invalidatePattern('*');\n    } else {\n      this.memory.clear();\n    }\n  }\n\n  /** Current in-memory cache size (for diagnostics). */\n  get size(): number {\n    return this.memory.size;\n  }\n}\n\n/**\n * Create a cache service instance.\n *\n * @param prefix - Namespace prefix (e.g., 'compliance:', 'plugin:')\n * @param defaultTtlSeconds - Default TTL in seconds (default 300 = 5 min)\n * @param redis - Optional Redis client for cross-process caching\n */\nexport function createCacheService(\n  prefix: string,\n  defaultTtlSeconds = 300,\n  redis?: RedisCacheClient,\n): CacheService {\n  return new CacheService({ prefix, defaultTtlSeconds, redis });\n}\n"]}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { ServiceConfig } from '../types/common';
|
|
2
|
+
/**
|
|
3
|
+
* Result of a compliance validation check.
|
|
4
|
+
*/
|
|
5
|
+
export interface ComplianceCheckResult {
|
|
6
|
+
passed: boolean;
|
|
7
|
+
violations: ComplianceViolation[];
|
|
8
|
+
warnings: ComplianceViolation[];
|
|
9
|
+
blocked: boolean;
|
|
10
|
+
rulesEvaluated: number;
|
|
11
|
+
rulesSkipped: number;
|
|
12
|
+
exemptionsApplied: string[];
|
|
13
|
+
}
|
|
14
|
+
export interface ComplianceViolation {
|
|
15
|
+
ruleId: string;
|
|
16
|
+
ruleName: string;
|
|
17
|
+
policyId?: string | null;
|
|
18
|
+
field: string;
|
|
19
|
+
operator: string;
|
|
20
|
+
expectedValue: unknown;
|
|
21
|
+
actualValue: unknown;
|
|
22
|
+
severity: string;
|
|
23
|
+
message: string;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Compliance service client interface.
|
|
27
|
+
* Uses fail-closed design: errors propagate to block the operation.
|
|
28
|
+
*/
|
|
29
|
+
export interface ComplianceClient {
|
|
30
|
+
/** Validate plugin attributes against org rules. Throws on service error (fail-closed). */
|
|
31
|
+
validatePlugin(orgId: string, attributes: Record<string, unknown>, authHeader: string, entityId?: string, entityName?: string, action?: string): Promise<ComplianceCheckResult>;
|
|
32
|
+
/** Validate pipeline attributes against org rules. Throws on service error (fail-closed). */
|
|
33
|
+
validatePipeline(orgId: string, attributes: Record<string, unknown>, authHeader: string, entityId?: string, entityName?: string, action?: string): Promise<ComplianceCheckResult>;
|
|
34
|
+
/** Pre-flight check for plugin (no audit, no notification). */
|
|
35
|
+
dryRunPlugin(orgId: string, attributes: Record<string, unknown>, authHeader: string): Promise<ComplianceCheckResult>;
|
|
36
|
+
/** Pre-flight check for pipeline (no audit, no notification). */
|
|
37
|
+
dryRunPipeline(orgId: string, attributes: Record<string, unknown>, authHeader: string): Promise<ComplianceCheckResult>;
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Create a compliance client.
|
|
41
|
+
*
|
|
42
|
+
* IMPORTANT: This client is fail-closed — if the compliance service is
|
|
43
|
+
* unreachable or returns an error, the error propagates and the calling
|
|
44
|
+
* operation (plugin upload, pipeline create) is rejected.
|
|
45
|
+
*/
|
|
46
|
+
export declare function createComplianceClient(config?: Partial<ServiceConfig>): ComplianceClient;
|
|
@@ -0,0 +1,102 @@
|
|
|
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.createComplianceClient = createComplianceClient;
|
|
6
|
+
const http_client_1 = require("./http-client");
|
|
7
|
+
const logger_1 = require("../utils/logger");
|
|
8
|
+
const logger = (0, logger_1.createLogger)('compliance-client');
|
|
9
|
+
/**
|
|
10
|
+
* When true, compliance checks are bypassed if the service is unavailable
|
|
11
|
+
* (fail-open). Default is fail-closed (errors propagate and block the operation).
|
|
12
|
+
*/
|
|
13
|
+
const COMPLIANCE_BYPASS = process.env.COMPLIANCE_BYPASS === 'true';
|
|
14
|
+
/**
|
|
15
|
+
* Return a pass-through result when COMPLIANCE_BYPASS is enabled and the
|
|
16
|
+
* compliance service is unreachable.
|
|
17
|
+
*/
|
|
18
|
+
function bypassResult(context) {
|
|
19
|
+
logger.warn('COMPLIANCE_BYPASS: Service unavailable, allowing request', context);
|
|
20
|
+
return {
|
|
21
|
+
passed: true,
|
|
22
|
+
blocked: false,
|
|
23
|
+
violations: [],
|
|
24
|
+
warnings: [{ ruleId: '', ruleName: '', field: '', operator: '', expectedValue: null, actualValue: null, severity: 'warning', message: 'Compliance check skipped (service unavailable)' }],
|
|
25
|
+
rulesEvaluated: 0,
|
|
26
|
+
rulesSkipped: 0,
|
|
27
|
+
exemptionsApplied: [],
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Build common request headers for compliance calls.
|
|
32
|
+
*/
|
|
33
|
+
function buildHeaders(orgId, authHeader) {
|
|
34
|
+
const headers = {
|
|
35
|
+
'x-org-id': orgId,
|
|
36
|
+
'x-internal-service': 'true',
|
|
37
|
+
};
|
|
38
|
+
if (authHeader)
|
|
39
|
+
headers.Authorization = authHeader;
|
|
40
|
+
return headers;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Create a compliance client.
|
|
44
|
+
*
|
|
45
|
+
* IMPORTANT: This client is fail-closed — if the compliance service is
|
|
46
|
+
* unreachable or returns an error, the error propagates and the calling
|
|
47
|
+
* operation (plugin upload, pipeline create) is rejected.
|
|
48
|
+
*/
|
|
49
|
+
function createComplianceClient(config) {
|
|
50
|
+
const serviceConfig = {
|
|
51
|
+
host: config?.host ?? process.env.COMPLIANCE_SERVICE_HOST ?? 'compliance',
|
|
52
|
+
port: config?.port ?? parseInt(process.env.COMPLIANCE_SERVICE_PORT ?? '3000', 10),
|
|
53
|
+
};
|
|
54
|
+
const client = new http_client_1.InternalHttpClient(serviceConfig);
|
|
55
|
+
return {
|
|
56
|
+
async validatePlugin(orgId, attributes, authHeader, entityId, entityName, action) {
|
|
57
|
+
try {
|
|
58
|
+
const response = await client.post('/compliance/validate/plugin', { attributes, entityId, entityName, action: action ?? 'upload' }, { headers: buildHeaders(orgId, authHeader) });
|
|
59
|
+
return response.body.data;
|
|
60
|
+
}
|
|
61
|
+
catch (error) {
|
|
62
|
+
if (COMPLIANCE_BYPASS)
|
|
63
|
+
return bypassResult({ orgId, entityType: 'plugin', entityId, entityName });
|
|
64
|
+
throw error;
|
|
65
|
+
}
|
|
66
|
+
},
|
|
67
|
+
async validatePipeline(orgId, attributes, authHeader, entityId, entityName, action) {
|
|
68
|
+
try {
|
|
69
|
+
const response = await client.post('/compliance/validate/pipeline', { attributes, entityId, entityName, action: action ?? 'create' }, { headers: buildHeaders(orgId, authHeader) });
|
|
70
|
+
return response.body.data;
|
|
71
|
+
}
|
|
72
|
+
catch (error) {
|
|
73
|
+
if (COMPLIANCE_BYPASS)
|
|
74
|
+
return bypassResult({ orgId, entityType: 'pipeline', entityId, entityName });
|
|
75
|
+
throw error;
|
|
76
|
+
}
|
|
77
|
+
},
|
|
78
|
+
async dryRunPlugin(orgId, attributes, authHeader) {
|
|
79
|
+
try {
|
|
80
|
+
const response = await client.post('/compliance/validate/plugin/dry-run', { attributes }, { headers: buildHeaders(orgId, authHeader) });
|
|
81
|
+
return response.body.data;
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
if (COMPLIANCE_BYPASS)
|
|
85
|
+
return bypassResult({ orgId, entityType: 'plugin', dryRun: true });
|
|
86
|
+
throw error;
|
|
87
|
+
}
|
|
88
|
+
},
|
|
89
|
+
async dryRunPipeline(orgId, attributes, authHeader) {
|
|
90
|
+
try {
|
|
91
|
+
const response = await client.post('/compliance/validate/pipeline/dry-run', { attributes }, { headers: buildHeaders(orgId, authHeader) });
|
|
92
|
+
return response.body.data;
|
|
93
|
+
}
|
|
94
|
+
catch (error) {
|
|
95
|
+
if (COMPLIANCE_BYPASS)
|
|
96
|
+
return bypassResult({ orgId, entityType: 'pipeline', dryRun: true });
|
|
97
|
+
throw error;
|
|
98
|
+
}
|
|
99
|
+
},
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"compliance-client.js","sourceRoot":"","sources":["../../src/services/compliance-client.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { InternalHttpClient } from './http-client';\nimport { ServiceConfig } from '../types/common';\nimport { createLogger } from '../utils/logger';\n\nconst logger = createLogger('compliance-client');\n\n/**\n * When true, compliance checks are bypassed if the service is unavailable\n * (fail-open). Default is fail-closed (errors propagate and block the operation).\n */\nconst COMPLIANCE_BYPASS = process.env.COMPLIANCE_BYPASS === 'true';\n\n/**\n * Result of a compliance validation check.\n */\nexport interface ComplianceCheckResult {\n  passed: boolean;\n  violations: ComplianceViolation[];\n  warnings: ComplianceViolation[];\n  blocked: boolean;\n  rulesEvaluated: number;\n  rulesSkipped: number;\n  exemptionsApplied: string[];\n}\n\nexport interface ComplianceViolation {\n  ruleId: string;\n  ruleName: string;\n  policyId?: string | null;\n  field: string;\n  operator: string;\n  expectedValue: unknown;\n  actualValue: unknown;\n  severity: string;\n  message: string;\n}\n\n/**\n * Compliance service client interface.\n * Uses fail-closed design: errors propagate to block the operation.\n */\nexport interface ComplianceClient {\n  /** Validate plugin attributes against org rules. Throws on service error (fail-closed). */\n  validatePlugin(\n    orgId: string,\n    attributes: Record<string, unknown>,\n    authHeader: string,\n    entityId?: string,\n    entityName?: string,\n    action?: string,\n  ): Promise<ComplianceCheckResult>;\n\n  /** Validate pipeline attributes against org rules. Throws on service error (fail-closed). */\n  validatePipeline(\n    orgId: string,\n    attributes: Record<string, unknown>,\n    authHeader: string,\n    entityId?: string,\n    entityName?: string,\n    action?: string,\n  ): Promise<ComplianceCheckResult>;\n\n  /** Pre-flight check for plugin (no audit, no notification). */\n  dryRunPlugin(\n    orgId: string,\n    attributes: Record<string, unknown>,\n    authHeader: string,\n  ): Promise<ComplianceCheckResult>;\n\n  /** Pre-flight check for pipeline (no audit, no notification). */\n  dryRunPipeline(\n    orgId: string,\n    attributes: Record<string, unknown>,\n    authHeader: string,\n  ): Promise<ComplianceCheckResult>;\n}\n\n/**\n * Return a pass-through result when COMPLIANCE_BYPASS is enabled and the\n * compliance service is unreachable.\n */\nfunction bypassResult(context: Record<string, unknown>): ComplianceCheckResult {\n  logger.warn('COMPLIANCE_BYPASS: Service unavailable, allowing request', context);\n  return {\n    passed: true,\n    blocked: false,\n    violations: [],\n    warnings: [{ ruleId: '', ruleName: '', field: '', operator: '', expectedValue: null, actualValue: null, severity: 'warning', message: 'Compliance check skipped (service unavailable)' }],\n    rulesEvaluated: 0,\n    rulesSkipped: 0,\n    exemptionsApplied: [],\n  };\n}\n\n/**\n * Build common request headers for compliance calls.\n */\nfunction buildHeaders(orgId: string, authHeader: string): Record<string, string> {\n  const headers: Record<string, string> = {\n    'x-org-id': orgId,\n    'x-internal-service': 'true',\n  };\n  if (authHeader) headers.Authorization = authHeader;\n  return headers;\n}\n\n/**\n * Create a compliance client.\n *\n * IMPORTANT: This client is fail-closed — if the compliance service is\n * unreachable or returns an error, the error propagates and the calling\n * operation (plugin upload, pipeline create) is rejected.\n */\nexport function createComplianceClient(config?: Partial<ServiceConfig>): ComplianceClient {\n  const serviceConfig: ServiceConfig = {\n    host: config?.host ?? process.env.COMPLIANCE_SERVICE_HOST ?? 'compliance',\n    port: config?.port ?? parseInt(process.env.COMPLIANCE_SERVICE_PORT ?? '3000', 10),\n  };\n\n  const client = new InternalHttpClient(serviceConfig);\n\n  return {\n    async validatePlugin(orgId, attributes, authHeader, entityId, entityName, action) {\n      try {\n        const response = await client.post<{ success: boolean; data: ComplianceCheckResult }>(\n          '/compliance/validate/plugin',\n          { attributes, entityId, entityName, action: action ?? 'upload' },\n          { headers: buildHeaders(orgId, authHeader) },\n        );\n        return response.body.data;\n      } catch (error) {\n        if (COMPLIANCE_BYPASS) return bypassResult({ orgId, entityType: 'plugin', entityId, entityName });\n        throw error;\n      }\n    },\n\n    async validatePipeline(orgId, attributes, authHeader, entityId, entityName, action) {\n      try {\n        const response = await client.post<{ success: boolean; data: ComplianceCheckResult }>(\n          '/compliance/validate/pipeline',\n          { attributes, entityId, entityName, action: action ?? 'create' },\n          { headers: buildHeaders(orgId, authHeader) },\n        );\n        return response.body.data;\n      } catch (error) {\n        if (COMPLIANCE_BYPASS) return bypassResult({ orgId, entityType: 'pipeline', entityId, entityName });\n        throw error;\n      }\n    },\n\n    async dryRunPlugin(orgId, attributes, authHeader) {\n      try {\n        const response = await client.post<{ success: boolean; data: ComplianceCheckResult }>(\n          '/compliance/validate/plugin/dry-run',\n          { attributes },\n          { headers: buildHeaders(orgId, authHeader) },\n        );\n        return response.body.data;\n      } catch (error) {\n        if (COMPLIANCE_BYPASS) return bypassResult({ orgId, entityType: 'plugin', dryRun: true });\n        throw error;\n      }\n    },\n\n    async dryRunPipeline(orgId, attributes, authHeader) {\n      try {\n        const response = await client.post<{ success: boolean; data: ComplianceCheckResult }>(\n          '/compliance/validate/pipeline/dry-run',\n          { attributes },\n          { headers: buildHeaders(orgId, authHeader) },\n        );\n        return response.body.data;\n      } catch (error) {\n        if (COMPLIANCE_BYPASS) return bypassResult({ orgId, entityType: 'pipeline', dryRun: true });\n        throw error;\n      }\n    },\n  };\n}\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { type ServiceConfig } from '../types/common';
|
|
2
|
+
/**
|
|
3
|
+
* Registers an entity event subscriber that forwards events to the compliance service.
|
|
4
|
+
*
|
|
5
|
+
* Call this at service startup (in index.ts) to enable automatic compliance
|
|
6
|
+
* notification on entity mutations. Events are fire-and-forget — failures
|
|
7
|
+
* are logged but never block the original request.
|
|
8
|
+
*
|
|
9
|
+
* @param config - Optional service config override (defaults to COMPLIANCE_SERVICE_HOST/PORT env vars)
|
|
10
|
+
*/
|
|
11
|
+
export declare function registerComplianceEventSubscriber(config?: Partial<ServiceConfig>): void;
|
|
@@ -0,0 +1,60 @@
|
|
|
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.registerComplianceEventSubscriber = registerComplianceEventSubscriber;
|
|
6
|
+
const compliance_queue_1 = require("./compliance-queue");
|
|
7
|
+
const entity_events_1 = require("./entity-events");
|
|
8
|
+
const http_client_1 = require("./http-client");
|
|
9
|
+
const logger_1 = require("../utils/logger");
|
|
10
|
+
const logger = (0, logger_1.createLogger)('compliance-events');
|
|
11
|
+
/**
|
|
12
|
+
* Registers an entity event subscriber that forwards events to the compliance service.
|
|
13
|
+
*
|
|
14
|
+
* Call this at service startup (in index.ts) to enable automatic compliance
|
|
15
|
+
* notification on entity mutations. Events are fire-and-forget — failures
|
|
16
|
+
* are logged but never block the original request.
|
|
17
|
+
*
|
|
18
|
+
* @param config - Optional service config override (defaults to COMPLIANCE_SERVICE_HOST/PORT env vars)
|
|
19
|
+
*/
|
|
20
|
+
function registerComplianceEventSubscriber(config) {
|
|
21
|
+
const serviceConfig = {
|
|
22
|
+
host: config?.host ?? process.env.COMPLIANCE_SERVICE_HOST ?? 'compliance',
|
|
23
|
+
port: config?.port ?? parseInt(process.env.COMPLIANCE_SERVICE_PORT ?? '3000', 10),
|
|
24
|
+
};
|
|
25
|
+
const client = new http_client_1.InternalHttpClient(serviceConfig);
|
|
26
|
+
const subscriber = {
|
|
27
|
+
async onEntityEvent(event) {
|
|
28
|
+
try {
|
|
29
|
+
await client.post('/compliance/events/entity', event, {
|
|
30
|
+
headers: { 'x-internal-service': 'true' },
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
catch (err) {
|
|
34
|
+
// Fire-and-forget: log and swallow. Compliance notification is non-fatal.
|
|
35
|
+
logger.warn('Failed to notify compliance service of entity event', {
|
|
36
|
+
target: event.target,
|
|
37
|
+
eventType: event.eventType,
|
|
38
|
+
entityId: event.entityId,
|
|
39
|
+
error: err instanceof Error ? err.message : String(err),
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
// Also enqueue for async re-validation (catches rule changes after creation)
|
|
43
|
+
void (0, compliance_queue_1.enqueueComplianceEvent)({
|
|
44
|
+
eventType: 'validate',
|
|
45
|
+
target: event.target,
|
|
46
|
+
entityId: event.entityId,
|
|
47
|
+
orgId: event.orgId,
|
|
48
|
+
userId: event.userId,
|
|
49
|
+
attributes: event.attributes,
|
|
50
|
+
timestamp: event.timestamp.toISOString(),
|
|
51
|
+
});
|
|
52
|
+
},
|
|
53
|
+
};
|
|
54
|
+
entity_events_1.entityEvents.subscribe(subscriber);
|
|
55
|
+
logger.info('Compliance event subscriber registered', {
|
|
56
|
+
host: serviceConfig.host,
|
|
57
|
+
port: serviceConfig.port,
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tcGxpYW5jZS1ldmVudC1zdWJzY3JpYmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NlcnZpY2VzL2NvbXBsaWFuY2UtZXZlbnQtc3Vic2NyaWJlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7QUFtQnRDLDhFQTBDQztBQTNERCx5REFBNEQ7QUFDNUQsbURBQTZGO0FBQzdGLCtDQUFtRDtBQUVuRCw0Q0FBK0M7QUFFL0MsTUFBTSxNQUFNLEdBQUcsSUFBQSxxQkFBWSxFQUFDLG1CQUFtQixDQUFDLENBQUM7QUFFakQ7Ozs7Ozs7O0dBUUc7QUFDSCxTQUFnQixpQ0FBaUMsQ0FBQyxNQUErQjtJQUMvRSxNQUFNLGFBQWEsR0FBa0I7UUFDbkMsSUFBSSxFQUFFLE1BQU0sRUFBRSxJQUFJLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyx1QkFBdUIsSUFBSSxZQUFZO1FBQ3pFLElBQUksRUFBRSxNQUFNLEVBQUUsSUFBSSxJQUFJLFFBQVEsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLHVCQUF1QixJQUFJLE1BQU0sRUFBRSxFQUFFLENBQUM7S0FDbEYsQ0FBQztJQUVGLE1BQU0sTUFBTSxHQUFHLElBQUksZ0NBQWtCLENBQUMsYUFBYSxDQUFDLENBQUM7SUFFckQsTUFBTSxVQUFVLEdBQTBCO1FBQ3hDLEtBQUssQ0FBQyxhQUFhLENBQUMsS0FBa0I7WUFDcEMsSUFBSSxDQUFDO2dCQUNILE1BQU0sTUFBTSxDQUFDLElBQUksQ0FBQywyQkFBMkIsRUFBRSxLQUFLLEVBQUU7b0JBQ3BELE9BQU8sRUFBRSxFQUFFLG9CQUFvQixFQUFFLE1BQU0sRUFBRTtpQkFDMUMsQ0FBQyxDQUFDO1lBQ0wsQ0FBQztZQUFDLE9BQU8sR0FBRyxFQUFFLENBQUM7Z0JBQ2IsMEVBQTBFO2dCQUMxRSxNQUFNLENBQUMsSUFBSSxDQUFDLHFEQUFxRCxFQUFFO29CQUNqRSxNQUFNLEVBQUUsS0FBSyxDQUFDLE1BQU07b0JBQ3BCLFNBQVMsRUFBRSxLQUFLLENBQUMsU0FBUztvQkFDMUIsUUFBUSxFQUFFLEtBQUssQ0FBQyxRQUFRO29CQUN4QixLQUFLLEVBQUUsR0FBRyxZQUFZLEtBQUssQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQztpQkFDeEQsQ0FBQyxDQUFDO1lBQ0wsQ0FBQztZQUVELDZFQUE2RTtZQUM3RSxLQUFLLElBQUEseUNBQXNCLEVBQUM7Z0JBQzFCLFNBQVMsRUFBRSxVQUFVO2dCQUNyQixNQUFNLEVBQUUsS0FBSyxDQUFDLE1BQStCO2dCQUM3QyxRQUFRLEVBQUUsS0FBSyxDQUFDLFFBQVE7Z0JBQ3hCLEtBQUssRUFBRSxLQUFLLENBQUMsS0FBSztnQkFDbEIsTUFBTSxFQUFFLEtBQUssQ0FBQyxNQUFNO2dCQUNwQixVQUFVLEVBQUUsS0FBSyxDQUFDLFVBQVU7Z0JBQzVCLFNBQVMsRUFBRSxLQUFLLENBQUMsU0FBUyxDQUFDLFdBQVcsRUFBRTthQUN6QyxDQUFDLENBQUM7UUFDTCxDQUFDO0tBQ0YsQ0FBQztJQUVGLDRCQUFZLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBQyxDQUFDO0lBQ25DLE1BQU0sQ0FBQyxJQUFJLENBQUMsd0NBQXdDLEVBQUU7UUFDcEQsSUFBSSxFQUFFLGFBQWEsQ0FBQyxJQUFJO1FBQ3hCLElBQUksRUFBRSxhQUFhLENBQUMsSUFBSTtLQUN6QixDQUFDLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IDIwMjYgUGlwZWxpbmUgQnVpbGRlciBDb250cmlidXRvcnNcbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5cbmltcG9ydCB7IGVucXVldWVDb21wbGlhbmNlRXZlbnQgfSBmcm9tICcuL2NvbXBsaWFuY2UtcXVldWUnO1xuaW1wb3J0IHsgZW50aXR5RXZlbnRzLCB0eXBlIEVudGl0eUV2ZW50LCB0eXBlIEVudGl0eUV2ZW50U3Vic2NyaWJlciB9IGZyb20gJy4vZW50aXR5LWV2ZW50cyc7XG5pbXBvcnQgeyBJbnRlcm5hbEh0dHBDbGllbnQgfSBmcm9tICcuL2h0dHAtY2xpZW50JztcbmltcG9ydCB7IHR5cGUgU2VydmljZUNvbmZpZyB9IGZyb20gJy4uL3R5cGVzL2NvbW1vbic7XG5pbXBvcnQgeyBjcmVhdGVMb2dnZXIgfSBmcm9tICcuLi91dGlscy9sb2dnZXInO1xuXG5jb25zdCBsb2dnZXIgPSBjcmVhdGVMb2dnZXIoJ2NvbXBsaWFuY2UtZXZlbnRzJyk7XG5cbi8qKlxuICogUmVnaXN0ZXJzIGFuIGVudGl0eSBldmVudCBzdWJzY3JpYmVyIHRoYXQgZm9yd2FyZHMgZXZlbnRzIHRvIHRoZSBjb21wbGlhbmNlIHNlcnZpY2UuXG4gKlxuICogQ2FsbCB0aGlzIGF0IHNlcnZpY2Ugc3RhcnR1cCAoaW4gaW5kZXgudHMpIHRvIGVuYWJsZSBhdXRvbWF0aWMgY29tcGxpYW5jZVxuICogbm90aWZpY2F0aW9uIG9uIGVudGl0eSBtdXRhdGlvbnMuIEV2ZW50cyBhcmUgZmlyZS1hbmQtZm9yZ2V0IOKAlCBmYWlsdXJlc1xuICogYXJlIGxvZ2dlZCBidXQgbmV2ZXIgYmxvY2sgdGhlIG9yaWdpbmFsIHJlcXVlc3QuXG4gKlxuICogQHBhcmFtIGNvbmZpZyAtIE9wdGlvbmFsIHNlcnZpY2UgY29uZmlnIG92ZXJyaWRlIChkZWZhdWx0cyB0byBDT01QTElBTkNFX1NFUlZJQ0VfSE9TVC9QT1JUIGVudiB2YXJzKVxuICovXG5leHBvcnQgZnVuY3Rpb24gcmVnaXN0ZXJDb21wbGlhbmNlRXZlbnRTdWJzY3JpYmVyKGNvbmZpZz86IFBhcnRpYWw8U2VydmljZUNvbmZpZz4pOiB2b2lkIHtcbiAgY29uc3Qgc2VydmljZUNvbmZpZzogU2VydmljZUNvbmZpZyA9IHtcbiAgICBob3N0OiBjb25maWc/Lmhvc3QgPz8gcHJvY2Vzcy5lbnYuQ09NUExJQU5DRV9TRVJWSUNFX0hPU1QgPz8gJ2NvbXBsaWFuY2UnLFxuICAgIHBvcnQ6IGNvbmZpZz8ucG9ydCA/PyBwYXJzZUludChwcm9jZXNzLmVudi5DT01QTElBTkNFX1NFUlZJQ0VfUE9SVCA/PyAnMzAwMCcsIDEwKSxcbiAgfTtcblxuICBjb25zdCBjbGllbnQgPSBuZXcgSW50ZXJuYWxIdHRwQ2xpZW50KHNlcnZpY2VDb25maWcpO1xuXG4gIGNvbnN0IHN1YnNjcmliZXI6IEVudGl0eUV2ZW50U3Vic2NyaWJlciA9IHtcbiAgICBhc3luYyBvbkVudGl0eUV2ZW50KGV2ZW50OiBFbnRpdHlFdmVudCk6IFByb21pc2U8dm9pZD4ge1xuICAgICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgY2xpZW50LnBvc3QoJy9jb21wbGlhbmNlL2V2ZW50cy9lbnRpdHknLCBldmVudCwge1xuICAgICAgICAgIGhlYWRlcnM6IHsgJ3gtaW50ZXJuYWwtc2VydmljZSc6ICd0cnVlJyB9LFxuICAgICAgICB9KTtcbiAgICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgICAvLyBGaXJlLWFuZC1mb3JnZXQ6IGxvZyBhbmQgc3dhbGxvdy4gQ29tcGxpYW5jZSBub3RpZmljYXRpb24gaXMgbm9uLWZhdGFsLlxuICAgICAgICBsb2dnZXIud2FybignRmFpbGVkIHRvIG5vdGlmeSBjb21wbGlhbmNlIHNlcnZpY2Ugb2YgZW50aXR5IGV2ZW50Jywge1xuICAgICAgICAgIHRhcmdldDogZXZlbnQudGFyZ2V0LFxuICAgICAgICAgIGV2ZW50VHlwZTogZXZlbnQuZXZlbnRUeXBlLFxuICAgICAgICAgIGVudGl0eUlkOiBldmVudC5lbnRpdHlJZCxcbiAgICAgICAgICBlcnJvcjogZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpLFxuICAgICAgICB9KTtcbiAgICAgIH1cblxuICAgICAgLy8gQWxzbyBlbnF1ZXVlIGZvciBhc3luYyByZS12YWxpZGF0aW9uIChjYXRjaGVzIHJ1bGUgY2hhbmdlcyBhZnRlciBjcmVhdGlvbilcbiAgICAgIHZvaWQgZW5xdWV1ZUNvbXBsaWFuY2VFdmVudCh7XG4gICAgICAgIGV2ZW50VHlwZTogJ3ZhbGlkYXRlJyxcbiAgICAgICAgdGFyZ2V0OiBldmVudC50YXJnZXQgYXMgJ3BsdWdpbicgfCAncGlwZWxpbmUnLFxuICAgICAgICBlbnRpdHlJZDogZXZlbnQuZW50aXR5SWQsXG4gICAgICAgIG9yZ0lkOiBldmVudC5vcmdJZCxcbiAgICAgICAgdXNlcklkOiBldmVudC51c2VySWQsXG4gICAgICAgIGF0dHJpYnV0ZXM6IGV2ZW50LmF0dHJpYnV0ZXMsXG4gICAgICAgIHRpbWVzdGFtcDogZXZlbnQudGltZXN0YW1wLnRvSVNPU3RyaW5nKCksXG4gICAgICB9KTtcbiAgICB9LFxuICB9O1xuXG4gIGVudGl0eUV2ZW50cy5zdWJzY3JpYmUoc3Vic2NyaWJlcik7XG4gIGxvZ2dlci5pbmZvKCdDb21wbGlhbmNlIGV2ZW50IHN1YnNjcmliZXIgcmVnaXN0ZXJlZCcsIHtcbiAgICBob3N0OiBzZXJ2aWNlQ29uZmlnLmhvc3QsXG4gICAgcG9ydDogc2VydmljZUNvbmZpZy5wb3J0LFxuICB9KTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export interface ComplianceEvent {
|
|
2
|
+
eventType: 'validate' | 'scan' | 'notify';
|
|
3
|
+
target: 'plugin' | 'pipeline';
|
|
4
|
+
entityId: string;
|
|
5
|
+
orgId: string;
|
|
6
|
+
userId: string;
|
|
7
|
+
attributes: Record<string, unknown>;
|
|
8
|
+
timestamp: string;
|
|
9
|
+
}
|
|
10
|
+
export declare function registerComplianceQueueBackend(fn: (event: ComplianceEvent) => Promise<void>): void;
|
|
11
|
+
export declare function enqueueComplianceEvent(event: ComplianceEvent): Promise<void>;
|