@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,38 @@
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.registerComplianceQueueBackend = registerComplianceQueueBackend;
6
+ exports.enqueueComplianceEvent = enqueueComplianceEvent;
7
+ const logger_1 = require("../utils/logger");
8
+ const logger = (0, logger_1.createLogger)('compliance-queue');
9
+ /**
10
+ * Pluggable compliance event queue.
11
+ * Services register a backend (BullMQ, SQS, or in-memory) at startup.
12
+ * If no backend is registered, events are logged and discarded.
13
+ */
14
+ let enqueueFn = null;
15
+ function registerComplianceQueueBackend(fn) {
16
+ enqueueFn = fn;
17
+ logger.info('Compliance queue backend registered');
18
+ }
19
+ async function enqueueComplianceEvent(event) {
20
+ if (!enqueueFn) {
21
+ logger.debug('Compliance event discarded (no queue backend registered)', {
22
+ eventType: event.eventType, target: event.target, entityId: event.entityId,
23
+ });
24
+ return;
25
+ }
26
+ try {
27
+ await enqueueFn(event);
28
+ logger.debug('Compliance event enqueued', { eventType: event.eventType, target: event.target, entityId: event.entityId });
29
+ }
30
+ catch (err) {
31
+ logger.warn('Failed to enqueue compliance event', {
32
+ eventType: event.eventType,
33
+ entityId: event.entityId,
34
+ error: err instanceof Error ? err.message : String(err),
35
+ });
36
+ }
37
+ }
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tcGxpYW5jZS1xdWV1ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zZXJ2aWNlcy9jb21wbGlhbmNlLXF1ZXVlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDOztBQXVCdEMsd0VBR0M7QUFFRCx3REFpQkM7QUEzQ0QsNENBQStDO0FBRS9DLE1BQU0sTUFBTSxHQUFHLElBQUEscUJBQVksRUFBQyxrQkFBa0IsQ0FBQyxDQUFDO0FBWWhEOzs7O0dBSUc7QUFDSCxJQUFJLFNBQVMsR0FBdUQsSUFBSSxDQUFDO0FBRXpFLFNBQWdCLDhCQUE4QixDQUFDLEVBQTZDO0lBQzFGLFNBQVMsR0FBRyxFQUFFLENBQUM7SUFDZixNQUFNLENBQUMsSUFBSSxDQUFDLHFDQUFxQyxDQUFDLENBQUM7QUFDckQsQ0FBQztBQUVNLEtBQUssVUFBVSxzQkFBc0IsQ0FBQyxLQUFzQjtJQUNqRSxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDZixNQUFNLENBQUMsS0FBSyxDQUFDLDBEQUEwRCxFQUFFO1lBQ3ZFLFNBQVMsRUFBRSxLQUFLLENBQUMsU0FBUyxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsTUFBTSxFQUFFLFFBQVEsRUFBRSxLQUFLLENBQUMsUUFBUTtTQUMzRSxDQUFDLENBQUM7UUFDSCxPQUFPO0lBQ1QsQ0FBQztJQUNELElBQUksQ0FBQztRQUNILE1BQU0sU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3ZCLE1BQU0sQ0FBQyxLQUFLLENBQUMsMkJBQTJCLEVBQUUsRUFBRSxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVMsRUFBRSxNQUFNLEVBQUUsS0FBSyxDQUFDLE1BQU0sRUFBRSxRQUFRLEVBQUUsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7SUFDNUgsQ0FBQztJQUFDLE9BQU8sR0FBRyxFQUFFLENBQUM7UUFDYixNQUFNLENBQUMsSUFBSSxDQUFDLG9DQUFvQyxFQUFFO1lBQ2hELFNBQVMsRUFBRSxLQUFLLENBQUMsU0FBUztZQUMxQixRQUFRLEVBQUUsS0FBSyxDQUFDLFFBQVE7WUFDeEIsS0FBSyxFQUFFLEdBQUcsWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUM7U0FDeEQsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuaW1wb3J0IHsgY3JlYXRlTG9nZ2VyIH0gZnJvbSAnLi4vdXRpbHMvbG9nZ2VyJztcblxuY29uc3QgbG9nZ2VyID0gY3JlYXRlTG9nZ2VyKCdjb21wbGlhbmNlLXF1ZXVlJyk7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29tcGxpYW5jZUV2ZW50IHtcbiAgZXZlbnRUeXBlOiAndmFsaWRhdGUnIHwgJ3NjYW4nIHwgJ25vdGlmeSc7XG4gIHRhcmdldDogJ3BsdWdpbicgfCAncGlwZWxpbmUnO1xuICBlbnRpdHlJZDogc3RyaW5nO1xuICBvcmdJZDogc3RyaW5nO1xuICB1c2VySWQ6IHN0cmluZztcbiAgYXR0cmlidXRlczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIHRpbWVzdGFtcDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFBsdWdnYWJsZSBjb21wbGlhbmNlIGV2ZW50IHF1ZXVlLlxuICogU2VydmljZXMgcmVnaXN0ZXIgYSBiYWNrZW5kIChCdWxsTVEsIFNRUywgb3IgaW4tbWVtb3J5KSBhdCBzdGFydHVwLlxuICogSWYgbm8gYmFja2VuZCBpcyByZWdpc3RlcmVkLCBldmVudHMgYXJlIGxvZ2dlZCBhbmQgZGlzY2FyZGVkLlxuICovXG5sZXQgZW5xdWV1ZUZuOiAoKGV2ZW50OiBDb21wbGlhbmNlRXZlbnQpID0+IFByb21pc2U8dm9pZD4pIHwgbnVsbCA9IG51bGw7XG5cbmV4cG9ydCBmdW5jdGlvbiByZWdpc3RlckNvbXBsaWFuY2VRdWV1ZUJhY2tlbmQoZm46IChldmVudDogQ29tcGxpYW5jZUV2ZW50KSA9PiBQcm9taXNlPHZvaWQ+KTogdm9pZCB7XG4gIGVucXVldWVGbiA9IGZuO1xuICBsb2dnZXIuaW5mbygnQ29tcGxpYW5jZSBxdWV1ZSBiYWNrZW5kIHJlZ2lzdGVyZWQnKTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGVucXVldWVDb21wbGlhbmNlRXZlbnQoZXZlbnQ6IENvbXBsaWFuY2VFdmVudCk6IFByb21pc2U8dm9pZD4ge1xuICBpZiAoIWVucXVldWVGbikge1xuICAgIGxvZ2dlci5kZWJ1ZygnQ29tcGxpYW5jZSBldmVudCBkaXNjYXJkZWQgKG5vIHF1ZXVlIGJhY2tlbmQgcmVnaXN0ZXJlZCknLCB7XG4gICAgICBldmVudFR5cGU6IGV2ZW50LmV2ZW50VHlwZSwgdGFyZ2V0OiBldmVudC50YXJnZXQsIGVudGl0eUlkOiBldmVudC5lbnRpdHlJZCxcbiAgICB9KTtcbiAgICByZXR1cm47XG4gIH1cbiAgdHJ5IHtcbiAgICBhd2FpdCBlbnF1ZXVlRm4oZXZlbnQpO1xuICAgIGxvZ2dlci5kZWJ1ZygnQ29tcGxpYW5jZSBldmVudCBlbnF1ZXVlZCcsIHsgZXZlbnRUeXBlOiBldmVudC5ldmVudFR5cGUsIHRhcmdldDogZXZlbnQudGFyZ2V0LCBlbnRpdHlJZDogZXZlbnQuZW50aXR5SWQgfSk7XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIGxvZ2dlci53YXJuKCdGYWlsZWQgdG8gZW5xdWV1ZSBjb21wbGlhbmNlIGV2ZW50Jywge1xuICAgICAgZXZlbnRUeXBlOiBldmVudC5ldmVudFR5cGUsXG4gICAgICBlbnRpdHlJZDogZXZlbnQuZW50aXR5SWQsXG4gICAgICBlcnJvcjogZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpLFxuICAgIH0pO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,44 @@
1
+ export type EntityEventType = 'created' | 'updated' | 'deleted';
2
+ export type EntityTarget = 'plugin' | 'pipeline' | 'message' | 'organization' | 'user';
3
+ export interface EntityEvent {
4
+ /** The type of mutation that occurred */
5
+ eventType: EntityEventType;
6
+ /** The entity type that was mutated */
7
+ target: EntityTarget;
8
+ /** The entity's unique ID */
9
+ entityId: string;
10
+ /** The organization that owns this entity */
11
+ orgId: string;
12
+ /** The user who performed the mutation */
13
+ userId: string;
14
+ /** When the event occurred */
15
+ timestamp: Date;
16
+ /** Snapshot of entity attributes (for compliance evaluation) */
17
+ attributes: Record<string, unknown>;
18
+ }
19
+ export interface EntityEventSubscriber {
20
+ onEntityEvent(event: EntityEvent): Promise<void>;
21
+ }
22
+ declare class EntityEventEmitter {
23
+ private subscribers;
24
+ /**
25
+ * Register a subscriber to receive entity events.
26
+ */
27
+ subscribe(subscriber: EntityEventSubscriber): void;
28
+ /**
29
+ * Remove a subscriber.
30
+ */
31
+ unsubscribe(subscriber: EntityEventSubscriber): void;
32
+ /**
33
+ * Emit an entity event to all subscribers.
34
+ * Fire-and-forget: errors are caught and logged, never thrown.
35
+ */
36
+ emit(event: EntityEvent): void;
37
+ /**
38
+ * Get the current subscriber count (for testing/diagnostics).
39
+ */
40
+ get subscriberCount(): number;
41
+ }
42
+ /** Singleton entity event emitter — shared across the process */
43
+ export declare const entityEvents: EntityEventEmitter;
44
+ export {};
@@ -0,0 +1,63 @@
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.entityEvents = void 0;
6
+ /**
7
+ * Lightweight in-process event emitter for entity lifecycle events.
8
+ *
9
+ * Services emit events after mutations (create/update/delete) via CrudService hooks.
10
+ * Subscribers react asynchronously to entity changes.
11
+ *
12
+ * Active subscribers:
13
+ * - **Compliance event subscriber** (`registerComplianceEventSubscriber()`) — forwards
14
+ * entity events to the compliance service for post-mutation rule evaluation.
15
+ * Registered at startup in plugin and pipeline service index.ts.
16
+ *
17
+ * Design:
18
+ * - Fire-and-forget: emit() never throws, subscriber errors are logged
19
+ * - In-process only: no network, no Redis, no infrastructure
20
+ * - Subscribers run async: never block the original request
21
+ */
22
+ const logger_1 = require("../utils/logger");
23
+ const logger = (0, logger_1.createLogger)('entity-events');
24
+ class EntityEventEmitter {
25
+ subscribers = [];
26
+ /**
27
+ * Register a subscriber to receive entity events.
28
+ */
29
+ subscribe(subscriber) {
30
+ this.subscribers.push(subscriber);
31
+ }
32
+ /**
33
+ * Remove a subscriber.
34
+ */
35
+ unsubscribe(subscriber) {
36
+ this.subscribers = this.subscribers.filter((s) => s !== subscriber);
37
+ }
38
+ /**
39
+ * Emit an entity event to all subscribers.
40
+ * Fire-and-forget: errors are caught and logged, never thrown.
41
+ */
42
+ emit(event) {
43
+ for (const subscriber of this.subscribers) {
44
+ subscriber.onEntityEvent(event).catch((err) => {
45
+ logger.debug('Entity event subscriber failed', {
46
+ target: event.target,
47
+ eventType: event.eventType,
48
+ entityId: event.entityId,
49
+ error: err instanceof Error ? err.message : String(err),
50
+ });
51
+ });
52
+ }
53
+ }
54
+ /**
55
+ * Get the current subscriber count (for testing/diagnostics).
56
+ */
57
+ get subscriberCount() {
58
+ return this.subscribers.length;
59
+ }
60
+ }
61
+ /** Singleton entity event emitter — shared across the process */
62
+ exports.entityEvents = new EntityEventEmitter();
63
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW50aXR5LWV2ZW50cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zZXJ2aWNlcy9lbnRpdHktZXZlbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDOzs7QUFFdEM7Ozs7Ozs7Ozs7Ozs7OztHQWVHO0FBRUgsNENBQStDO0FBRS9DLE1BQU0sTUFBTSxHQUFHLElBQUEscUJBQVksRUFBQyxlQUFlLENBQUMsQ0FBQztBQTBCN0MsTUFBTSxrQkFBa0I7SUFDZCxXQUFXLEdBQTRCLEVBQUUsQ0FBQztJQUVsRDs7T0FFRztJQUNILFNBQVMsQ0FBQyxVQUFpQztRQUN6QyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUNwQyxDQUFDO0lBRUQ7O09BRUc7SUFDSCxXQUFXLENBQUMsVUFBaUM7UUFDM0MsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxLQUFLLFVBQVUsQ0FBQyxDQUFDO0lBQ3RFLENBQUM7SUFFRDs7O09BR0c7SUFDSCxJQUFJLENBQUMsS0FBa0I7UUFDckIsS0FBSyxNQUFNLFVBQVUsSUFBSSxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDMUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxHQUFHLEVBQUUsRUFBRTtnQkFDNUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxnQ0FBZ0MsRUFBRTtvQkFDN0MsTUFBTSxFQUFFLEtBQUssQ0FBQyxNQUFNO29CQUNwQixTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7b0JBQzFCLFFBQVEsRUFBRSxLQUFLLENBQUMsUUFBUTtvQkFDeEIsS0FBSyxFQUFFLEdBQUcsWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUM7aUJBQ3hELENBQUMsQ0FBQztZQUNMLENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNILElBQUksZUFBZTtRQUNqQixPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDO0lBQ2pDLENBQUM7Q0FDRjtBQUVELGlFQUFpRTtBQUNwRCxRQUFBLFlBQVksR0FBRyxJQUFJLGtCQUFrQixFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuLyoqXG4gKiBMaWdodHdlaWdodCBpbi1wcm9jZXNzIGV2ZW50IGVtaXR0ZXIgZm9yIGVudGl0eSBsaWZlY3ljbGUgZXZlbnRzLlxuICpcbiAqIFNlcnZpY2VzIGVtaXQgZXZlbnRzIGFmdGVyIG11dGF0aW9ucyAoY3JlYXRlL3VwZGF0ZS9kZWxldGUpIHZpYSBDcnVkU2VydmljZSBob29rcy5cbiAqIFN1YnNjcmliZXJzIHJlYWN0IGFzeW5jaHJvbm91c2x5IHRvIGVudGl0eSBjaGFuZ2VzLlxuICpcbiAqIEFjdGl2ZSBzdWJzY3JpYmVyczpcbiAqIC0gKipDb21wbGlhbmNlIGV2ZW50IHN1YnNjcmliZXIqKiAoYHJlZ2lzdGVyQ29tcGxpYW5jZUV2ZW50U3Vic2NyaWJlcigpYCkg4oCUIGZvcndhcmRzXG4gKiAgIGVudGl0eSBldmVudHMgdG8gdGhlIGNvbXBsaWFuY2Ugc2VydmljZSBmb3IgcG9zdC1tdXRhdGlvbiBydWxlIGV2YWx1YXRpb24uXG4gKiAgIFJlZ2lzdGVyZWQgYXQgc3RhcnR1cCBpbiBwbHVnaW4gYW5kIHBpcGVsaW5lIHNlcnZpY2UgaW5kZXgudHMuXG4gKlxuICogRGVzaWduOlxuICogLSBGaXJlLWFuZC1mb3JnZXQ6IGVtaXQoKSBuZXZlciB0aHJvd3MsIHN1YnNjcmliZXIgZXJyb3JzIGFyZSBsb2dnZWRcbiAqIC0gSW4tcHJvY2VzcyBvbmx5OiBubyBuZXR3b3JrLCBubyBSZWRpcywgbm8gaW5mcmFzdHJ1Y3R1cmVcbiAqIC0gU3Vic2NyaWJlcnMgcnVuIGFzeW5jOiBuZXZlciBibG9jayB0aGUgb3JpZ2luYWwgcmVxdWVzdFxuICovXG5cbmltcG9ydCB7IGNyZWF0ZUxvZ2dlciB9IGZyb20gJy4uL3V0aWxzL2xvZ2dlcic7XG5cbmNvbnN0IGxvZ2dlciA9IGNyZWF0ZUxvZ2dlcignZW50aXR5LWV2ZW50cycpO1xuXG5leHBvcnQgdHlwZSBFbnRpdHlFdmVudFR5cGUgPSAnY3JlYXRlZCcgfCAndXBkYXRlZCcgfCAnZGVsZXRlZCc7XG5leHBvcnQgdHlwZSBFbnRpdHlUYXJnZXQgPSAncGx1Z2luJyB8ICdwaXBlbGluZScgfCAnbWVzc2FnZScgfCAnb3JnYW5pemF0aW9uJyB8ICd1c2VyJztcblxuZXhwb3J0IGludGVyZmFjZSBFbnRpdHlFdmVudCB7XG4gIC8qKiBUaGUgdHlwZSBvZiBtdXRhdGlvbiB0aGF0IG9jY3VycmVkICovXG4gIGV2ZW50VHlwZTogRW50aXR5RXZlbnRUeXBlO1xuICAvKiogVGhlIGVudGl0eSB0eXBlIHRoYXQgd2FzIG11dGF0ZWQgKi9cbiAgdGFyZ2V0OiBFbnRpdHlUYXJnZXQ7XG4gIC8qKiBUaGUgZW50aXR5J3MgdW5pcXVlIElEICovXG4gIGVudGl0eUlkOiBzdHJpbmc7XG4gIC8qKiBUaGUgb3JnYW5pemF0aW9uIHRoYXQgb3ducyB0aGlzIGVudGl0eSAqL1xuICBvcmdJZDogc3RyaW5nO1xuICAvKiogVGhlIHVzZXIgd2hvIHBlcmZvcm1lZCB0aGUgbXV0YXRpb24gKi9cbiAgdXNlcklkOiBzdHJpbmc7XG4gIC8qKiBXaGVuIHRoZSBldmVudCBvY2N1cnJlZCAqL1xuICB0aW1lc3RhbXA6IERhdGU7XG4gIC8qKiBTbmFwc2hvdCBvZiBlbnRpdHkgYXR0cmlidXRlcyAoZm9yIGNvbXBsaWFuY2UgZXZhbHVhdGlvbikgKi9cbiAgYXR0cmlidXRlczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRW50aXR5RXZlbnRTdWJzY3JpYmVyIHtcbiAgb25FbnRpdHlFdmVudChldmVudDogRW50aXR5RXZlbnQpOiBQcm9taXNlPHZvaWQ+O1xufVxuXG5jbGFzcyBFbnRpdHlFdmVudEVtaXR0ZXIge1xuICBwcml2YXRlIHN1YnNjcmliZXJzOiBFbnRpdHlFdmVudFN1YnNjcmliZXJbXSA9IFtdO1xuXG4gIC8qKlxuICAgKiBSZWdpc3RlciBhIHN1YnNjcmliZXIgdG8gcmVjZWl2ZSBlbnRpdHkgZXZlbnRzLlxuICAgKi9cbiAgc3Vic2NyaWJlKHN1YnNjcmliZXI6IEVudGl0eUV2ZW50U3Vic2NyaWJlcik6IHZvaWQge1xuICAgIHRoaXMuc3Vic2NyaWJlcnMucHVzaChzdWJzY3JpYmVyKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW1vdmUgYSBzdWJzY3JpYmVyLlxuICAgKi9cbiAgdW5zdWJzY3JpYmUoc3Vic2NyaWJlcjogRW50aXR5RXZlbnRTdWJzY3JpYmVyKTogdm9pZCB7XG4gICAgdGhpcy5zdWJzY3JpYmVycyA9IHRoaXMuc3Vic2NyaWJlcnMuZmlsdGVyKChzKSA9PiBzICE9PSBzdWJzY3JpYmVyKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFbWl0IGFuIGVudGl0eSBldmVudCB0byBhbGwgc3Vic2NyaWJlcnMuXG4gICAqIEZpcmUtYW5kLWZvcmdldDogZXJyb3JzIGFyZSBjYXVnaHQgYW5kIGxvZ2dlZCwgbmV2ZXIgdGhyb3duLlxuICAgKi9cbiAgZW1pdChldmVudDogRW50aXR5RXZlbnQpOiB2b2lkIHtcbiAgICBmb3IgKGNvbnN0IHN1YnNjcmliZXIgb2YgdGhpcy5zdWJzY3JpYmVycykge1xuICAgICAgc3Vic2NyaWJlci5vbkVudGl0eUV2ZW50KGV2ZW50KS5jYXRjaCgoZXJyKSA9PiB7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZygnRW50aXR5IGV2ZW50IHN1YnNjcmliZXIgZmFpbGVkJywge1xuICAgICAgICAgIHRhcmdldDogZXZlbnQudGFyZ2V0LFxuICAgICAgICAgIGV2ZW50VHlwZTogZXZlbnQuZXZlbnRUeXBlLFxuICAgICAgICAgIGVudGl0eUlkOiBldmVudC5lbnRpdHlJZCxcbiAgICAgICAgICBlcnJvcjogZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpLFxuICAgICAgICB9KTtcbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgdGhlIGN1cnJlbnQgc3Vic2NyaWJlciBjb3VudCAoZm9yIHRlc3RpbmcvZGlhZ25vc3RpY3MpLlxuICAgKi9cbiAgZ2V0IHN1YnNjcmliZXJDb3VudCgpOiBudW1iZXIge1xuICAgIHJldHVybiB0aGlzLnN1YnNjcmliZXJzLmxlbmd0aDtcbiAgfVxufVxuXG4vKiogU2luZ2xldG9uIGVudGl0eSBldmVudCBlbWl0dGVyIOKAlCBzaGFyZWQgYWNyb3NzIHRoZSBwcm9jZXNzICovXG5leHBvcnQgY29uc3QgZW50aXR5RXZlbnRzID0gbmV3IEVudGl0eUV2ZW50RW1pdHRlcigpO1xuIl19
@@ -0,0 +1,108 @@
1
+ import * as http from 'http';
2
+ import { ServiceConfig } from '../types/common';
3
+ export { parseRetryAfter, addJitter } from './retry-strategy';
4
+ /**
5
+ * HTTP request options.
6
+ */
7
+ export interface RequestOptions {
8
+ /** Request headers */
9
+ headers?: Record<string, string>;
10
+ /** Request timeout in milliseconds */
11
+ timeout?: number;
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
+ /** Optional request ID for distributed tracing (added as X-Request-Id header) */
19
+ requestId?: string;
20
+ }
21
+ /**
22
+ * HTTP client response wrapper.
23
+ */
24
+ export interface HttpClientResponse<T = unknown> {
25
+ /** HTTP status code */
26
+ statusCode: number;
27
+ /** Response body (parsed JSON) */
28
+ body: T;
29
+ /** Response headers */
30
+ headers: http.IncomingHttpHeaders;
31
+ }
32
+ /**
33
+ * Internal HTTP client for service-to-service communication.
34
+ *
35
+ * @example
36
+ * ```typescript
37
+ * const client = new InternalHttpClient({
38
+ * host: 'quota',
39
+ * port: 3000,
40
+ * timeout: 5000,
41
+ * });
42
+ *
43
+ * const response = await client.get('/org123/apiCalls');
44
+ * const result = await client.post('/org123/increment', { quotaType: 'apiCalls' });
45
+ * ```
46
+ */
47
+ export declare class InternalHttpClient {
48
+ private config;
49
+ private agent;
50
+ /**
51
+ * Create a new HTTP client instance.
52
+ *
53
+ * @param config - Service configuration
54
+ */
55
+ constructor(config: ServiceConfig);
56
+ /**
57
+ * Make a GET request.
58
+ */
59
+ get<T = unknown>(path: string, options?: RequestOptions): Promise<HttpClientResponse<T>>;
60
+ /**
61
+ * Make a POST request.
62
+ */
63
+ post<T = unknown>(path: string, body?: unknown, options?: RequestOptions): Promise<HttpClientResponse<T>>;
64
+ /**
65
+ * Make a PUT request.
66
+ */
67
+ put<T = unknown>(path: string, body?: unknown, options?: RequestOptions): Promise<HttpClientResponse<T>>;
68
+ /**
69
+ * Make a DELETE request.
70
+ */
71
+ delete<T = unknown>(path: string, options?: RequestOptions): Promise<HttpClientResponse<T>>;
72
+ /**
73
+ * Request with retry logic for transient failures.
74
+ *
75
+ * - 429 (rate limit): respects `Retry-After` header, uses longer base delay (4x),
76
+ * retries up to `maxRateLimitRetries` times (default 4).
77
+ * - 502/503/504 (server errors): standard exponential backoff, up to `maxRetries` (default 2).
78
+ * - Connection errors / timeouts: retries up to `maxRetries`.
79
+ * - All delays include ±25% jitter to prevent thundering herd.
80
+ */
81
+ private requestWithRetry;
82
+ private sleep;
83
+ /**
84
+ * Internal request method.
85
+ */
86
+ private request;
87
+ }
88
+ /**
89
+ * Create an HTTP client with error handling that returns null on failure.
90
+ * Useful for fail-open scenarios.
91
+ *
92
+ * @param config - Service configuration
93
+ * @returns Client wrapper with safe methods
94
+ */
95
+ export declare function createSafeClient(config: ServiceConfig): {
96
+ /**
97
+ * Safe GET request - returns null on error.
98
+ */
99
+ get<T>(path: string, options?: RequestOptions): Promise<HttpClientResponse<T> | null>;
100
+ /**
101
+ * Safe POST request - returns null on error.
102
+ */
103
+ post<T>(path: string, body?: unknown, options?: RequestOptions): Promise<HttpClientResponse<T> | null>;
104
+ /**
105
+ * Safe PUT request - returns null on error.
106
+ */
107
+ put<T>(path: string, body?: unknown, options?: RequestOptions): Promise<HttpClientResponse<T> | null>;
108
+ };
@@ -0,0 +1,285 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
5
+ if (k2 === undefined) k2 = k;
6
+ var desc = Object.getOwnPropertyDescriptor(m, k);
7
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
8
+ desc = { enumerable: true, get: function() { return m[k]; } };
9
+ }
10
+ Object.defineProperty(o, k2, desc);
11
+ }) : (function(o, m, k, k2) {
12
+ if (k2 === undefined) k2 = k;
13
+ o[k2] = m[k];
14
+ }));
15
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
16
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
17
+ }) : function(o, v) {
18
+ o["default"] = v;
19
+ });
20
+ var __importStar = (this && this.__importStar) || (function () {
21
+ var ownKeys = function(o) {
22
+ ownKeys = Object.getOwnPropertyNames || function (o) {
23
+ var ar = [];
24
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
25
+ return ar;
26
+ };
27
+ return ownKeys(o);
28
+ };
29
+ return function (mod) {
30
+ if (mod && mod.__esModule) return mod;
31
+ var result = {};
32
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
33
+ __setModuleDefault(result, mod);
34
+ return result;
35
+ };
36
+ })();
37
+ Object.defineProperty(exports, "__esModule", { value: true });
38
+ exports.InternalHttpClient = exports.addJitter = exports.parseRetryAfter = void 0;
39
+ exports.createSafeClient = createSafeClient;
40
+ const http = __importStar(require("http"));
41
+ const retry_strategy_1 = require("./retry-strategy");
42
+ const http_status_1 = require("../constants/http-status");
43
+ const logger_1 = require("../utils/logger");
44
+ // Re-export retry utilities for backward compatibility
45
+ var retry_strategy_2 = require("./retry-strategy");
46
+ Object.defineProperty(exports, "parseRetryAfter", { enumerable: true, get: function () { return retry_strategy_2.parseRetryAfter; } });
47
+ Object.defineProperty(exports, "addJitter", { enumerable: true, get: function () { return retry_strategy_2.addJitter; } });
48
+ const logger = (0, logger_1.createLogger)('http-client');
49
+ /**
50
+ * Default request timeout in milliseconds (env: `HTTP_CLIENT_TIMEOUT`).
51
+ */
52
+ const DEFAULT_TIMEOUT = parseInt(process.env.HTTP_CLIENT_TIMEOUT || '5000', 10);
53
+ /**
54
+ * Internal HTTP client for service-to-service communication.
55
+ *
56
+ * @example
57
+ * ```typescript
58
+ * const client = new InternalHttpClient({
59
+ * host: 'quota',
60
+ * port: 3000,
61
+ * timeout: 5000,
62
+ * });
63
+ *
64
+ * const response = await client.get('/org123/apiCalls');
65
+ * const result = await client.post('/org123/increment', { quotaType: 'apiCalls' });
66
+ * ```
67
+ */
68
+ class InternalHttpClient {
69
+ config;
70
+ agent;
71
+ /**
72
+ * Create a new HTTP client instance.
73
+ *
74
+ * @param config - Service configuration
75
+ */
76
+ constructor(config) {
77
+ this.config = {
78
+ host: config.host,
79
+ port: config.port,
80
+ timeout: config.timeout ?? DEFAULT_TIMEOUT,
81
+ };
82
+ this.agent = new http.Agent({ keepAlive: true });
83
+ }
84
+ /**
85
+ * Make a GET request.
86
+ */
87
+ async get(path, options) {
88
+ return this.requestWithRetry('GET', path, undefined, options);
89
+ }
90
+ /**
91
+ * Make a POST request.
92
+ */
93
+ async post(path, body, options) {
94
+ return this.requestWithRetry('POST', path, body, options);
95
+ }
96
+ /**
97
+ * Make a PUT request.
98
+ */
99
+ async put(path, body, options) {
100
+ return this.requestWithRetry('PUT', path, body, options);
101
+ }
102
+ /**
103
+ * Make a DELETE request.
104
+ */
105
+ async delete(path, options) {
106
+ return this.requestWithRetry('DELETE', path, undefined, options);
107
+ }
108
+ /**
109
+ * Request with retry logic for transient failures.
110
+ *
111
+ * - 429 (rate limit): respects `Retry-After` header, uses longer base delay (4x),
112
+ * retries up to `maxRateLimitRetries` times (default 4).
113
+ * - 502/503/504 (server errors): standard exponential backoff, up to `maxRetries` (default 2).
114
+ * - Connection errors / timeouts: retries up to `maxRetries`.
115
+ * - All delays include ±25% jitter to prevent thundering herd.
116
+ */
117
+ async requestWithRetry(method, path, body, options) {
118
+ const retryConfig = {
119
+ maxRetries: options?.maxRetries ?? retry_strategy_1.DEFAULT_MAX_RETRIES,
120
+ maxRateLimitRetries: options?.maxRateLimitRetries ?? retry_strategy_1.DEFAULT_MAX_RATE_LIMIT_RETRIES,
121
+ retryDelayMs: options?.retryDelayMs ?? retry_strategy_1.DEFAULT_RETRY_DELAY_MS,
122
+ };
123
+ const totalMaxAttempts = Math.max(retryConfig.maxRetries, retryConfig.maxRateLimitRetries);
124
+ let lastError;
125
+ for (let attempt = 0; attempt <= totalMaxAttempts; attempt++) {
126
+ try {
127
+ const response = await this.request(method, path, body, options);
128
+ const decision = (0, retry_strategy_1.getRetryDecision)(response.statusCode, response.headers, attempt, retryConfig);
129
+ if (decision.shouldRetry) {
130
+ logger.debug(decision.reason + ', retrying', { method, path, attempt: attempt + 1, delayMs: decision.delayMs });
131
+ await this.sleep(decision.delayMs);
132
+ continue;
133
+ }
134
+ return response;
135
+ }
136
+ catch (error) {
137
+ lastError = error instanceof Error ? error : new Error(String(error));
138
+ const decision = (0, retry_strategy_1.getErrorRetryDecision)(attempt, retryConfig);
139
+ if (decision.shouldRetry) {
140
+ logger.debug('Retrying after error', { method, path, error: lastError.message, attempt: attempt + 1 });
141
+ await this.sleep(decision.delayMs);
142
+ continue;
143
+ }
144
+ }
145
+ }
146
+ throw lastError;
147
+ }
148
+ sleep(ms) {
149
+ return new Promise(resolve => setTimeout(resolve, ms));
150
+ }
151
+ /**
152
+ * Internal request method.
153
+ */
154
+ request(method, path, body, options) {
155
+ return new Promise((resolve, reject) => {
156
+ const bodyStr = body ? JSON.stringify(body) : undefined;
157
+ const headers = {
158
+ 'Content-Type': 'application/json',
159
+ ...options?.headers,
160
+ };
161
+ // Propagate request ID for distributed tracing
162
+ if (options?.requestId) {
163
+ headers['X-Request-Id'] = options.requestId;
164
+ }
165
+ if (bodyStr) {
166
+ headers['Content-Length'] = Buffer.byteLength(bodyStr);
167
+ }
168
+ // Validate path to prevent protocol injection / request smuggling
169
+ if (path.includes('://') || path.startsWith('//') || /[\r\n\0]/.test(path)) {
170
+ throw new Error(`Invalid request path: ${path}`);
171
+ }
172
+ const requestOptions = {
173
+ hostname: this.config.host,
174
+ port: this.config.port,
175
+ path: path.startsWith('/') ? path : `/${path}`,
176
+ method,
177
+ headers,
178
+ timeout: options?.timeout ?? this.config.timeout,
179
+ agent: this.agent,
180
+ };
181
+ const req = http.request(requestOptions, (res) => {
182
+ let data = '';
183
+ res.on('data', (chunk) => {
184
+ data += chunk;
185
+ });
186
+ res.on('end', () => {
187
+ try {
188
+ const parsedBody = data ? JSON.parse(data) : {};
189
+ resolve({
190
+ statusCode: res.statusCode || http_status_1.HttpStatus.INTERNAL_SERVER_ERROR,
191
+ body: parsedBody,
192
+ headers: res.headers,
193
+ });
194
+ }
195
+ catch (parseError) {
196
+ logger.warn('Failed to parse response body', {
197
+ host: this.config.host,
198
+ path,
199
+ error: parseError instanceof Error ? parseError.message : String(parseError),
200
+ });
201
+ resolve({
202
+ statusCode: res.statusCode || http_status_1.HttpStatus.INTERNAL_SERVER_ERROR,
203
+ body: {},
204
+ headers: res.headers,
205
+ });
206
+ }
207
+ });
208
+ });
209
+ req.on('error', (error) => {
210
+ logger.error('HTTP request failed', {
211
+ host: this.config.host,
212
+ port: this.config.port,
213
+ path,
214
+ method,
215
+ error: error.message,
216
+ });
217
+ reject(error);
218
+ });
219
+ req.on('timeout', () => {
220
+ req.destroy();
221
+ const error = new Error(`Request timeout after ${this.config.timeout}ms`);
222
+ logger.warn('HTTP request timeout', {
223
+ host: this.config.host,
224
+ path,
225
+ timeout: this.config.timeout,
226
+ });
227
+ reject(error);
228
+ });
229
+ if (bodyStr) {
230
+ req.write(bodyStr);
231
+ }
232
+ req.end();
233
+ });
234
+ }
235
+ }
236
+ exports.InternalHttpClient = InternalHttpClient;
237
+ /**
238
+ * Create an HTTP client with error handling that returns null on failure.
239
+ * Useful for fail-open scenarios.
240
+ *
241
+ * @param config - Service configuration
242
+ * @returns Client wrapper with safe methods
243
+ */
244
+ function createSafeClient(config) {
245
+ const client = new InternalHttpClient(config);
246
+ return {
247
+ /**
248
+ * Safe GET request - returns null on error.
249
+ */
250
+ async get(path, options) {
251
+ try {
252
+ return await client.get(path, options);
253
+ }
254
+ catch (err) {
255
+ logger.debug('Safe GET failed, returning null', { path, error: err instanceof Error ? err.message : String(err) });
256
+ return null;
257
+ }
258
+ },
259
+ /**
260
+ * Safe POST request - returns null on error.
261
+ */
262
+ async post(path, body, options) {
263
+ try {
264
+ return await client.post(path, body, options);
265
+ }
266
+ catch (err) {
267
+ logger.debug('Safe POST failed, returning null', { path, error: err instanceof Error ? err.message : String(err) });
268
+ return null;
269
+ }
270
+ },
271
+ /**
272
+ * Safe PUT request - returns null on error.
273
+ */
274
+ async put(path, body, options) {
275
+ try {
276
+ return await client.put(path, body, options);
277
+ }
278
+ catch (err) {
279
+ logger.debug('Safe PUT failed, returning null', { path, error: err instanceof Error ? err.message : String(err) });
280
+ return null;
281
+ }
282
+ },
283
+ };
284
+ }
285
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http-client.js","sourceRoot":"","sources":["../../src/services/http-client.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';\nimport {\n  DEFAULT_MAX_RETRIES,\n  DEFAULT_RETRY_DELAY_MS,\n  DEFAULT_MAX_RATE_LIMIT_RETRIES,\n  getRetryDecision,\n  getErrorRetryDecision,\n  RetryConfig,\n} from './retry-strategy';\nimport { HttpStatus } from '../constants/http-status';\nimport { ServiceConfig } from '../types/common';\nimport { createLogger } from '../utils/logger';\n\n// Re-export retry utilities for backward compatibility\nexport { parseRetryAfter, addJitter } from './retry-strategy';\n\nconst logger = createLogger('http-client');\n\n/**\n * Default request timeout in milliseconds (env: `HTTP_CLIENT_TIMEOUT`).\n */\nconst DEFAULT_TIMEOUT = parseInt(process.env.HTTP_CLIENT_TIMEOUT || '5000', 10);\n\n/**\n * HTTP request options.\n */\nexport interface RequestOptions {\n  /** Request headers */\n  headers?: Record<string, string>;\n  /** Request timeout in milliseconds */\n  timeout?: number;\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  /** Optional request ID for distributed tracing (added as X-Request-Id header) */\n  requestId?: string;\n}\n\n/**\n * HTTP client response wrapper.\n */\nexport interface HttpClientResponse<T = unknown> {\n  /** HTTP status code */\n  statusCode: number;\n  /** Response body (parsed JSON) */\n  body: T;\n  /** Response headers */\n  headers: http.IncomingHttpHeaders;\n}\n\n/**\n * Internal HTTP client for service-to-service communication.\n *\n * @example\n * ```typescript\n * const client = new InternalHttpClient({\n *   host: 'quota',\n *   port: 3000,\n *   timeout: 5000,\n * });\n *\n * const response = await client.get('/org123/apiCalls');\n * const result = await client.post('/org123/increment', { quotaType: 'apiCalls' });\n * ```\n */\nexport class InternalHttpClient {\n  private config: Required<ServiceConfig>;\n  private agent: http.Agent;\n\n  /**\n   * Create a new HTTP client instance.\n   *\n   * @param config - Service configuration\n   */\n  constructor(config: ServiceConfig) {\n    this.config = {\n      host: config.host,\n      port: config.port,\n      timeout: config.timeout ?? DEFAULT_TIMEOUT,\n    };\n    this.agent = new http.Agent({ keepAlive: true });\n  }\n\n  /**\n   * Make a GET request.\n   */\n  async get<T = unknown>(path: string, options?: RequestOptions): Promise<HttpClientResponse<T>> {\n    return this.requestWithRetry<T>('GET', path, undefined, options);\n  }\n\n  /**\n   * Make a POST request.\n   */\n  async post<T = unknown>(\n    path: string,\n    body?: unknown,\n    options?: RequestOptions,\n  ): Promise<HttpClientResponse<T>> {\n    return this.requestWithRetry<T>('POST', path, body, options);\n  }\n\n  /**\n   * Make a PUT request.\n   */\n  async put<T = unknown>(\n    path: string,\n    body?: unknown,\n    options?: RequestOptions,\n  ): Promise<HttpClientResponse<T>> {\n    return this.requestWithRetry<T>('PUT', path, body, options);\n  }\n\n  /**\n   * Make a DELETE request.\n   */\n  async delete<T = unknown>(path: string, options?: RequestOptions): Promise<HttpClientResponse<T>> {\n    return this.requestWithRetry<T>('DELETE', path, undefined, options);\n  }\n\n  /**\n   * Request with retry logic for transient failures.\n   *\n   * - 429 (rate limit): respects `Retry-After` header, uses longer base delay (4x),\n   *   retries up to `maxRateLimitRetries` times (default 4).\n   * - 502/503/504 (server errors): standard exponential backoff, up to `maxRetries` (default 2).\n   * - Connection errors / timeouts: retries up to `maxRetries`.\n   * - All delays include ±25% jitter to prevent thundering herd.\n   */\n  private async requestWithRetry<T>(\n    method: string,\n    path: string,\n    body?: unknown,\n    options?: RequestOptions,\n  ): Promise<HttpClientResponse<T>> {\n    const retryConfig: RetryConfig = {\n      maxRetries: options?.maxRetries ?? DEFAULT_MAX_RETRIES,\n      maxRateLimitRetries: options?.maxRateLimitRetries ?? DEFAULT_MAX_RATE_LIMIT_RETRIES,\n      retryDelayMs: options?.retryDelayMs ?? DEFAULT_RETRY_DELAY_MS,\n    };\n    const totalMaxAttempts = Math.max(retryConfig.maxRetries, retryConfig.maxRateLimitRetries);\n\n    let lastError: Error | undefined;\n\n    for (let attempt = 0; attempt <= totalMaxAttempts; attempt++) {\n      try {\n        const response = await this.request<T>(method, path, body, options);\n\n        const decision = getRetryDecision(response.statusCode, response.headers, attempt, retryConfig);\n        if (decision.shouldRetry) {\n          logger.debug(decision.reason + ', retrying', { method, path, attempt: attempt + 1, delayMs: decision.delayMs });\n          await this.sleep(decision.delayMs);\n          continue;\n        }\n\n        return response;\n      } catch (error) {\n        lastError = error instanceof Error ? error : new Error(String(error));\n\n        const decision = getErrorRetryDecision(attempt, retryConfig);\n        if (decision.shouldRetry) {\n          logger.debug('Retrying after error', { method, path, error: lastError.message, attempt: attempt + 1 });\n          await this.sleep(decision.delayMs);\n          continue;\n        }\n      }\n    }\n\n    throw lastError!;\n  }\n\n  private sleep(ms: number): Promise<void> {\n    return new Promise(resolve => setTimeout(resolve, ms));\n  }\n\n  /**\n   * Internal request method.\n   */\n  private request<T>(\n    method: string,\n    path: string,\n    body?: unknown,\n    options?: RequestOptions,\n  ): Promise<HttpClientResponse<T>> {\n    return new Promise((resolve, reject) => {\n      const bodyStr = body ? JSON.stringify(body) : undefined;\n\n      const headers: http.OutgoingHttpHeaders = {\n        'Content-Type': 'application/json',\n        ...options?.headers,\n      };\n\n      // Propagate request ID for distributed tracing\n      if (options?.requestId) {\n        headers['X-Request-Id'] = options.requestId;\n      }\n\n      if (bodyStr) {\n        headers['Content-Length'] = Buffer.byteLength(bodyStr);\n      }\n\n      // Validate path to prevent protocol injection / request smuggling\n      if (path.includes('://') || path.startsWith('//') || /[\\r\\n\\0]/.test(path)) {\n        throw new Error(`Invalid request path: ${path}`);\n      }\n\n      const requestOptions: http.RequestOptions = {\n        hostname: this.config.host,\n        port: this.config.port,\n        path: path.startsWith('/') ? path : `/${path}`,\n        method,\n        headers,\n        timeout: options?.timeout ?? this.config.timeout,\n        agent: this.agent,\n      };\n\n      const req = http.request(requestOptions, (res) => {\n        let data = '';\n\n        res.on('data', (chunk) => {\n          data += chunk;\n        });\n\n        res.on('end', () => {\n          try {\n            const parsedBody = data ? JSON.parse(data) : {};\n            resolve({\n              statusCode: res.statusCode || HttpStatus.INTERNAL_SERVER_ERROR,\n              body: parsedBody as T,\n              headers: res.headers,\n            });\n          } catch (parseError) {\n            logger.warn('Failed to parse response body', {\n              host: this.config.host,\n              path,\n              error: parseError instanceof Error ? parseError.message : String(parseError),\n            });\n            resolve({\n              statusCode: res.statusCode || HttpStatus.INTERNAL_SERVER_ERROR,\n              body: {} as T,\n              headers: res.headers,\n            });\n          }\n        });\n      });\n\n      req.on('error', (error) => {\n        logger.error('HTTP request failed', {\n          host: this.config.host,\n          port: this.config.port,\n          path,\n          method,\n          error: error.message,\n        });\n        reject(error);\n      });\n\n      req.on('timeout', () => {\n        req.destroy();\n        const error = new Error(`Request timeout after ${this.config.timeout}ms`);\n        logger.warn('HTTP request timeout', {\n          host: this.config.host,\n          path,\n          timeout: this.config.timeout,\n        });\n        reject(error);\n      });\n\n      if (bodyStr) {\n        req.write(bodyStr);\n      }\n\n      req.end();\n    });\n  }\n}\n\n/**\n * Create an HTTP client with error handling that returns null on failure.\n * Useful for fail-open scenarios.\n *\n * @param config - Service configuration\n * @returns Client wrapper with safe methods\n */\nexport function createSafeClient(config: ServiceConfig) {\n  const client = new InternalHttpClient(config);\n\n  return {\n    /**\n     * Safe GET request - returns null on error.\n     */\n    async get<T>(path: string, options?: RequestOptions): Promise<HttpClientResponse<T> | null> {\n      try {\n        return await client.get<T>(path, options);\n      } catch (err) {\n        logger.debug('Safe GET failed, returning null', { path, error: err instanceof Error ? err.message : String(err) });\n        return null;\n      }\n    },\n\n    /**\n     * Safe POST request - returns null on error.\n     */\n    async post<T>(\n      path: string,\n      body?: unknown,\n      options?: RequestOptions,\n    ): Promise<HttpClientResponse<T> | null> {\n      try {\n        return await client.post<T>(path, body, options);\n      } catch (err) {\n        logger.debug('Safe POST failed, returning null', { path, error: err instanceof Error ? err.message : String(err) });\n        return null;\n      }\n    },\n\n    /**\n     * Safe PUT request - returns null on error.\n     */\n    async put<T>(\n      path: string,\n      body?: unknown,\n      options?: RequestOptions,\n    ): Promise<HttpClientResponse<T> | null> {\n      try {\n        return await client.put<T>(path, body, options);\n      } catch (err) {\n        logger.debug('Safe PUT failed, returning null', { path, error: err instanceof Error ? err.message : String(err) });\n        return null;\n      }\n    },\n  };\n}\n"]}
@@ -0,0 +1,10 @@
1
+ export * from './cache-service';
2
+ export * from './compliance-client';
3
+ export * from './compliance-event-subscriber';
4
+ export * from './entity-events';
5
+ export * from './http-client';
6
+ export { DEFAULT_MAX_RETRIES, DEFAULT_RETRY_DELAY_MS, DEFAULT_MAX_RATE_LIMIT_RETRIES, calculateBackoff, isTransientStatusCode, isRateLimited, getRetryDecision, getErrorRetryDecision, } from './retry-strategy';
7
+ export type { RetryConfig, RetryDecision } from './retry-strategy';
8
+ export * from './quota';
9
+ export { logAdminAction, type AuditEntry } from './admin-audit';
10
+ export { enqueueComplianceEvent, registerComplianceQueueBackend, type ComplianceEvent } from './compliance-queue';
@@ -0,0 +1,40 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
5
+ if (k2 === undefined) k2 = k;
6
+ var desc = Object.getOwnPropertyDescriptor(m, k);
7
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
8
+ desc = { enumerable: true, get: function() { return m[k]; } };
9
+ }
10
+ Object.defineProperty(o, k2, desc);
11
+ }) : (function(o, m, k, k2) {
12
+ if (k2 === undefined) k2 = k;
13
+ o[k2] = m[k];
14
+ }));
15
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
16
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
17
+ };
18
+ Object.defineProperty(exports, "__esModule", { value: true });
19
+ exports.registerComplianceQueueBackend = exports.enqueueComplianceEvent = exports.logAdminAction = exports.getErrorRetryDecision = exports.getRetryDecision = exports.isRateLimited = exports.isTransientStatusCode = exports.calculateBackoff = exports.DEFAULT_MAX_RATE_LIMIT_RETRIES = exports.DEFAULT_RETRY_DELAY_MS = exports.DEFAULT_MAX_RETRIES = void 0;
20
+ __exportStar(require("./cache-service"), exports);
21
+ __exportStar(require("./compliance-client"), exports);
22
+ __exportStar(require("./compliance-event-subscriber"), exports);
23
+ __exportStar(require("./entity-events"), exports);
24
+ __exportStar(require("./http-client"), exports);
25
+ var retry_strategy_1 = require("./retry-strategy");
26
+ Object.defineProperty(exports, "DEFAULT_MAX_RETRIES", { enumerable: true, get: function () { return retry_strategy_1.DEFAULT_MAX_RETRIES; } });
27
+ Object.defineProperty(exports, "DEFAULT_RETRY_DELAY_MS", { enumerable: true, get: function () { return retry_strategy_1.DEFAULT_RETRY_DELAY_MS; } });
28
+ Object.defineProperty(exports, "DEFAULT_MAX_RATE_LIMIT_RETRIES", { enumerable: true, get: function () { return retry_strategy_1.DEFAULT_MAX_RATE_LIMIT_RETRIES; } });
29
+ Object.defineProperty(exports, "calculateBackoff", { enumerable: true, get: function () { return retry_strategy_1.calculateBackoff; } });
30
+ Object.defineProperty(exports, "isTransientStatusCode", { enumerable: true, get: function () { return retry_strategy_1.isTransientStatusCode; } });
31
+ Object.defineProperty(exports, "isRateLimited", { enumerable: true, get: function () { return retry_strategy_1.isRateLimited; } });
32
+ Object.defineProperty(exports, "getRetryDecision", { enumerable: true, get: function () { return retry_strategy_1.getRetryDecision; } });
33
+ Object.defineProperty(exports, "getErrorRetryDecision", { enumerable: true, get: function () { return retry_strategy_1.getErrorRetryDecision; } });
34
+ __exportStar(require("./quota"), exports);
35
+ var admin_audit_1 = require("./admin-audit");
36
+ Object.defineProperty(exports, "logAdminAction", { enumerable: true, get: function () { return admin_audit_1.logAdminAction; } });
37
+ var compliance_queue_1 = require("./compliance-queue");
38
+ Object.defineProperty(exports, "enqueueComplianceEvent", { enumerable: true, get: function () { return compliance_queue_1.enqueueComplianceEvent; } });
39
+ Object.defineProperty(exports, "registerComplianceQueueBackend", { enumerable: true, get: function () { return compliance_queue_1.registerComplianceQueueBackend; } });
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2VydmljZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRXRDLGtEQUFnQztBQUNoQyxzREFBb0M7QUFDcEMsZ0VBQThDO0FBQzlDLGtEQUFnQztBQUNoQyxnREFBOEI7QUFDOUIsbURBUzBCO0FBUnhCLHFIQUFBLG1CQUFtQixPQUFBO0FBQ25CLHdIQUFBLHNCQUFzQixPQUFBO0FBQ3RCLGdJQUFBLDhCQUE4QixPQUFBO0FBQzlCLGtIQUFBLGdCQUFnQixPQUFBO0FBQ2hCLHVIQUFBLHFCQUFxQixPQUFBO0FBQ3JCLCtHQUFBLGFBQWEsT0FBQTtBQUNiLGtIQUFBLGdCQUFnQixPQUFBO0FBQ2hCLHVIQUFBLHFCQUFxQixPQUFBO0FBR3ZCLDBDQUF3QjtBQUN4Qiw2Q0FBZ0U7QUFBdkQsNkdBQUEsY0FBYyxPQUFBO0FBQ3ZCLHVEQUFrSDtBQUF6RywwSEFBQSxzQkFBc0IsT0FBQTtBQUFFLGtJQUFBLDhCQUE4QixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IDIwMjYgUGlwZWxpbmUgQnVpbGRlciBDb250cmlidXRvcnNcbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5cbmV4cG9ydCAqIGZyb20gJy4vY2FjaGUtc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2NvbXBsaWFuY2UtY2xpZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY29tcGxpYW5jZS1ldmVudC1zdWJzY3JpYmVyJztcbmV4cG9ydCAqIGZyb20gJy4vZW50aXR5LWV2ZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2h0dHAtY2xpZW50JztcbmV4cG9ydCB7XG4gIERFRkFVTFRfTUFYX1JFVFJJRVMsXG4gIERFRkFVTFRfUkVUUllfREVMQVlfTVMsXG4gIERFRkFVTFRfTUFYX1JBVEVfTElNSVRfUkVUUklFUyxcbiAgY2FsY3VsYXRlQmFja29mZixcbiAgaXNUcmFuc2llbnRTdGF0dXNDb2RlLFxuICBpc1JhdGVMaW1pdGVkLFxuICBnZXRSZXRyeURlY2lzaW9uLFxuICBnZXRFcnJvclJldHJ5RGVjaXNpb24sXG59IGZyb20gJy4vcmV0cnktc3RyYXRlZ3knO1xuZXhwb3J0IHR5cGUgeyBSZXRyeUNvbmZpZywgUmV0cnlEZWNpc2lvbiB9IGZyb20gJy4vcmV0cnktc3RyYXRlZ3knO1xuZXhwb3J0ICogZnJvbSAnLi9xdW90YSc7XG5leHBvcnQgeyBsb2dBZG1pbkFjdGlvbiwgdHlwZSBBdWRpdEVudHJ5IH0gZnJvbSAnLi9hZG1pbi1hdWRpdCc7XG5leHBvcnQgeyBlbnF1ZXVlQ29tcGxpYW5jZUV2ZW50LCByZWdpc3RlckNvbXBsaWFuY2VRdWV1ZUJhY2tlbmQsIHR5cGUgQ29tcGxpYW5jZUV2ZW50IH0gZnJvbSAnLi9jb21wbGlhbmNlLXF1ZXVlJztcbiJdfQ==