@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,59 @@
|
|
|
1
|
+
import { QuotaType, QuotaCheckResult } from '../types/common';
|
|
2
|
+
/**
|
|
3
|
+
* Quota service client interface.
|
|
4
|
+
*/
|
|
5
|
+
export interface QuotaService {
|
|
6
|
+
/** Check if quota is available (fail-open on error). */
|
|
7
|
+
check(orgId: string, quotaType: QuotaType, authHeader: string, requestId?: string): Promise<QuotaCheckResult>;
|
|
8
|
+
/** Increment quota usage. Returns a promise so callers can optionally handle errors. */
|
|
9
|
+
increment(orgId: string, quotaType: QuotaType, authHeader: string, amount?: number, requestId?: string): Promise<void>;
|
|
10
|
+
/** Update quota limits. Returns true on success. */
|
|
11
|
+
updateLimits(orgId: string, limits: Partial<Record<QuotaType, number>>, authHeader: string, requestId?: string): Promise<boolean>;
|
|
12
|
+
/** Reset quota usage. Returns true on success. */
|
|
13
|
+
reset(orgId: string, quotaType?: QuotaType, authHeader?: string, requestId?: string): Promise<boolean>;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Configuration for quota service client.
|
|
17
|
+
*/
|
|
18
|
+
export interface QuotaServiceConfig {
|
|
19
|
+
/** Quota service host (default: env QUOTA_SERVICE_HOST or 'quota') */
|
|
20
|
+
host?: string;
|
|
21
|
+
/** Quota service port (default: env QUOTA_SERVICE_PORT or 3000) */
|
|
22
|
+
port?: number;
|
|
23
|
+
/** Request timeout in milliseconds (default: 5000) */
|
|
24
|
+
timeout?: number;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Create a quota service client.
|
|
28
|
+
*
|
|
29
|
+
* @param config - Optional service configuration
|
|
30
|
+
* @returns Quota service client
|
|
31
|
+
*
|
|
32
|
+
* @example
|
|
33
|
+
* ```typescript
|
|
34
|
+
* const quotaService = createQuotaService();
|
|
35
|
+
*
|
|
36
|
+
* // Check quota before processing
|
|
37
|
+
* const quota = await quotaService.check(orgId, 'apiCalls', authHeader);
|
|
38
|
+
* if (!quota.allowed) {
|
|
39
|
+
* return res.status(429).json({ error: 'Quota exceeded' });
|
|
40
|
+
* }
|
|
41
|
+
*
|
|
42
|
+
* // Increment quota after success
|
|
43
|
+
* quotaService.increment(orgId, 'apiCalls', authHeader).catch(err => logger.warn('Quota increment failed', { error: err }));
|
|
44
|
+
* ```
|
|
45
|
+
*/
|
|
46
|
+
export declare function createQuotaService(config?: QuotaServiceConfig): QuotaService;
|
|
47
|
+
/**
|
|
48
|
+
* Fire-and-forget quota increment with standardized error logging.
|
|
49
|
+
*
|
|
50
|
+
* Wraps `quotaService.increment()` with a `.catch()` that logs a warning.
|
|
51
|
+
* Eliminates the identical one-liner repeated across every read route.
|
|
52
|
+
*
|
|
53
|
+
* @param quotaService - Quota service client
|
|
54
|
+
* @param orgId - Organization ID
|
|
55
|
+
* @param quotaType - Quota type to increment
|
|
56
|
+
* @param authHeader - Authorization header value
|
|
57
|
+
* @param logWarn - Logging function for warnings
|
|
58
|
+
*/
|
|
59
|
+
export declare function incrementQuota(quotaService: QuotaService, orgId: string, quotaType: QuotaType, authHeader: string, logWarn: (message: string, data?: unknown) => void): void;
|
|
@@ -0,0 +1,137 @@
|
|
|
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.createQuotaService = createQuotaService;
|
|
6
|
+
exports.incrementQuota = incrementQuota;
|
|
7
|
+
const http_client_1 = require("./http-client");
|
|
8
|
+
const logger_1 = require("../utils/logger");
|
|
9
|
+
/** Retry options for quota calls — fail fast since quota is fail-open. */
|
|
10
|
+
const QUOTA_REQUEST_OPTIONS = {
|
|
11
|
+
maxRateLimitRetries: 1,
|
|
12
|
+
maxRetries: 1,
|
|
13
|
+
};
|
|
14
|
+
const logger = (0, logger_1.createLogger)('quota');
|
|
15
|
+
/**
|
|
16
|
+
* Create a fail-open quota result (allows the request).
|
|
17
|
+
*/
|
|
18
|
+
function createFailOpenResult() {
|
|
19
|
+
return {
|
|
20
|
+
allowed: true,
|
|
21
|
+
limit: -1,
|
|
22
|
+
used: 0,
|
|
23
|
+
remaining: -1,
|
|
24
|
+
resetAt: new Date().toISOString(),
|
|
25
|
+
unlimited: true,
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Build common request headers with optional request ID for distributed tracing.
|
|
30
|
+
*/
|
|
31
|
+
function buildHeaders(orgId, authHeader, requestId) {
|
|
32
|
+
const headers = { 'x-org-id': orgId };
|
|
33
|
+
if (authHeader)
|
|
34
|
+
headers.Authorization = authHeader;
|
|
35
|
+
if (requestId)
|
|
36
|
+
headers['X-Request-Id'] = requestId;
|
|
37
|
+
return headers;
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Create a quota service client.
|
|
41
|
+
*
|
|
42
|
+
* @param config - Optional service configuration
|
|
43
|
+
* @returns Quota service client
|
|
44
|
+
*
|
|
45
|
+
* @example
|
|
46
|
+
* ```typescript
|
|
47
|
+
* const quotaService = createQuotaService();
|
|
48
|
+
*
|
|
49
|
+
* // Check quota before processing
|
|
50
|
+
* const quota = await quotaService.check(orgId, 'apiCalls', authHeader);
|
|
51
|
+
* if (!quota.allowed) {
|
|
52
|
+
* return res.status(429).json({ error: 'Quota exceeded' });
|
|
53
|
+
* }
|
|
54
|
+
*
|
|
55
|
+
* // Increment quota after success
|
|
56
|
+
* quotaService.increment(orgId, 'apiCalls', authHeader).catch(err => logger.warn('Quota increment failed', { error: err }));
|
|
57
|
+
* ```
|
|
58
|
+
*/
|
|
59
|
+
function createQuotaService(config = {}) {
|
|
60
|
+
const serviceConfig = {
|
|
61
|
+
host: config.host ?? process.env.QUOTA_SERVICE_HOST ?? 'quota',
|
|
62
|
+
port: config.port ?? parseInt(process.env.QUOTA_SERVICE_PORT ?? '3000', 10),
|
|
63
|
+
timeout: config.timeout ?? 5000,
|
|
64
|
+
};
|
|
65
|
+
const client = (0, http_client_1.createSafeClient)(serviceConfig);
|
|
66
|
+
return {
|
|
67
|
+
async check(orgId, quotaType, authHeader, requestId) {
|
|
68
|
+
const path = `/quotas/${encodeURIComponent(orgId)}/${encodeURIComponent(quotaType)}`;
|
|
69
|
+
const response = await client.get(path, { headers: buildHeaders(orgId, authHeader, requestId), ...QUOTA_REQUEST_OPTIONS });
|
|
70
|
+
if (!response) {
|
|
71
|
+
logger.warn('QUOTA_FAIL_OPEN: Quota service unreachable, allowing request', { orgId, quotaType });
|
|
72
|
+
return createFailOpenResult();
|
|
73
|
+
}
|
|
74
|
+
if (response.statusCode !== 200 || !response.body.success || !response.body.data?.status) {
|
|
75
|
+
logger.warn('QUOTA_FAIL_OPEN: Quota check returned non-ok, allowing request', {
|
|
76
|
+
orgId, quotaType, statusCode: response.statusCode, message: response.body.message,
|
|
77
|
+
});
|
|
78
|
+
return createFailOpenResult();
|
|
79
|
+
}
|
|
80
|
+
return response.body.data.status;
|
|
81
|
+
},
|
|
82
|
+
async increment(orgId, quotaType, authHeader, amount = 1, requestId) {
|
|
83
|
+
const path = `/quotas/${encodeURIComponent(orgId)}/increment`;
|
|
84
|
+
const response = await client
|
|
85
|
+
.post(path, { quotaType, amount }, { headers: buildHeaders(orgId, authHeader, requestId), ...QUOTA_REQUEST_OPTIONS });
|
|
86
|
+
if (!response || response.statusCode !== 200) {
|
|
87
|
+
logger.warn('Failed to increment quota', {
|
|
88
|
+
orgId, quotaType, amount, statusCode: response?.statusCode,
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
logger.debug('Quota incremented', { orgId, quotaType, amount });
|
|
93
|
+
}
|
|
94
|
+
},
|
|
95
|
+
async updateLimits(orgId, limits, authHeader, requestId) {
|
|
96
|
+
const path = `/quotas/${encodeURIComponent(orgId)}`;
|
|
97
|
+
const response = await client.put(path, limits, { headers: buildHeaders(orgId, authHeader, requestId) });
|
|
98
|
+
if (!response || response.statusCode !== 200) {
|
|
99
|
+
logger.warn('Failed to update quota limits', {
|
|
100
|
+
orgId, limits, statusCode: response?.statusCode,
|
|
101
|
+
});
|
|
102
|
+
return false;
|
|
103
|
+
}
|
|
104
|
+
logger.info('Quota limits updated', { orgId, limits });
|
|
105
|
+
return true;
|
|
106
|
+
},
|
|
107
|
+
async reset(orgId, quotaType, authHeader, requestId) {
|
|
108
|
+
const path = `/quotas/${encodeURIComponent(orgId)}/reset`;
|
|
109
|
+
const body = quotaType ? { quotaType } : {};
|
|
110
|
+
const response = await client.post(path, body, { headers: buildHeaders(orgId, authHeader ?? '', requestId) });
|
|
111
|
+
if (!response || response.statusCode !== 200) {
|
|
112
|
+
logger.warn('Failed to reset quota', {
|
|
113
|
+
orgId, quotaType, statusCode: response?.statusCode,
|
|
114
|
+
});
|
|
115
|
+
return false;
|
|
116
|
+
}
|
|
117
|
+
logger.info('Quota reset', { orgId, quotaType: quotaType ?? 'all' });
|
|
118
|
+
return true;
|
|
119
|
+
},
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Fire-and-forget quota increment with standardized error logging.
|
|
124
|
+
*
|
|
125
|
+
* Wraps `quotaService.increment()` with a `.catch()` that logs a warning.
|
|
126
|
+
* Eliminates the identical one-liner repeated across every read route.
|
|
127
|
+
*
|
|
128
|
+
* @param quotaService - Quota service client
|
|
129
|
+
* @param orgId - Organization ID
|
|
130
|
+
* @param quotaType - Quota type to increment
|
|
131
|
+
* @param authHeader - Authorization header value
|
|
132
|
+
* @param logWarn - Logging function for warnings
|
|
133
|
+
*/
|
|
134
|
+
function incrementQuota(quotaService, orgId, quotaType, authHeader, logWarn) {
|
|
135
|
+
quotaService.increment(orgId, quotaType, authHeader).catch((err) => logWarn('Quota increment failed', { error: err instanceof Error ? err.message : String(err) }));
|
|
136
|
+
}
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quota.js","sourceRoot":"","sources":["../../src/services/quota.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createSafeClient, RequestOptions } from './http-client';\nimport { QuotaType, QuotaCheckResult, ServiceConfig } from '../types/common';\nimport { createLogger } from '../utils/logger';\n\n/** Retry options for quota calls — fail fast since quota is fail-open. */\nconst QUOTA_REQUEST_OPTIONS: Pick<RequestOptions, 'maxRateLimitRetries' | 'maxRetries'> = {\n  maxRateLimitRetries: 1,\n  maxRetries: 1,\n};\n\nconst logger = createLogger('quota');\n\n/**\n * Quota service client interface.\n */\nexport interface QuotaService {\n  /** Check if quota is available (fail-open on error). */\n  check(orgId: string, quotaType: QuotaType, authHeader: string, requestId?: string): Promise<QuotaCheckResult>;\n  /** Increment quota usage. Returns a promise so callers can optionally handle errors. */\n  increment(orgId: string, quotaType: QuotaType, authHeader: string, amount?: number, requestId?: string): Promise<void>;\n  /** Update quota limits. Returns true on success. */\n  updateLimits(orgId: string, limits: Partial<Record<QuotaType, number>>, authHeader: string, requestId?: string): Promise<boolean>;\n  /** Reset quota usage. Returns true on success. */\n  reset(orgId: string, quotaType?: QuotaType, authHeader?: string, requestId?: string): Promise<boolean>;\n}\n\n/**\n * Configuration for quota service client.\n */\nexport interface QuotaServiceConfig {\n  /** Quota service host (default: env QUOTA_SERVICE_HOST or 'quota') */\n  host?: string;\n  /** Quota service port (default: env QUOTA_SERVICE_PORT or 3000) */\n  port?: number;\n  /** Request timeout in milliseconds (default: 5000) */\n  timeout?: number;\n}\n\n/**\n * Create a fail-open quota result (allows the request).\n */\nfunction createFailOpenResult(): QuotaCheckResult {\n  return {\n    allowed: true,\n    limit: -1,\n    used: 0,\n    remaining: -1,\n    resetAt: new Date().toISOString(),\n    unlimited: true,\n  };\n}\n\n/**\n * Build common request headers with optional request ID for distributed tracing.\n */\nfunction buildHeaders(orgId: string, authHeader?: string, requestId?: string): Record<string, string> {\n  const headers: Record<string, string> = { 'x-org-id': orgId };\n  if (authHeader) headers.Authorization = authHeader;\n  if (requestId) headers['X-Request-Id'] = requestId;\n  return headers;\n}\n\n/**\n * Create a quota service client.\n *\n * @param config - Optional service configuration\n * @returns Quota service client\n *\n * @example\n * ```typescript\n * const quotaService = createQuotaService();\n *\n * // Check quota before processing\n * const quota = await quotaService.check(orgId, 'apiCalls', authHeader);\n * if (!quota.allowed) {\n *   return res.status(429).json({ error: 'Quota exceeded' });\n * }\n *\n * // Increment quota after success\n * quotaService.increment(orgId, 'apiCalls', authHeader).catch(err => logger.warn('Quota increment failed', { error: err }));\n * ```\n */\nexport function createQuotaService(config: QuotaServiceConfig = {}): QuotaService {\n  const serviceConfig: ServiceConfig = {\n    host: config.host ?? process.env.QUOTA_SERVICE_HOST ?? 'quota',\n    port: config.port ?? parseInt(process.env.QUOTA_SERVICE_PORT ?? '3000', 10),\n    timeout: config.timeout ?? 5000,\n  };\n\n  const client = createSafeClient(serviceConfig);\n\n  return {\n    async check(orgId: string, quotaType: QuotaType, authHeader: string, requestId?: string): Promise<QuotaCheckResult> {\n      const path = `/quotas/${encodeURIComponent(orgId)}/${encodeURIComponent(quotaType)}`;\n\n      const response = await client.get<{\n        success: boolean;\n        data?: { quotaType: string; status: QuotaCheckResult };\n        message?: string;\n      }>(path, { headers: buildHeaders(orgId, authHeader, requestId), ...QUOTA_REQUEST_OPTIONS });\n\n      if (!response) {\n        logger.warn('QUOTA_FAIL_OPEN: Quota service unreachable, allowing request', { orgId, quotaType });\n        return createFailOpenResult();\n      }\n\n      if (response.statusCode !== 200 || !response.body.success || !response.body.data?.status) {\n        logger.warn('QUOTA_FAIL_OPEN: Quota check returned non-ok, allowing request', {\n          orgId, quotaType, statusCode: response.statusCode, message: response.body.message,\n        });\n        return createFailOpenResult();\n      }\n\n      return response.body.data.status;\n    },\n\n    async increment(orgId: string, quotaType: QuotaType, authHeader: string, amount: number = 1, requestId?: string): Promise<void> {\n      const path = `/quotas/${encodeURIComponent(orgId)}/increment`;\n\n      const response = await client\n        .post(path, { quotaType, amount }, { headers: buildHeaders(orgId, authHeader, requestId), ...QUOTA_REQUEST_OPTIONS });\n\n      if (!response || response.statusCode !== 200) {\n        logger.warn('Failed to increment quota', {\n          orgId, quotaType, amount, statusCode: response?.statusCode,\n        });\n      } else {\n        logger.debug('Quota incremented', { orgId, quotaType, amount });\n      }\n    },\n\n    async updateLimits(\n      orgId: string,\n      limits: Partial<Record<QuotaType, number>>,\n      authHeader: string,\n      requestId?: string,\n    ): Promise<boolean> {\n      const path = `/quotas/${encodeURIComponent(orgId)}`;\n\n      const response = await client.put(path, limits, { headers: buildHeaders(orgId, authHeader, requestId) });\n\n      if (!response || response.statusCode !== 200) {\n        logger.warn('Failed to update quota limits', {\n          orgId, limits, statusCode: response?.statusCode,\n        });\n        return false;\n      }\n\n      logger.info('Quota limits updated', { orgId, limits });\n      return true;\n    },\n\n    async reset(orgId: string, quotaType?: QuotaType, authHeader?: string, requestId?: string): Promise<boolean> {\n      const path = `/quotas/${encodeURIComponent(orgId)}/reset`;\n\n      const body = quotaType ? { quotaType } : {};\n      const response = await client.post(path, body, { headers: buildHeaders(orgId, authHeader ?? '', requestId) });\n\n      if (!response || response.statusCode !== 200) {\n        logger.warn('Failed to reset quota', {\n          orgId, quotaType, statusCode: response?.statusCode,\n        });\n        return false;\n      }\n\n      logger.info('Quota reset', { orgId, quotaType: quotaType ?? 'all' });\n      return true;\n    },\n  };\n}\n\n/**\n * Fire-and-forget quota increment with standardized error logging.\n *\n * Wraps `quotaService.increment()` with a `.catch()` that logs a warning.\n * Eliminates the identical one-liner repeated across every read route.\n *\n * @param quotaService - Quota service client\n * @param orgId - Organization ID\n * @param quotaType - Quota type to increment\n * @param authHeader - Authorization header value\n * @param logWarn - Logging function for warnings\n */\nexport function incrementQuota(\n  quotaService: QuotaService,\n  orgId: string,\n  quotaType: QuotaType,\n  authHeader: string,\n  logWarn: (message: string, data?: unknown) => void,\n): void {\n  quotaService.increment(orgId, quotaType, authHeader).catch((err: unknown) =>\n    logWarn('Quota increment failed', { error: err instanceof Error ? err.message : String(err) }),\n  );\n}\n"]}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import * as http from 'http';
|
|
2
|
+
/**
|
|
3
|
+
* Default retry configuration (env: `HTTP_CLIENT_MAX_RETRIES`, `HTTP_CLIENT_RETRY_DELAY_MS`).
|
|
4
|
+
*/
|
|
5
|
+
export declare const DEFAULT_MAX_RETRIES: number;
|
|
6
|
+
export declare const DEFAULT_RETRY_DELAY_MS: number;
|
|
7
|
+
export declare const DEFAULT_MAX_RATE_LIMIT_RETRIES: number;
|
|
8
|
+
/**
|
|
9
|
+
* Configuration for retry behavior.
|
|
10
|
+
*/
|
|
11
|
+
export interface RetryConfig {
|
|
12
|
+
/** Maximum retry attempts for transient failures (default: 2) */
|
|
13
|
+
maxRetries: number;
|
|
14
|
+
/** Base delay between retries in ms — doubles each attempt (default: 200) */
|
|
15
|
+
retryDelayMs: number;
|
|
16
|
+
/** Maximum retry attempts specifically for 429 rate limiting (default: 4) */
|
|
17
|
+
maxRateLimitRetries: number;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Result of a retry decision: whether to retry, and how long to wait.
|
|
21
|
+
*/
|
|
22
|
+
export interface RetryDecision {
|
|
23
|
+
/** Whether the request should be retried */
|
|
24
|
+
shouldRetry: boolean;
|
|
25
|
+
/** Delay in milliseconds before retrying (with jitter applied) */
|
|
26
|
+
delayMs: number;
|
|
27
|
+
/** Human-readable reason for the retry decision */
|
|
28
|
+
reason: string;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Parse a `Retry-After` header value into milliseconds.
|
|
32
|
+
* Supports numeric seconds (e.g. "5") and HTTP-date format.
|
|
33
|
+
* Returns `undefined` for missing or invalid values.
|
|
34
|
+
*/
|
|
35
|
+
export declare function parseRetryAfter(header: string | string[] | undefined): number | undefined;
|
|
36
|
+
/**
|
|
37
|
+
* Apply +/-25% random jitter to a delay to prevent thundering herd.
|
|
38
|
+
*/
|
|
39
|
+
export declare function addJitter(delay: number): number;
|
|
40
|
+
/**
|
|
41
|
+
* Calculate exponential backoff delay for a given attempt.
|
|
42
|
+
*
|
|
43
|
+
* @param baseDelay - Base delay in milliseconds
|
|
44
|
+
* @param attempt - Zero-based attempt number
|
|
45
|
+
* @returns Delay in milliseconds (without jitter)
|
|
46
|
+
*/
|
|
47
|
+
export declare function calculateBackoff(baseDelay: number, attempt: number): number;
|
|
48
|
+
/**
|
|
49
|
+
* Determine whether a status code represents a transient server error
|
|
50
|
+
* that is eligible for retry (502, 503, 504).
|
|
51
|
+
*/
|
|
52
|
+
export declare function isTransientStatusCode(statusCode: number): boolean;
|
|
53
|
+
/**
|
|
54
|
+
* Determine whether a status code indicates rate limiting (429).
|
|
55
|
+
*/
|
|
56
|
+
export declare function isRateLimited(statusCode: number): boolean;
|
|
57
|
+
/**
|
|
58
|
+
* Determine whether a failed response should be retried, and compute the delay.
|
|
59
|
+
*
|
|
60
|
+
* @param statusCode - HTTP status code of the response
|
|
61
|
+
* @param headers - Response headers (used to read Retry-After for 429)
|
|
62
|
+
* @param attempt - Zero-based attempt number
|
|
63
|
+
* @param config - Retry configuration
|
|
64
|
+
* @returns A RetryDecision indicating whether to retry and the delay
|
|
65
|
+
*/
|
|
66
|
+
export declare function getRetryDecision(statusCode: number, headers: http.IncomingHttpHeaders, attempt: number, config: RetryConfig): RetryDecision;
|
|
67
|
+
/**
|
|
68
|
+
* Determine whether a connection/timeout error should be retried, and compute the delay.
|
|
69
|
+
*
|
|
70
|
+
* @param attempt - Zero-based attempt number
|
|
71
|
+
* @param config - Retry configuration
|
|
72
|
+
* @returns A RetryDecision indicating whether to retry and the delay
|
|
73
|
+
*/
|
|
74
|
+
export declare function getErrorRetryDecision(attempt: number, config: RetryConfig): RetryDecision;
|
|
@@ -0,0 +1,127 @@
|
|
|
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.DEFAULT_MAX_RATE_LIMIT_RETRIES = exports.DEFAULT_RETRY_DELAY_MS = exports.DEFAULT_MAX_RETRIES = void 0;
|
|
6
|
+
exports.parseRetryAfter = parseRetryAfter;
|
|
7
|
+
exports.addJitter = addJitter;
|
|
8
|
+
exports.calculateBackoff = calculateBackoff;
|
|
9
|
+
exports.isTransientStatusCode = isTransientStatusCode;
|
|
10
|
+
exports.isRateLimited = isRateLimited;
|
|
11
|
+
exports.getRetryDecision = getRetryDecision;
|
|
12
|
+
exports.getErrorRetryDecision = getErrorRetryDecision;
|
|
13
|
+
/**
|
|
14
|
+
* Default retry configuration (env: `HTTP_CLIENT_MAX_RETRIES`, `HTTP_CLIENT_RETRY_DELAY_MS`).
|
|
15
|
+
*/
|
|
16
|
+
exports.DEFAULT_MAX_RETRIES = parseInt(process.env.HTTP_CLIENT_MAX_RETRIES || '2', 10);
|
|
17
|
+
exports.DEFAULT_RETRY_DELAY_MS = parseInt(process.env.HTTP_CLIENT_RETRY_DELAY_MS || '200', 10);
|
|
18
|
+
exports.DEFAULT_MAX_RATE_LIMIT_RETRIES = parseInt(process.env.HTTP_CLIENT_MAX_RATE_LIMIT_RETRIES || '4', 10);
|
|
19
|
+
/** Max Retry-After value we'll honor (60 seconds). */
|
|
20
|
+
const MAX_RETRY_AFTER_MS = 60_000;
|
|
21
|
+
/** HTTP status codes considered transient server errors eligible for retry. */
|
|
22
|
+
const TRANSIENT_STATUS_CODES = [502, 503, 504];
|
|
23
|
+
/**
|
|
24
|
+
* Parse a `Retry-After` header value into milliseconds.
|
|
25
|
+
* Supports numeric seconds (e.g. "5") and HTTP-date format.
|
|
26
|
+
* Returns `undefined` for missing or invalid values.
|
|
27
|
+
*/
|
|
28
|
+
function parseRetryAfter(header) {
|
|
29
|
+
if (!header)
|
|
30
|
+
return undefined;
|
|
31
|
+
const value = Array.isArray(header) ? header[0] : header;
|
|
32
|
+
if (!value)
|
|
33
|
+
return undefined;
|
|
34
|
+
// Try numeric seconds first
|
|
35
|
+
const seconds = Number(value);
|
|
36
|
+
if (!isNaN(seconds) && seconds >= 0) {
|
|
37
|
+
return Math.min(seconds * 1000, MAX_RETRY_AFTER_MS);
|
|
38
|
+
}
|
|
39
|
+
// Try HTTP-date
|
|
40
|
+
const date = Date.parse(value);
|
|
41
|
+
if (!isNaN(date)) {
|
|
42
|
+
const delayMs = date - Date.now();
|
|
43
|
+
if (delayMs > 0)
|
|
44
|
+
return Math.min(delayMs, MAX_RETRY_AFTER_MS);
|
|
45
|
+
}
|
|
46
|
+
return undefined;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Apply +/-25% random jitter to a delay to prevent thundering herd.
|
|
50
|
+
*/
|
|
51
|
+
function addJitter(delay) {
|
|
52
|
+
const jitter = delay * 0.25 * (2 * Math.random() - 1); // -25% to +25%
|
|
53
|
+
return Math.max(0, Math.round(delay + jitter));
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Calculate exponential backoff delay for a given attempt.
|
|
57
|
+
*
|
|
58
|
+
* @param baseDelay - Base delay in milliseconds
|
|
59
|
+
* @param attempt - Zero-based attempt number
|
|
60
|
+
* @returns Delay in milliseconds (without jitter)
|
|
61
|
+
*/
|
|
62
|
+
function calculateBackoff(baseDelay, attempt) {
|
|
63
|
+
return baseDelay * Math.pow(2, attempt);
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Determine whether a status code represents a transient server error
|
|
67
|
+
* that is eligible for retry (502, 503, 504).
|
|
68
|
+
*/
|
|
69
|
+
function isTransientStatusCode(statusCode) {
|
|
70
|
+
return TRANSIENT_STATUS_CODES.includes(statusCode);
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Determine whether a status code indicates rate limiting (429).
|
|
74
|
+
*/
|
|
75
|
+
function isRateLimited(statusCode) {
|
|
76
|
+
return statusCode === 429;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Determine whether a failed response should be retried, and compute the delay.
|
|
80
|
+
*
|
|
81
|
+
* @param statusCode - HTTP status code of the response
|
|
82
|
+
* @param headers - Response headers (used to read Retry-After for 429)
|
|
83
|
+
* @param attempt - Zero-based attempt number
|
|
84
|
+
* @param config - Retry configuration
|
|
85
|
+
* @returns A RetryDecision indicating whether to retry and the delay
|
|
86
|
+
*/
|
|
87
|
+
function getRetryDecision(statusCode, headers, attempt, config) {
|
|
88
|
+
// 429 rate limiting — use Retry-After or longer backoff (4x base)
|
|
89
|
+
if (isRateLimited(statusCode) && attempt < config.maxRateLimitRetries) {
|
|
90
|
+
const retryAfter = parseRetryAfter(headers['retry-after']);
|
|
91
|
+
const rawDelay = retryAfter ?? (config.retryDelayMs * 4 * Math.pow(2, attempt));
|
|
92
|
+
return {
|
|
93
|
+
shouldRetry: true,
|
|
94
|
+
delayMs: addJitter(rawDelay),
|
|
95
|
+
reason: 'Rate limited (429)',
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
// 5xx transient server errors — standard exponential backoff
|
|
99
|
+
if (isTransientStatusCode(statusCode) && attempt < config.maxRetries) {
|
|
100
|
+
const rawDelay = calculateBackoff(config.retryDelayMs, attempt);
|
|
101
|
+
return {
|
|
102
|
+
shouldRetry: true,
|
|
103
|
+
delayMs: addJitter(rawDelay),
|
|
104
|
+
reason: `Transient server error (${statusCode})`,
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
return { shouldRetry: false, delayMs: 0, reason: 'Not retryable' };
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Determine whether a connection/timeout error should be retried, and compute the delay.
|
|
111
|
+
*
|
|
112
|
+
* @param attempt - Zero-based attempt number
|
|
113
|
+
* @param config - Retry configuration
|
|
114
|
+
* @returns A RetryDecision indicating whether to retry and the delay
|
|
115
|
+
*/
|
|
116
|
+
function getErrorRetryDecision(attempt, config) {
|
|
117
|
+
if (attempt < config.maxRetries) {
|
|
118
|
+
const rawDelay = calculateBackoff(config.retryDelayMs, attempt);
|
|
119
|
+
return {
|
|
120
|
+
shouldRetry: true,
|
|
121
|
+
delayMs: addJitter(rawDelay),
|
|
122
|
+
reason: 'Connection or timeout error',
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
return { shouldRetry: false, delayMs: 0, reason: 'Max retries exceeded' };
|
|
126
|
+
}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"retry-strategy.js","sourceRoot":"","sources":["../../src/services/retry-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport * as http from 'http';\n\n/**\n * Default retry configuration (env: `HTTP_CLIENT_MAX_RETRIES`, `HTTP_CLIENT_RETRY_DELAY_MS`).\n */\nexport const DEFAULT_MAX_RETRIES = parseInt(process.env.HTTP_CLIENT_MAX_RETRIES || '2', 10);\nexport const DEFAULT_RETRY_DELAY_MS = parseInt(process.env.HTTP_CLIENT_RETRY_DELAY_MS || '200', 10);\nexport const DEFAULT_MAX_RATE_LIMIT_RETRIES = parseInt(process.env.HTTP_CLIENT_MAX_RATE_LIMIT_RETRIES || '4', 10);\n\n/** Max Retry-After value we'll honor (60 seconds). */\nconst MAX_RETRY_AFTER_MS = 60_000;\n\n/** HTTP status codes considered transient server errors eligible for retry. */\nconst TRANSIENT_STATUS_CODES = [502, 503, 504];\n\n/**\n * Configuration for retry behavior.\n */\nexport interface RetryConfig {\n  /** Maximum retry attempts for transient failures (default: 2) */\n  maxRetries: number;\n  /** Base delay between retries in ms — doubles each attempt (default: 200) */\n  retryDelayMs: number;\n  /** Maximum retry attempts specifically for 429 rate limiting (default: 4) */\n  maxRateLimitRetries: number;\n}\n\n/**\n * Result of a retry decision: whether to retry, and how long to wait.\n */\nexport interface RetryDecision {\n  /** Whether the request should be retried */\n  shouldRetry: boolean;\n  /** Delay in milliseconds before retrying (with jitter applied) */\n  delayMs: number;\n  /** Human-readable reason for the retry decision */\n  reason: string;\n}\n\n/**\n * Parse a `Retry-After` header value into milliseconds.\n * Supports numeric seconds (e.g. \"5\") and HTTP-date format.\n * Returns `undefined` for missing or invalid values.\n */\nexport function parseRetryAfter(header: string | string[] | undefined): number | undefined {\n  if (!header) return undefined;\n  const value = Array.isArray(header) ? header[0] : header;\n  if (!value) return undefined;\n\n  // Try numeric seconds first\n  const seconds = Number(value);\n  if (!isNaN(seconds) && seconds >= 0) {\n    return Math.min(seconds * 1000, MAX_RETRY_AFTER_MS);\n  }\n\n  // Try HTTP-date\n  const date = Date.parse(value);\n  if (!isNaN(date)) {\n    const delayMs = date - Date.now();\n    if (delayMs > 0) return Math.min(delayMs, MAX_RETRY_AFTER_MS);\n  }\n\n  return undefined;\n}\n\n/**\n * Apply +/-25% random jitter to a delay to prevent thundering herd.\n */\nexport function addJitter(delay: number): number {\n  const jitter = delay * 0.25 * (2 * Math.random() - 1); // -25% to +25%\n  return Math.max(0, Math.round(delay + jitter));\n}\n\n/**\n * Calculate exponential backoff delay for a given attempt.\n *\n * @param baseDelay - Base delay in milliseconds\n * @param attempt - Zero-based attempt number\n * @returns Delay in milliseconds (without jitter)\n */\nexport function calculateBackoff(baseDelay: number, attempt: number): number {\n  return baseDelay * Math.pow(2, attempt);\n}\n\n/**\n * Determine whether a status code represents a transient server error\n * that is eligible for retry (502, 503, 504).\n */\nexport function isTransientStatusCode(statusCode: number): boolean {\n  return TRANSIENT_STATUS_CODES.includes(statusCode);\n}\n\n/**\n * Determine whether a status code indicates rate limiting (429).\n */\nexport function isRateLimited(statusCode: number): boolean {\n  return statusCode === 429;\n}\n\n/**\n * Determine whether a failed response should be retried, and compute the delay.\n *\n * @param statusCode - HTTP status code of the response\n * @param headers - Response headers (used to read Retry-After for 429)\n * @param attempt - Zero-based attempt number\n * @param config - Retry configuration\n * @returns A RetryDecision indicating whether to retry and the delay\n */\nexport function getRetryDecision(\n  statusCode: number,\n  headers: http.IncomingHttpHeaders,\n  attempt: number,\n  config: RetryConfig,\n): RetryDecision {\n  // 429 rate limiting — use Retry-After or longer backoff (4x base)\n  if (isRateLimited(statusCode) && attempt < config.maxRateLimitRetries) {\n    const retryAfter = parseRetryAfter(headers['retry-after']);\n    const rawDelay = retryAfter ?? (config.retryDelayMs * 4 * Math.pow(2, attempt));\n    return {\n      shouldRetry: true,\n      delayMs: addJitter(rawDelay),\n      reason: 'Rate limited (429)',\n    };\n  }\n\n  // 5xx transient server errors — standard exponential backoff\n  if (isTransientStatusCode(statusCode) && attempt < config.maxRetries) {\n    const rawDelay = calculateBackoff(config.retryDelayMs, attempt);\n    return {\n      shouldRetry: true,\n      delayMs: addJitter(rawDelay),\n      reason: `Transient server error (${statusCode})`,\n    };\n  }\n\n  return { shouldRetry: false, delayMs: 0, reason: 'Not retryable' };\n}\n\n/**\n * Determine whether a connection/timeout error should be retried, and compute the delay.\n *\n * @param attempt - Zero-based attempt number\n * @param config - Retry configuration\n * @returns A RetryDecision indicating whether to retry and the delay\n */\nexport function getErrorRetryDecision(\n  attempt: number,\n  config: RetryConfig,\n): RetryDecision {\n  if (attempt < config.maxRetries) {\n    const rawDelay = calculateBackoff(config.retryDelayMs, attempt);\n    return {\n      shouldRetry: true,\n      delayMs: addJitter(rawDelay),\n      reason: 'Connection or timeout error',\n    };\n  }\n\n  return { shouldRetry: false, delayMs: 0, reason: 'Max retries exceeded' };\n}\n"]}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import type { QuotaTier } from './quota-tiers';
|
|
2
|
+
/** Billing interval for subscriptions. */
|
|
3
|
+
export type BillingInterval = 'monthly' | 'annual';
|
|
4
|
+
/** Subscription lifecycle status. */
|
|
5
|
+
export type SubscriptionStatus = 'active' | 'canceled' | 'past_due' | 'trialing' | 'incomplete';
|
|
6
|
+
/** Payment transaction status. */
|
|
7
|
+
export type PaymentStatus = 'succeeded' | 'pending' | 'failed' | 'refunded';
|
|
8
|
+
/** Billing event types for audit logging. */
|
|
9
|
+
export type BillingEventType = 'subscription_created' | 'subscription_updated' | 'subscription_canceled' | 'subscription_reactivated' | 'plan_changed' | 'interval_changed' | 'payment_succeeded' | 'payment_failed';
|
|
10
|
+
/** Price definition for a plan (in cents). */
|
|
11
|
+
export interface PlanPrices {
|
|
12
|
+
monthly: number;
|
|
13
|
+
annual: number;
|
|
14
|
+
}
|
|
15
|
+
/** Plan definition returned by the billing API. */
|
|
16
|
+
export interface PlanDefinition {
|
|
17
|
+
id: string;
|
|
18
|
+
name: string;
|
|
19
|
+
description: string;
|
|
20
|
+
tier: QuotaTier;
|
|
21
|
+
prices: PlanPrices;
|
|
22
|
+
features: string[];
|
|
23
|
+
isDefault: boolean;
|
|
24
|
+
sortOrder: number;
|
|
25
|
+
}
|
|
26
|
+
/** Subscription info returned by the billing API. */
|
|
27
|
+
export interface SubscriptionInfo {
|
|
28
|
+
id: string;
|
|
29
|
+
orgId: string;
|
|
30
|
+
planId: string;
|
|
31
|
+
status: SubscriptionStatus;
|
|
32
|
+
interval: BillingInterval;
|
|
33
|
+
currentPeriodStart: string;
|
|
34
|
+
currentPeriodEnd: string;
|
|
35
|
+
cancelAtPeriodEnd: boolean;
|
|
36
|
+
createdAt: string;
|
|
37
|
+
updatedAt: string;
|
|
38
|
+
}
|
|
39
|
+
/** Billing event info returned by the admin API. */
|
|
40
|
+
export interface BillingEventInfo {
|
|
41
|
+
id: string;
|
|
42
|
+
orgId: string;
|
|
43
|
+
subscriptionId?: string;
|
|
44
|
+
type: BillingEventType;
|
|
45
|
+
details: Record<string, unknown>;
|
|
46
|
+
createdAt: string;
|
|
47
|
+
}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright 2026 Pipeline Builder Contributors
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmlsbGluZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90eXBlcy9iaWxsaW5nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IDIwMjYgUGlwZWxpbmUgQnVpbGRlciBDb250cmlidXRvcnNcbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5cbmltcG9ydCB0eXBlIHsgUXVvdGFUaWVyIH0gZnJvbSAnLi9xdW90YS10aWVycyc7XG5cbi8qKiBCaWxsaW5nIGludGVydmFsIGZvciBzdWJzY3JpcHRpb25zLiAqL1xuZXhwb3J0IHR5cGUgQmlsbGluZ0ludGVydmFsID0gJ21vbnRobHknIHwgJ2FubnVhbCc7XG5cbi8qKiBTdWJzY3JpcHRpb24gbGlmZWN5Y2xlIHN0YXR1cy4gKi9cbmV4cG9ydCB0eXBlIFN1YnNjcmlwdGlvblN0YXR1cyA9ICdhY3RpdmUnIHwgJ2NhbmNlbGVkJyB8ICdwYXN0X2R1ZScgfCAndHJpYWxpbmcnIHwgJ2luY29tcGxldGUnO1xuXG4vKiogUGF5bWVudCB0cmFuc2FjdGlvbiBzdGF0dXMuICovXG5leHBvcnQgdHlwZSBQYXltZW50U3RhdHVzID0gJ3N1Y2NlZWRlZCcgfCAncGVuZGluZycgfCAnZmFpbGVkJyB8ICdyZWZ1bmRlZCc7XG5cbi8qKiBCaWxsaW5nIGV2ZW50IHR5cGVzIGZvciBhdWRpdCBsb2dnaW5nLiAqL1xuZXhwb3J0IHR5cGUgQmlsbGluZ0V2ZW50VHlwZSA9XG4gIHwgJ3N1YnNjcmlwdGlvbl9jcmVhdGVkJ1xuICB8ICdzdWJzY3JpcHRpb25fdXBkYXRlZCdcbiAgfCAnc3Vic2NyaXB0aW9uX2NhbmNlbGVkJ1xuICB8ICdzdWJzY3JpcHRpb25fcmVhY3RpdmF0ZWQnXG4gIHwgJ3BsYW5fY2hhbmdlZCdcbiAgfCAnaW50ZXJ2YWxfY2hhbmdlZCdcbiAgfCAncGF5bWVudF9zdWNjZWVkZWQnXG4gIHwgJ3BheW1lbnRfZmFpbGVkJztcblxuLyoqIFByaWNlIGRlZmluaXRpb24gZm9yIGEgcGxhbiAoaW4gY2VudHMpLiAqL1xuZXhwb3J0IGludGVyZmFjZSBQbGFuUHJpY2VzIHtcbiAgbW9udGhseTogbnVtYmVyO1xuICBhbm51YWw6IG51bWJlcjtcbn1cblxuLyoqIFBsYW4gZGVmaW5pdGlvbiByZXR1cm5lZCBieSB0aGUgYmlsbGluZyBBUEkuICovXG5leHBvcnQgaW50ZXJmYWNlIFBsYW5EZWZpbml0aW9uIHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICB0aWVyOiBRdW90YVRpZXI7XG4gIHByaWNlczogUGxhblByaWNlcztcbiAgZmVhdHVyZXM6IHN0cmluZ1tdO1xuICBpc0RlZmF1bHQ6IGJvb2xlYW47XG4gIHNvcnRPcmRlcjogbnVtYmVyO1xufVxuXG4vKiogU3Vic2NyaXB0aW9uIGluZm8gcmV0dXJuZWQgYnkgdGhlIGJpbGxpbmcgQVBJLiAqL1xuZXhwb3J0IGludGVyZmFjZSBTdWJzY3JpcHRpb25JbmZvIHtcbiAgaWQ6IHN0cmluZztcbiAgb3JnSWQ6IHN0cmluZztcbiAgcGxhbklkOiBzdHJpbmc7XG4gIHN0YXR1czogU3Vic2NyaXB0aW9uU3RhdHVzO1xuICBpbnRlcnZhbDogQmlsbGluZ0ludGVydmFsO1xuICBjdXJyZW50UGVyaW9kU3RhcnQ6IHN0cmluZztcbiAgY3VycmVudFBlcmlvZEVuZDogc3RyaW5nO1xuICBjYW5jZWxBdFBlcmlvZEVuZDogYm9vbGVhbjtcbiAgY3JlYXRlZEF0OiBzdHJpbmc7XG4gIHVwZGF0ZWRBdDogc3RyaW5nO1xufVxuXG4vKiogQmlsbGluZyBldmVudCBpbmZvIHJldHVybmVkIGJ5IHRoZSBhZG1pbiBBUEkuICovXG5leHBvcnQgaW50ZXJmYWNlIEJpbGxpbmdFdmVudEluZm8ge1xuICBpZDogc3RyaW5nO1xuICBvcmdJZDogc3RyaW5nO1xuICBzdWJzY3JpcHRpb25JZD86IHN0cmluZztcbiAgdHlwZTogQmlsbGluZ0V2ZW50VHlwZTtcbiAgZGV0YWlsczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIGNyZWF0ZWRBdDogc3RyaW5nO1xufVxuIl19
|
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Quota type identifiers.
|
|
3
|
+
*/
|
|
4
|
+
export type QuotaType = 'plugins' | 'pipelines' | 'apiCalls';
|
|
5
|
+
/**
|
|
6
|
+
* Valid quota type values.
|
|
7
|
+
*/
|
|
8
|
+
export declare const VALID_QUOTA_TYPES: readonly ["plugins", "pipelines", "apiCalls"];
|
|
9
|
+
/**
|
|
10
|
+
* Type guard to check if a value is a valid QuotaType.
|
|
11
|
+
*
|
|
12
|
+
* @param value - Value to check
|
|
13
|
+
* @returns True if value is a valid QuotaType
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* ```typescript
|
|
17
|
+
* if (isValidQuotaType(req.body.quotaType)) {
|
|
18
|
+
* // quotaType is guaranteed to be QuotaType
|
|
19
|
+
* }
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
export declare function isValidQuotaType(value: unknown): value is QuotaType;
|
|
23
|
+
/**
|
|
24
|
+
* Validate and assert that a value is a valid QuotaType.
|
|
25
|
+
* Throws an error if validation fails.
|
|
26
|
+
*
|
|
27
|
+
* @param value - Value to validate
|
|
28
|
+
* @param fieldName - Name of the field being validated (for error messages)
|
|
29
|
+
* @returns The validated QuotaType
|
|
30
|
+
* @throws Error if value is not a valid QuotaType
|
|
31
|
+
*
|
|
32
|
+
* @example
|
|
33
|
+
* ```typescript
|
|
34
|
+
* try {
|
|
35
|
+
* const quotaType = validateQuotaType(req.body.quotaType, 'quotaType');
|
|
36
|
+
* // Use quotaType safely
|
|
37
|
+
* } catch (err) {
|
|
38
|
+
* return sendError(res, 400, err.message);
|
|
39
|
+
* }
|
|
40
|
+
* ```
|
|
41
|
+
*/
|
|
42
|
+
export declare function validateQuotaType(value: unknown, fieldName?: string): QuotaType;
|
|
43
|
+
/**
|
|
44
|
+
* Result from quota check operation.
|
|
45
|
+
*/
|
|
46
|
+
export interface QuotaCheckResult {
|
|
47
|
+
/** Whether the request is allowed */
|
|
48
|
+
allowed: boolean;
|
|
49
|
+
/** Maximum quota limit (-1 for unlimited) */
|
|
50
|
+
limit: number;
|
|
51
|
+
/** Current usage count */
|
|
52
|
+
used: number;
|
|
53
|
+
/** Remaining quota (-1 for unlimited) */
|
|
54
|
+
remaining: number;
|
|
55
|
+
/** ISO timestamp when quota resets */
|
|
56
|
+
resetAt: string;
|
|
57
|
+
/** Whether quota is unlimited */
|
|
58
|
+
unlimited: boolean;
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Quota information for error responses.
|
|
62
|
+
*/
|
|
63
|
+
export interface QuotaInfo {
|
|
64
|
+
type: QuotaType;
|
|
65
|
+
limit: number;
|
|
66
|
+
used: number;
|
|
67
|
+
remaining: number;
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Standard API success response.
|
|
71
|
+
*/
|
|
72
|
+
export interface ApiSuccessResponse<T = unknown> {
|
|
73
|
+
success: true;
|
|
74
|
+
statusCode: number;
|
|
75
|
+
data?: T;
|
|
76
|
+
message?: string;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Standard API error response.
|
|
80
|
+
*/
|
|
81
|
+
export interface ApiErrorResponse {
|
|
82
|
+
success: false;
|
|
83
|
+
statusCode: number;
|
|
84
|
+
message: string;
|
|
85
|
+
code?: string;
|
|
86
|
+
details?: unknown;
|
|
87
|
+
quota?: QuotaInfo;
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Combined API response type.
|
|
91
|
+
*/
|
|
92
|
+
export type ApiResponse<T = unknown> = ApiSuccessResponse<T> | ApiErrorResponse;
|
|
93
|
+
/**
|
|
94
|
+
* JWT payload from access tokens.
|
|
95
|
+
*
|
|
96
|
+
* Users can belong to multiple organizations. The token is scoped to one
|
|
97
|
+
* active organization at a time. The `role` field is the user's per-org
|
|
98
|
+
* role in that organization (from the UserOrganization junction collection),
|
|
99
|
+
* and `isAdmin` is derived as `role === 'admin' || role === 'owner'`.
|
|
100
|
+
*
|
|
101
|
+
* Use `POST /auth/switch-org` to change the active organization, which
|
|
102
|
+
* re-issues tokens with the new org's role and context.
|
|
103
|
+
*/
|
|
104
|
+
export interface JwtPayload {
|
|
105
|
+
/** User ID (subject) */
|
|
106
|
+
sub: string;
|
|
107
|
+
/** Username */
|
|
108
|
+
username: string;
|
|
109
|
+
/** User email */
|
|
110
|
+
email: string;
|
|
111
|
+
/** Per-org role in the active organization ('owner' | 'admin' | 'member'). Not a global role. */
|
|
112
|
+
role: 'owner' | 'admin' | 'member';
|
|
113
|
+
/** Derived: true when role is 'admin' or 'owner' in the active organization */
|
|
114
|
+
isAdmin?: boolean;
|
|
115
|
+
/** Organization's quota tier ('developer' | 'pro' | 'unlimited') */
|
|
116
|
+
tier?: string;
|
|
117
|
+
/** Resolved feature flags for this user/org */
|
|
118
|
+
features?: string[];
|
|
119
|
+
/** Active organization ID (from UserOrganization membership) */
|
|
120
|
+
organizationId?: string;
|
|
121
|
+
/** Active organization name */
|
|
122
|
+
organizationName?: string;
|
|
123
|
+
/** Token type */
|
|
124
|
+
type: 'access' | 'refresh';
|
|
125
|
+
/** Issued at timestamp */
|
|
126
|
+
iat?: number;
|
|
127
|
+
/** Expiration timestamp */
|
|
128
|
+
exp?: number;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Extended Express Request with user property.
|
|
132
|
+
*/
|
|
133
|
+
declare global {
|
|
134
|
+
namespace Express {
|
|
135
|
+
interface Request {
|
|
136
|
+
user?: JwtPayload;
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Service configuration for internal HTTP client.
|
|
142
|
+
*/
|
|
143
|
+
export interface ServiceConfig {
|
|
144
|
+
/** Service hostname */
|
|
145
|
+
host: string;
|
|
146
|
+
/** Service port */
|
|
147
|
+
port: number;
|
|
148
|
+
/** Request timeout in milliseconds */
|
|
149
|
+
timeout?: number;
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Health check response.
|
|
153
|
+
*/
|
|
154
|
+
export interface HealthCheckResponse {
|
|
155
|
+
status: 'healthy' | 'unhealthy';
|
|
156
|
+
service: string;
|
|
157
|
+
timestamp: string;
|
|
158
|
+
uptime: number;
|
|
159
|
+
version?: string;
|
|
160
|
+
dependencies?: Record<string, 'connected' | 'disconnected' | 'unknown'>;
|
|
161
|
+
}
|