@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.
Files changed (122) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +51 -0
  3. package/lib/constants/ai-providers.d.ts +41 -0
  4. package/lib/constants/ai-providers.js +88 -0
  5. package/lib/constants/http-status.d.ts +24 -0
  6. package/lib/constants/http-status.js +29 -0
  7. package/lib/constants/index.d.ts +3 -0
  8. package/lib/constants/index.js +22 -0
  9. package/lib/constants/time.d.ts +10 -0
  10. package/lib/constants/time.js +16 -0
  11. package/lib/errors/app-errors.d.ts +30 -0
  12. package/lib/errors/app-errors.js +62 -0
  13. package/lib/errors/index.d.ts +1 -0
  14. package/lib/errors/index.js +20 -0
  15. package/lib/helpers/access-helpers.d.ts +40 -0
  16. package/lib/helpers/access-helpers.js +56 -0
  17. package/lib/helpers/crud-helpers.d.ts +16 -0
  18. package/lib/helpers/crud-helpers.js +34 -0
  19. package/lib/helpers/index.d.ts +4 -0
  20. package/lib/helpers/index.js +23 -0
  21. package/lib/helpers/mask-helpers.d.ts +33 -0
  22. package/lib/helpers/mask-helpers.js +54 -0
  23. package/lib/helpers/sse-helpers.d.ts +13 -0
  24. package/lib/helpers/sse-helpers.js +40 -0
  25. package/lib/index.d.ts +57 -0
  26. package/lib/index.js +86 -0
  27. package/lib/middleware/auth.d.ts +50 -0
  28. package/lib/middleware/auth.js +171 -0
  29. package/lib/middleware/index.d.ts +1 -0
  30. package/lib/middleware/index.js +20 -0
  31. package/lib/openapi/extend-zod.d.ts +1 -0
  32. package/lib/openapi/extend-zod.js +8 -0
  33. package/lib/openapi/index.d.ts +2 -0
  34. package/lib/openapi/index.js +10 -0
  35. package/lib/openapi/registry.d.ts +17 -0
  36. package/lib/openapi/registry.js +42 -0
  37. package/lib/openapi/routes/billing-routes.d.ts +1 -0
  38. package/lib/openapi/routes/billing-routes.js +69 -0
  39. package/lib/openapi/routes/index.d.ts +5 -0
  40. package/lib/openapi/routes/index.js +22 -0
  41. package/lib/openapi/routes/message-routes.d.ts +1 -0
  42. package/lib/openapi/routes/message-routes.js +108 -0
  43. package/lib/openapi/routes/pipeline-routes.d.ts +1 -0
  44. package/lib/openapi/routes/pipeline-routes.js +90 -0
  45. package/lib/openapi/routes/plugin-routes.d.ts +1 -0
  46. package/lib/openapi/routes/plugin-routes.js +99 -0
  47. package/lib/openapi/routes/quota-routes.d.ts +1 -0
  48. package/lib/openapi/routes/quota-routes.js +65 -0
  49. package/lib/openapi/schema-registry.d.ts +25 -0
  50. package/lib/openapi/schema-registry.js +95 -0
  51. package/lib/routes/health.d.ts +47 -0
  52. package/lib/routes/health.js +81 -0
  53. package/lib/routes/index.d.ts +1 -0
  54. package/lib/routes/index.js +20 -0
  55. package/lib/services/admin-audit.d.ts +13 -0
  56. package/lib/services/admin-audit.js +31 -0
  57. package/lib/services/cache-service.d.ts +108 -0
  58. package/lib/services/cache-service.js +212 -0
  59. package/lib/services/compliance-client.d.ts +46 -0
  60. package/lib/services/compliance-client.js +102 -0
  61. package/lib/services/compliance-event-subscriber.d.ts +11 -0
  62. package/lib/services/compliance-event-subscriber.js +60 -0
  63. package/lib/services/compliance-queue.d.ts +11 -0
  64. package/lib/services/compliance-queue.js +38 -0
  65. package/lib/services/entity-events.d.ts +44 -0
  66. package/lib/services/entity-events.js +63 -0
  67. package/lib/services/http-client.d.ts +108 -0
  68. package/lib/services/http-client.js +285 -0
  69. package/lib/services/index.d.ts +10 -0
  70. package/lib/services/index.js +40 -0
  71. package/lib/services/quota.d.ts +59 -0
  72. package/lib/services/quota.js +137 -0
  73. package/lib/services/retry-strategy.d.ts +74 -0
  74. package/lib/services/retry-strategy.js +127 -0
  75. package/lib/types/billing.d.ts +47 -0
  76. package/lib/types/billing.js +5 -0
  77. package/lib/types/common.d.ts +161 -0
  78. package/lib/types/common.js +53 -0
  79. package/lib/types/error-codes.d.ts +38 -0
  80. package/lib/types/error-codes.js +77 -0
  81. package/lib/types/feature-flags.d.ts +38 -0
  82. package/lib/types/feature-flags.js +107 -0
  83. package/lib/types/http.d.ts +37 -0
  84. package/lib/types/http.js +5 -0
  85. package/lib/types/index.d.ts +7 -0
  86. package/lib/types/index.js +26 -0
  87. package/lib/types/pipeline.d.ts +70 -0
  88. package/lib/types/pipeline.js +44 -0
  89. package/lib/types/quota-tiers.d.ts +23 -0
  90. package/lib/types/quota-tiers.js +26 -0
  91. package/lib/utils/alias-resolver.d.ts +16 -0
  92. package/lib/utils/alias-resolver.js +49 -0
  93. package/lib/utils/headers.d.ts +18 -0
  94. package/lib/utils/headers.js +24 -0
  95. package/lib/utils/identity.d.ts +61 -0
  96. package/lib/utils/identity.js +75 -0
  97. package/lib/utils/index.d.ts +7 -0
  98. package/lib/utils/index.js +26 -0
  99. package/lib/utils/logger.d.ts +28 -0
  100. package/lib/utils/logger.js +77 -0
  101. package/lib/utils/object.d.ts +13 -0
  102. package/lib/utils/object.js +21 -0
  103. package/lib/utils/params.d.ts +89 -0
  104. package/lib/utils/params.js +148 -0
  105. package/lib/utils/response.d.ts +142 -0
  106. package/lib/utils/response.js +237 -0
  107. package/lib/validation/ai-schemas.d.ts +61 -0
  108. package/lib/validation/ai-schemas.js +81 -0
  109. package/lib/validation/common-schemas.d.ts +72 -0
  110. package/lib/validation/common-schemas.js +58 -0
  111. package/lib/validation/index.d.ts +6 -0
  112. package/lib/validation/index.js +25 -0
  113. package/lib/validation/message-schemas.d.ts +79 -0
  114. package/lib/validation/message-schemas.js +42 -0
  115. package/lib/validation/middleware.d.ts +60 -0
  116. package/lib/validation/middleware.js +77 -0
  117. package/lib/validation/pipeline-schemas.d.ts +135 -0
  118. package/lib/validation/pipeline-schemas.js +85 -0
  119. package/lib/validation/plugin-schemas.d.ts +127 -0
  120. package/lib/validation/plugin-schemas.js +84 -0
  121. package/openapi.yaml +292 -0
  122. 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
+ }