@anatix/runtime 1.0.1 → 1.0.2

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 (198) hide show
  1. package/dist/contracts/service-options.d.ts +0 -5
  2. package/dist/contracts/service-options.d.ts.map +1 -1
  3. package/dist/decorators/filters/exception-filter.config.d.ts +7 -0
  4. package/dist/decorators/filters/exception-filter.config.d.ts.map +1 -0
  5. package/dist/decorators/filters/exception-filter.config.js +9 -0
  6. package/dist/decorators/filters/exception-filter.config.js.map +1 -0
  7. package/dist/decorators/filters/global-exception.filter.d.ts +3 -2
  8. package/dist/decorators/filters/global-exception.filter.d.ts.map +1 -1
  9. package/dist/decorators/filters/global-exception.filter.js +14 -5
  10. package/dist/decorators/filters/global-exception.filter.js.map +1 -1
  11. package/dist/decorators/filters/index.d.ts +1 -0
  12. package/dist/decorators/filters/index.d.ts.map +1 -1
  13. package/dist/decorators/filters/index.js +1 -0
  14. package/dist/decorators/filters/index.js.map +1 -1
  15. package/dist/dtos/error-response.dto.d.ts +20 -0
  16. package/dist/dtos/error-response.dto.d.ts.map +1 -0
  17. package/dist/dtos/error-response.dto.js +42 -0
  18. package/dist/dtos/error-response.dto.js.map +1 -0
  19. package/dist/dtos/index.d.ts +3 -0
  20. package/dist/dtos/index.d.ts.map +1 -0
  21. package/dist/dtos/index.js +13 -0
  22. package/dist/dtos/index.js.map +1 -0
  23. package/dist/dtos/paginated-response.dto.d.ts +16 -0
  24. package/dist/dtos/paginated-response.dto.d.ts.map +1 -0
  25. package/dist/dtos/paginated-response.dto.js +31 -0
  26. package/dist/dtos/paginated-response.dto.js.map +1 -0
  27. package/dist/envelope/repositories/outbox.repository.d.ts.map +1 -1
  28. package/dist/envelope/repositories/outbox.repository.js +1 -1
  29. package/dist/envelope/repositories/outbox.repository.js.map +1 -1
  30. package/dist/envelope/services/inbox-retry.service.d.ts +3 -2
  31. package/dist/envelope/services/inbox-retry.service.d.ts.map +1 -1
  32. package/dist/envelope/services/inbox-retry.service.js +4 -0
  33. package/dist/envelope/services/inbox-retry.service.js.map +1 -1
  34. package/dist/envelope/services/outbox-dispatcher.service.d.ts +5 -2
  35. package/dist/envelope/services/outbox-dispatcher.service.d.ts.map +1 -1
  36. package/dist/envelope/services/outbox-dispatcher.service.js +61 -24
  37. package/dist/envelope/services/outbox-dispatcher.service.js.map +1 -1
  38. package/dist/envelope/tx-manager.d.ts +1 -1
  39. package/dist/envelope/tx-manager.d.ts.map +1 -1
  40. package/dist/event-bus/adapters/index.d.ts +4 -0
  41. package/dist/event-bus/adapters/index.d.ts.map +1 -0
  42. package/dist/event-bus/adapters/index.js +20 -0
  43. package/dist/event-bus/adapters/index.js.map +1 -0
  44. package/dist/event-bus/adapters/kafka.adapter.d.ts +28 -0
  45. package/dist/event-bus/adapters/kafka.adapter.d.ts.map +1 -0
  46. package/dist/event-bus/adapters/kafka.adapter.js +263 -0
  47. package/dist/event-bus/adapters/kafka.adapter.js.map +1 -0
  48. package/dist/event-bus/adapters/message-broker.interface.d.ts +39 -0
  49. package/dist/event-bus/adapters/message-broker.interface.d.ts.map +1 -0
  50. package/dist/event-bus/adapters/message-broker.interface.js +3 -0
  51. package/dist/event-bus/adapters/message-broker.interface.js.map +1 -0
  52. package/dist/event-bus/adapters/rabbitmq.adapter.d.ts +21 -0
  53. package/dist/event-bus/adapters/rabbitmq.adapter.d.ts.map +1 -0
  54. package/dist/event-bus/adapters/rabbitmq.adapter.js +232 -0
  55. package/dist/event-bus/adapters/rabbitmq.adapter.js.map +1 -0
  56. package/dist/event-bus/config/adapter.config.d.ts +65 -0
  57. package/dist/event-bus/config/adapter.config.d.ts.map +1 -0
  58. package/dist/event-bus/config/adapter.config.js +6 -0
  59. package/dist/event-bus/config/adapter.config.js.map +1 -0
  60. package/dist/event-bus/config/index.d.ts +2 -0
  61. package/dist/event-bus/config/index.d.ts.map +1 -0
  62. package/dist/event-bus/config/index.js +18 -0
  63. package/dist/event-bus/config/index.js.map +1 -0
  64. package/dist/event-bus/event-bus.config.d.ts +19 -0
  65. package/dist/event-bus/event-bus.config.d.ts.map +1 -1
  66. package/dist/event-bus/event-bus.config.js.map +1 -1
  67. package/dist/event-bus/services/event-bus.service.d.ts +7 -6
  68. package/dist/event-bus/services/event-bus.service.d.ts.map +1 -1
  69. package/dist/event-bus/services/event-bus.service.js +96 -210
  70. package/dist/event-bus/services/event-bus.service.js.map +1 -1
  71. package/dist/event-bus/types/event-bus-adapter.interface.d.ts +43 -0
  72. package/dist/event-bus/types/event-bus-adapter.interface.d.ts.map +1 -0
  73. package/dist/event-bus/types/event-bus-adapter.interface.js +3 -0
  74. package/dist/event-bus/types/event-bus-adapter.interface.js.map +1 -0
  75. package/dist/health/health.service.d.ts +1 -0
  76. package/dist/health/health.service.d.ts.map +1 -1
  77. package/dist/health/health.service.js +12 -3
  78. package/dist/health/health.service.js.map +1 -1
  79. package/dist/index.d.ts +7 -2
  80. package/dist/index.d.ts.map +1 -1
  81. package/dist/index.js +61 -2
  82. package/dist/index.js.map +1 -1
  83. package/dist/monitoring/index.d.ts +2 -0
  84. package/dist/monitoring/index.d.ts.map +1 -0
  85. package/dist/monitoring/index.js +18 -0
  86. package/dist/monitoring/index.js.map +1 -0
  87. package/dist/proxy/config/proxy.config.d.ts +6 -2
  88. package/dist/proxy/config/proxy.config.d.ts.map +1 -1
  89. package/dist/proxy/config/proxy.config.js +6 -3
  90. package/dist/proxy/config/proxy.config.js.map +1 -1
  91. package/dist/proxy/methods/proxy-delete.d.ts.map +1 -1
  92. package/dist/proxy/methods/proxy-delete.js +2 -2
  93. package/dist/proxy/methods/proxy-delete.js.map +1 -1
  94. package/dist/proxy/methods/proxy-get.d.ts.map +1 -1
  95. package/dist/proxy/methods/proxy-get.js +2 -2
  96. package/dist/proxy/methods/proxy-get.js.map +1 -1
  97. package/dist/proxy/methods/proxy-patch.d.ts.map +1 -1
  98. package/dist/proxy/methods/proxy-patch.js +2 -2
  99. package/dist/proxy/methods/proxy-patch.js.map +1 -1
  100. package/dist/proxy/methods/proxy-post.d.ts.map +1 -1
  101. package/dist/proxy/methods/proxy-post.js +2 -2
  102. package/dist/proxy/methods/proxy-post.js.map +1 -1
  103. package/dist/proxy/methods/proxy-put.d.ts.map +1 -1
  104. package/dist/proxy/methods/proxy-put.js +2 -2
  105. package/dist/proxy/methods/proxy-put.js.map +1 -1
  106. package/dist/repository/bulk.d.ts +12 -0
  107. package/dist/repository/bulk.d.ts.map +1 -0
  108. package/dist/repository/bulk.js +107 -0
  109. package/dist/repository/bulk.js.map +1 -0
  110. package/dist/repository/cursor.d.ts +12 -5
  111. package/dist/repository/cursor.d.ts.map +1 -1
  112. package/dist/repository/cursor.js +57 -29
  113. package/dist/repository/cursor.js.map +1 -1
  114. package/dist/repository/index.d.ts +6 -1
  115. package/dist/repository/index.d.ts.map +1 -1
  116. package/dist/repository/index.js +22 -15
  117. package/dist/repository/index.js.map +1 -1
  118. package/dist/repository/lock.d.ts +8 -0
  119. package/dist/repository/lock.d.ts.map +1 -0
  120. package/dist/repository/lock.js +35 -0
  121. package/dist/repository/lock.js.map +1 -0
  122. package/dist/repository/retry.d.ts +7 -0
  123. package/dist/repository/retry.d.ts.map +1 -0
  124. package/dist/repository/retry.js +43 -0
  125. package/dist/repository/retry.js.map +1 -0
  126. package/dist/repository/scope.d.ts +8 -0
  127. package/dist/repository/scope.d.ts.map +1 -0
  128. package/dist/repository/scope.js +16 -0
  129. package/dist/repository/scope.js.map +1 -0
  130. package/dist/result/index.d.ts +2 -0
  131. package/dist/result/index.d.ts.map +1 -0
  132. package/dist/result/index.js +16 -0
  133. package/dist/result/index.js.map +1 -0
  134. package/dist/result/result.d.ts +26 -0
  135. package/dist/result/result.d.ts.map +1 -0
  136. package/dist/result/result.js +73 -0
  137. package/dist/result/result.js.map +1 -0
  138. package/dist/security/bcrypt.adapter.d.ts +3 -0
  139. package/dist/security/bcrypt.adapter.d.ts.map +1 -1
  140. package/dist/security/bcrypt.adapter.js +6 -1
  141. package/dist/security/bcrypt.adapter.js.map +1 -1
  142. package/dist/security/index.d.ts +1 -0
  143. package/dist/security/index.d.ts.map +1 -1
  144. package/dist/security/index.js +1 -0
  145. package/dist/security/index.js.map +1 -1
  146. package/dist/security/security.config.d.ts +6 -0
  147. package/dist/security/security.config.d.ts.map +1 -0
  148. package/dist/security/security.config.js +8 -0
  149. package/dist/security/security.config.js.map +1 -0
  150. package/dist/service/cache.d.ts +11 -0
  151. package/dist/service/cache.d.ts.map +1 -0
  152. package/dist/service/cache.js +45 -0
  153. package/dist/service/cache.js.map +1 -0
  154. package/dist/service/circuit-breaker.d.ts +19 -0
  155. package/dist/service/circuit-breaker.d.ts.map +1 -0
  156. package/dist/service/circuit-breaker.js +73 -0
  157. package/dist/service/circuit-breaker.js.map +1 -0
  158. package/dist/service/enrichment.d.ts +8 -0
  159. package/dist/service/enrichment.d.ts.map +1 -0
  160. package/dist/service/enrichment.js +40 -0
  161. package/dist/service/enrichment.js.map +1 -0
  162. package/dist/service/idempotency.d.ts +8 -0
  163. package/dist/service/idempotency.d.ts.map +1 -0
  164. package/dist/service/idempotency.js +28 -0
  165. package/dist/service/idempotency.js.map +1 -0
  166. package/dist/service/index.d.ts +10 -0
  167. package/dist/service/index.d.ts.map +1 -0
  168. package/dist/service/index.js +45 -0
  169. package/dist/service/index.js.map +1 -0
  170. package/dist/service/projection.d.ts +14 -0
  171. package/dist/service/projection.d.ts.map +1 -0
  172. package/dist/service/projection.js +59 -0
  173. package/dist/service/projection.js.map +1 -0
  174. package/dist/service/rate-limit.d.ts +10 -0
  175. package/dist/service/rate-limit.d.ts.map +1 -0
  176. package/dist/service/rate-limit.js +42 -0
  177. package/dist/service/rate-limit.js.map +1 -0
  178. package/dist/service/retry.d.ts +8 -0
  179. package/dist/service/retry.d.ts.map +1 -0
  180. package/dist/service/retry.js +62 -0
  181. package/dist/service/retry.js.map +1 -0
  182. package/dist/service/timeout.d.ts +7 -0
  183. package/dist/service/timeout.d.ts.map +1 -0
  184. package/dist/service/timeout.js +39 -0
  185. package/dist/service/timeout.js.map +1 -0
  186. package/dist/service/transaction.d.ts +12 -0
  187. package/dist/service/transaction.d.ts.map +1 -0
  188. package/dist/service/transaction.js +17 -0
  189. package/dist/service/transaction.js.map +1 -0
  190. package/dist/utils/index.d.ts +1 -0
  191. package/dist/utils/index.d.ts.map +1 -1
  192. package/dist/utils/index.js +1 -0
  193. package/dist/utils/index.js.map +1 -1
  194. package/dist/utils/retry.d.ts +7 -0
  195. package/dist/utils/retry.d.ts.map +1 -0
  196. package/dist/utils/retry.js +27 -0
  197. package/dist/utils/retry.js.map +1 -0
  198. package/package.json +25 -4
@@ -0,0 +1,20 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./message-broker.interface"), exports);
18
+ __exportStar(require("./rabbitmq.adapter"), exports);
19
+ __exportStar(require("./kafka.adapter"), exports);
20
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/event-bus/adapters/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAOA,6DAA2C;AAG3C,qDAAmC;AACnC,kDAAgC"}
@@ -0,0 +1,28 @@
1
+ import type { MessageBrokerAdapter, BrokerConnectionConfig, MessageHandler, PublishOptions } from './message-broker.interface';
2
+ import type { EventMeta } from '../../envelope';
3
+ export declare class KafkaAdapter implements MessageBrokerAdapter {
4
+ private readonly logger;
5
+ private kafka?;
6
+ private producer?;
7
+ private consumer?;
8
+ private admin?;
9
+ private serviceName;
10
+ private handlers;
11
+ private rpcHandlers;
12
+ private connected;
13
+ connect(config: BrokerConnectionConfig): Promise<void>;
14
+ disconnect(): Promise<void>;
15
+ publish(topic: string, payload: any, meta?: EventMeta, options?: PublishOptions): Promise<void>;
16
+ request<T = any>(topic: string, payload: any, meta?: EventMeta, timeoutMs?: number): Promise<T>;
17
+ reply(replyTo: string, correlationId: string, payload: any, meta?: EventMeta): Promise<void>;
18
+ subscribe(topic: string, handler: MessageHandler): Promise<void>;
19
+ unsubscribe(topic: string): Promise<void>;
20
+ isConnected(): boolean;
21
+ createTopic(name: string, options?: {
22
+ numPartitions?: number;
23
+ replicationFactor?: number;
24
+ }): Promise<void>;
25
+ private handleMessage;
26
+ private encodeHeaders;
27
+ }
28
+ //# sourceMappingURL=kafka.adapter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"kafka.adapter.d.ts","sourceRoot":"","sources":["../../../src/event-bus/adapters/kafka.adapter.ts"],"names":[],"mappings":"AAWA,OAAO,KAAK,EACV,oBAAoB,EACpB,sBAAsB,EAEtB,cAAc,EACd,cAAc,EACf,MAAM,4BAA4B,CAAC;AACpC,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AA6BhD,qBAAa,YAAa,YAAW,oBAAoB;IACvD,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAiC;IAExD,OAAO,CAAC,KAAK,CAAC,CAAQ;IACtB,OAAO,CAAC,QAAQ,CAAC,CAAW;IAC5B,OAAO,CAAC,QAAQ,CAAC,CAAW;IAC5B,OAAO,CAAC,KAAK,CAAC,CAAQ;IACtB,OAAO,CAAC,WAAW,CAAqB;IACxC,OAAO,CAAC,QAAQ,CAAuC;IACvD,OAAO,CAAC,WAAW,CAA8C;IACjE,OAAO,CAAC,SAAS,CAAS;IAKpB,OAAO,CAAC,MAAM,EAAE,sBAAsB,GAAG,OAAO,CAAC,IAAI,CAAC;IA4DtD,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IA2B3B,OAAO,CACX,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,EAChB,OAAO,CAAC,EAAE,cAAc,GACvB,OAAO,CAAC,IAAI,CAAC;IAmCV,OAAO,CAAC,CAAC,GAAG,GAAG,EACnB,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,EAChB,SAAS,SAAO,GACf,OAAO,CAAC,CAAC,CAAC;IAgEP,KAAK,CACT,OAAO,EAAE,MAAM,EACf,aAAa,EAAE,MAAM,EACrB,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,GACf,OAAO,CAAC,IAAI,CAAC;IA0BV,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;IAuBhE,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAU/C,WAAW,IAAI,OAAO;IAOhB,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE;QAAE,aAAa,CAAC,EAAE,MAAM,CAAC;QAAC,iBAAiB,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC,IAAI,CAAC;YA2BlG,aAAa;IAkE3B,OAAO,CAAC,aAAa;CAStB"}
@@ -0,0 +1,263 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KafkaAdapter = void 0;
4
+ const common_1 = require("@nestjs/common");
5
+ const kafkajs_1 = require("kafkajs");
6
+ const envelope_1 = require("../../envelope");
7
+ class KafkaAdapter {
8
+ logger = new common_1.Logger(KafkaAdapter.name);
9
+ kafka;
10
+ producer;
11
+ consumer;
12
+ admin;
13
+ serviceName = 'default-service';
14
+ handlers = new Map();
15
+ rpcHandlers = new Map();
16
+ connected = false;
17
+ async connect(config) {
18
+ const brokers = Array.isArray(config.url)
19
+ ? config.url
20
+ : [config.url || 'localhost:9092'];
21
+ const clientId = config.options?.clientId ?? config.serviceName ?? 'anatix-client';
22
+ const groupId = config.options?.groupId ?? config.serviceName ?? 'anatix-group';
23
+ this.serviceName = config.serviceName ?? 'default-service';
24
+ try {
25
+ this.kafka = new kafkajs_1.Kafka({
26
+ clientId,
27
+ brokers,
28
+ logLevel: kafkajs_1.logLevel.ERROR,
29
+ ...config.options?.kafkaConfig,
30
+ });
31
+ const producerConfig = {
32
+ allowAutoTopicCreation: config.options?.producerConfig?.allowAutoTopicCreation ?? true,
33
+ idempotent: config.options?.producerConfig?.idempotent ?? false,
34
+ transactionTimeout: config.options?.producerConfig?.transactionTimeout ?? 60000,
35
+ maxInFlightRequests: config.options?.producerConfig?.maxInFlightRequests,
36
+ retry: config.options?.producerConfig?.retry,
37
+ };
38
+ this.producer = this.kafka.producer(producerConfig);
39
+ await this.producer.connect();
40
+ this.consumer = this.kafka.consumer({
41
+ groupId,
42
+ ...config.options?.consumerConfig,
43
+ });
44
+ await this.consumer.connect();
45
+ this.admin = this.kafka.admin();
46
+ await this.admin.connect();
47
+ await this.consumer.run({
48
+ eachMessage: async (payload) => this.handleMessage(payload),
49
+ });
50
+ this.connected = true;
51
+ this.logger.log(`Connected to Kafka: ${brokers.join(', ')} (clientId: ${clientId}, groupId: ${groupId})`);
52
+ }
53
+ catch (error) {
54
+ this.logger.error(`Failed to connect to Kafka: ${error?.message || error}`);
55
+ throw error;
56
+ }
57
+ }
58
+ async disconnect() {
59
+ try {
60
+ if (this.producer) {
61
+ await this.producer.disconnect();
62
+ this.producer = undefined;
63
+ }
64
+ if (this.consumer) {
65
+ await this.consumer.disconnect();
66
+ this.consumer = undefined;
67
+ }
68
+ if (this.admin) {
69
+ await this.admin.disconnect();
70
+ this.admin = undefined;
71
+ }
72
+ this.connected = false;
73
+ this.logger.log('Disconnected from Kafka');
74
+ }
75
+ catch (error) {
76
+ this.logger.error(`Error disconnecting from Kafka: ${error?.message || error}`);
77
+ }
78
+ }
79
+ async publish(topic, payload, meta, options) {
80
+ if (!this.producer) {
81
+ throw new Error('Kafka adapter not connected');
82
+ }
83
+ const eventMeta = (0, envelope_1.ensureMeta)(meta);
84
+ const message = {
85
+ __event: topic,
86
+ data: payload,
87
+ meta: eventMeta,
88
+ };
89
+ const value = JSON.stringify(message);
90
+ await this.producer.send({
91
+ topic,
92
+ messages: [
93
+ {
94
+ key: options?.key,
95
+ value,
96
+ partition: options?.partition,
97
+ headers: options?.headers
98
+ ? this.encodeHeaders(options.headers)
99
+ : undefined,
100
+ },
101
+ ],
102
+ });
103
+ }
104
+ async request(topic, payload, meta, timeoutMs = 5000) {
105
+ if (!this.producer || !this.consumer) {
106
+ throw new Error('Kafka adapter not connected');
107
+ }
108
+ const eventMeta = (0, envelope_1.ensureMeta)(meta);
109
+ const replyTopic = `${this.serviceName}-replies`;
110
+ const correlationId = eventMeta.correlationId;
111
+ if (!this.handlers.has(replyTopic)) {
112
+ await this.subscribe(replyTopic, async (msg) => {
113
+ const handler = this.rpcHandlers.get(msg.meta?.correlationId || '');
114
+ if (handler) {
115
+ handler(msg.payload);
116
+ this.rpcHandlers.delete(msg.meta?.correlationId || '');
117
+ }
118
+ await msg.ack();
119
+ });
120
+ }
121
+ const responsePromise = new Promise((resolve, reject) => {
122
+ this.rpcHandlers.set(correlationId, (response) => {
123
+ clearTimeout(timer);
124
+ resolve(response);
125
+ });
126
+ const timer = setTimeout(() => {
127
+ this.rpcHandlers.delete(correlationId);
128
+ reject(new Error(`RPC timeout for topic "${topic}" (correlationId: ${correlationId})`));
129
+ }, timeoutMs);
130
+ });
131
+ const message = {
132
+ __event: topic,
133
+ __replyTo: replyTopic,
134
+ __correlationId: correlationId,
135
+ data: payload,
136
+ meta: eventMeta,
137
+ };
138
+ await this.producer.send({
139
+ topic,
140
+ messages: [
141
+ {
142
+ value: JSON.stringify(message),
143
+ },
144
+ ],
145
+ });
146
+ return responsePromise;
147
+ }
148
+ async reply(replyTo, correlationId, payload, meta) {
149
+ if (!this.producer) {
150
+ throw new Error('Kafka adapter not connected');
151
+ }
152
+ const eventMeta = (0, envelope_1.ensureMeta)(meta);
153
+ const message = {
154
+ __correlationId: correlationId,
155
+ data: payload,
156
+ meta: eventMeta,
157
+ };
158
+ await this.producer.send({
159
+ topic: replyTo,
160
+ messages: [
161
+ {
162
+ value: JSON.stringify(message),
163
+ },
164
+ ],
165
+ });
166
+ }
167
+ async subscribe(topic, handler) {
168
+ if (!this.consumer) {
169
+ throw new Error('Kafka adapter not connected');
170
+ }
171
+ if (!this.handlers.has(topic)) {
172
+ this.handlers.set(topic, []);
173
+ await this.consumer.subscribe({ topic, fromBeginning: false });
174
+ this.logger.log(`Subscribed to topic: ${topic}`);
175
+ }
176
+ this.handlers.get(topic).push(handler);
177
+ }
178
+ async unsubscribe(topic) {
179
+ if (this.handlers.has(topic)) {
180
+ this.handlers.delete(topic);
181
+ this.logger.log(`Unsubscribed from topic: ${topic}`);
182
+ }
183
+ }
184
+ isConnected() {
185
+ return this.connected;
186
+ }
187
+ async createTopic(name, options) {
188
+ if (!this.admin) {
189
+ throw new Error('Kafka adapter not connected');
190
+ }
191
+ try {
192
+ await this.admin.createTopics({
193
+ topics: [
194
+ {
195
+ topic: name,
196
+ numPartitions: options?.numPartitions ?? 1,
197
+ replicationFactor: options?.replicationFactor ?? 1,
198
+ },
199
+ ],
200
+ });
201
+ this.logger.log(`Topic created: ${name}`);
202
+ }
203
+ catch (error) {
204
+ if (error.type !== 'TOPIC_ALREADY_EXISTS') {
205
+ throw error;
206
+ }
207
+ }
208
+ }
209
+ async handleMessage(payload) {
210
+ const { topic, message: kafkaMessage } = payload;
211
+ if (!kafkaMessage.value) {
212
+ return;
213
+ }
214
+ let content;
215
+ try {
216
+ content = JSON.parse(kafkaMessage.value.toString());
217
+ }
218
+ catch {
219
+ this.logger.error('Invalid JSON message; skipping');
220
+ return;
221
+ }
222
+ const eventTopic = content.__event || topic;
223
+ const handlers = this.handlers.get(topic) ?? [];
224
+ if (handlers.length === 0) {
225
+ return;
226
+ }
227
+ const meta = (0, envelope_1.ensureMeta)(content?.meta);
228
+ const brokerMessage = {
229
+ topic: eventTopic,
230
+ payload: content?.data ?? content,
231
+ meta,
232
+ replyTo: content.__replyTo && content.__correlationId
233
+ ? {
234
+ destination: String(content.__replyTo),
235
+ correlationId: String(content.__correlationId),
236
+ }
237
+ : undefined,
238
+ ack: async () => {
239
+ },
240
+ nack: async (_requeue = false) => {
241
+ this.logger.error(`Message processing failed for topic: ${topic}`);
242
+ },
243
+ };
244
+ try {
245
+ for (const handler of handlers) {
246
+ await handler(brokerMessage);
247
+ }
248
+ }
249
+ catch (error) {
250
+ this.logger.error(`Handler failed for topic="${topic}" (corr=${meta.correlationId}): ${error?.message || error}`);
251
+ throw error;
252
+ }
253
+ }
254
+ encodeHeaders(headers) {
255
+ const encoded = {};
256
+ for (const [key, value] of Object.entries(headers)) {
257
+ encoded[key] = Buffer.from(String(value));
258
+ }
259
+ return encoded;
260
+ }
261
+ }
262
+ exports.KafkaAdapter = KafkaAdapter;
263
+ //# sourceMappingURL=kafka.adapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"kafka.adapter.js","sourceRoot":"","sources":["../../../src/event-bus/adapters/kafka.adapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,39 @@
1
+ import type { EventMeta } from '../../envelope';
2
+ export interface BrokerConnectionConfig {
3
+ url?: string | string[];
4
+ serviceName?: string;
5
+ prefetch?: number;
6
+ options?: Record<string, any>;
7
+ }
8
+ export interface BrokerMessage {
9
+ topic: string;
10
+ payload: any;
11
+ meta?: EventMeta;
12
+ replyTo?: {
13
+ destination: string;
14
+ correlationId: string;
15
+ };
16
+ ack: () => Promise<void>;
17
+ nack: (requeue?: boolean) => Promise<void>;
18
+ }
19
+ export type MessageHandler = (message: BrokerMessage) => Promise<void>;
20
+ export interface MessageBrokerAdapter {
21
+ connect(config: BrokerConnectionConfig): Promise<void>;
22
+ disconnect(): Promise<void>;
23
+ publish(topic: string, payload: any, meta?: EventMeta, options?: PublishOptions): Promise<void>;
24
+ request<T = any>(topic: string, payload: any, meta?: EventMeta, timeoutMs?: number): Promise<T>;
25
+ reply(replyTo: string, correlationId: string, payload: any, meta?: EventMeta): Promise<void>;
26
+ subscribe(topic: string, handler: MessageHandler): Promise<void>;
27
+ unsubscribe(topic: string): Promise<void>;
28
+ isConnected(): boolean;
29
+ createExchange?(name: string, type?: string): Promise<void>;
30
+ }
31
+ export interface PublishOptions {
32
+ persistent?: boolean;
33
+ headers?: Record<string, any>;
34
+ delay?: number;
35
+ ttl?: number;
36
+ partition?: number;
37
+ key?: string;
38
+ }
39
+ //# sourceMappingURL=message-broker.interface.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"message-broker.interface.d.ts","sourceRoot":"","sources":["../../../src/event-bus/adapters/message-broker.interface.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAKhD,MAAM,WAAW,sBAAsB;IAErC,GAAG,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;IAExB,WAAW,CAAC,EAAE,MAAM,CAAC;IAErB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAElB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;CAC/B;AAKD,MAAM,WAAW,aAAa;IAE5B,KAAK,EAAE,MAAM,CAAC;IAEd,OAAO,EAAE,GAAG,CAAC;IAEb,IAAI,CAAC,EAAE,SAAS,CAAC;IAEjB,OAAO,CAAC,EAAE;QACR,WAAW,EAAE,MAAM,CAAC;QACpB,aAAa,EAAE,MAAM,CAAC;KACvB,CAAC;IAEF,GAAG,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAEzB,IAAI,EAAE,CAAC,OAAO,CAAC,EAAE,OAAO,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CAC5C;AAKD,MAAM,MAAM,cAAc,GAAG,CAAC,OAAO,EAAE,aAAa,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;AAKvE,MAAM,WAAW,oBAAoB;IAInC,OAAO,CAAC,MAAM,EAAE,sBAAsB,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAKvD,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;IAU5B,OAAO,CACL,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,EAChB,OAAO,CAAC,EAAE,cAAc,GACvB,OAAO,CAAC,IAAI,CAAC,CAAC;IAWjB,OAAO,CAAC,CAAC,GAAG,GAAG,EACb,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,EAChB,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,CAAC,CAAC,CAAC;IAUd,KAAK,CACH,OAAO,EAAE,MAAM,EACf,aAAa,EAAE,MAAM,EACrB,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,GACf,OAAO,CAAC,IAAI,CAAC,CAAC;IAQjB,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAOjE,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAK1C,WAAW,IAAI,OAAO,CAAC;IAQvB,cAAc,CAAC,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;CAC7D;AAKD,MAAM,WAAW,cAAc;IAE7B,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAE9B,KAAK,CAAC,EAAE,MAAM,CAAC;IAEf,GAAG,CAAC,EAAE,MAAM,CAAC;IAEb,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB,GAAG,CAAC,EAAE,MAAM,CAAC;CACd"}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=message-broker.interface.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"message-broker.interface.js","sourceRoot":"","sources":["../../../src/event-bus/adapters/message-broker.interface.ts"],"names":[],"mappings":""}
@@ -0,0 +1,21 @@
1
+ import type { MessageBrokerAdapter, BrokerConnectionConfig, MessageHandler, PublishOptions } from './message-broker.interface';
2
+ import type { EventMeta } from '../../envelope';
3
+ export declare class RabbitMQAdapter implements MessageBrokerAdapter {
4
+ private readonly logger;
5
+ private connection?;
6
+ private channel?;
7
+ private exchangeName;
8
+ private queueName;
9
+ private handlers;
10
+ connect(config: BrokerConnectionConfig): Promise<void>;
11
+ disconnect(): Promise<void>;
12
+ publish(topic: string, payload: any, meta?: EventMeta, options?: PublishOptions): Promise<void>;
13
+ request<T = any>(topic: string, payload: any, meta?: EventMeta, timeoutMs?: number): Promise<T>;
14
+ reply(replyTo: string, correlationId: string, payload: any, meta?: EventMeta): Promise<void>;
15
+ subscribe(topic: string, handler: MessageHandler): Promise<void>;
16
+ unsubscribe(topic: string): Promise<void>;
17
+ isConnected(): boolean;
18
+ createExchange(name: string, type?: string): Promise<void>;
19
+ private handleMessage;
20
+ }
21
+ //# sourceMappingURL=rabbitmq.adapter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rabbitmq.adapter.d.ts","sourceRoot":"","sources":["../../../src/event-bus/adapters/rabbitmq.adapter.ts"],"names":[],"mappings":"AAWA,OAAO,KAAK,EACV,oBAAoB,EACpB,sBAAsB,EAEtB,cAAc,EACd,cAAc,EACf,MAAM,4BAA4B,CAAC;AACpC,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AA4BhD,qBAAa,eAAgB,YAAW,oBAAoB;IAC1D,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAoC;IAE3D,OAAO,CAAC,UAAU,CAAC,CAAe;IAClC,OAAO,CAAC,OAAO,CAAC,CAAU;IAC1B,OAAO,CAAC,YAAY,CAAsB;IAC1C,OAAO,CAAC,SAAS,CAAmB;IACpC,OAAO,CAAC,QAAQ,CAAuC;IAKjD,OAAO,CAAC,MAAM,EAAE,sBAAsB,GAAG,OAAO,CAAC,IAAI,CAAC;IAiDtD,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB3B,OAAO,CACX,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,EAChB,OAAO,CAAC,EAAE,cAAc,GACvB,OAAO,CAAC,IAAI,CAAC;IAkCV,OAAO,CAAC,CAAC,GAAG,GAAG,EACnB,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,EAChB,SAAS,SAAO,GACf,OAAO,CAAC,CAAC,CAAC;IA4EP,KAAK,CACT,OAAO,EAAE,MAAM,EACf,aAAa,EAAE,MAAM,EACrB,OAAO,EAAE,GAAG,EACZ,IAAI,CAAC,EAAE,SAAS,GACf,OAAO,CAAC,IAAI,CAAC;IAsBV,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;IAoBhE,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAgB/C,WAAW,IAAI,OAAO;IAOhB,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,SAAU,GAAG,OAAO,CAAC,IAAI,CAAC;YAYnD,aAAa;CAyE5B"}
@@ -0,0 +1,232 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.RabbitMQAdapter = void 0;
4
+ const common_1 = require("@nestjs/common");
5
+ const amqplib_1 = require("amqplib");
6
+ const envelope_1 = require("../../envelope");
7
+ class RabbitMQAdapter {
8
+ logger = new common_1.Logger(RabbitMQAdapter.name);
9
+ connection;
10
+ channel;
11
+ exchangeName = 'anatix-event-bus';
12
+ queueName = 'default-queue';
13
+ handlers = new Map();
14
+ async connect(config) {
15
+ const url = Array.isArray(config.url) ? config.url[0] : config.url || 'amqp://localhost';
16
+ const prefetch = config.prefetch ?? 1;
17
+ const exchangeName = config.options?.exchange ?? 'anatix-event-bus';
18
+ const exchangeType = config.options?.exchangeType ?? 'topic';
19
+ const queueName = config.serviceName ?? 'default-queue';
20
+ this.exchangeName = exchangeName;
21
+ this.queueName = queueName;
22
+ try {
23
+ this.connection = await (0, amqplib_1.connect)(url);
24
+ this.channel = await this.connection.createChannel();
25
+ await this.channel.assertExchange(this.exchangeName, exchangeType, {
26
+ durable: true,
27
+ });
28
+ await this.channel.assertQueue(this.queueName, {
29
+ durable: true,
30
+ });
31
+ await this.channel.prefetch(prefetch);
32
+ await this.channel.consume(this.queueName, (msg) => this.handleMessage(msg), { noAck: false });
33
+ this.logger.log(`Connected to RabbitMQ: ${url} (exchange: ${this.exchangeName}, queue: ${this.queueName})`);
34
+ }
35
+ catch (error) {
36
+ this.logger.error(`Failed to connect to RabbitMQ: ${error?.message || error}`);
37
+ throw error;
38
+ }
39
+ }
40
+ async disconnect() {
41
+ try {
42
+ if (this.channel) {
43
+ await this.channel.close();
44
+ this.channel = undefined;
45
+ }
46
+ if (this.connection) {
47
+ await this.connection.close();
48
+ this.connection = undefined;
49
+ }
50
+ this.logger.log('Disconnected from RabbitMQ');
51
+ }
52
+ catch (error) {
53
+ this.logger.error(`Error disconnecting from RabbitMQ: ${error?.message || error}`);
54
+ }
55
+ }
56
+ async publish(topic, payload, meta, options) {
57
+ if (!this.channel) {
58
+ throw new Error('RabbitMQ adapter not connected');
59
+ }
60
+ const eventMeta = (0, envelope_1.ensureMeta)(meta);
61
+ const message = {
62
+ __event: topic,
63
+ data: payload,
64
+ meta: eventMeta,
65
+ };
66
+ const buffer = Buffer.from(JSON.stringify(message));
67
+ const publishOptions = {
68
+ contentType: 'application/json',
69
+ persistent: options?.persistent ?? true,
70
+ };
71
+ if (options?.headers) {
72
+ publishOptions.headers = options.headers;
73
+ }
74
+ if (options?.ttl) {
75
+ publishOptions.expiration = String(options.ttl);
76
+ }
77
+ this.channel.publish(this.exchangeName, topic, buffer, publishOptions);
78
+ }
79
+ async request(topic, payload, meta, timeoutMs = 5000) {
80
+ if (!this.channel) {
81
+ throw new Error('RabbitMQ adapter not connected');
82
+ }
83
+ const eventMeta = (0, envelope_1.ensureMeta)(meta);
84
+ const { queue: replyQueue } = await this.channel.assertQueue('', {
85
+ exclusive: true,
86
+ durable: false,
87
+ autoDelete: true,
88
+ });
89
+ return new Promise((resolve, reject) => {
90
+ let consumerTag = '';
91
+ const onReply = (msg) => {
92
+ if (!msg)
93
+ return;
94
+ try {
95
+ const content = JSON.parse(msg.content.toString());
96
+ if (content.__correlationId === eventMeta.correlationId) {
97
+ clearTimeout(timer);
98
+ this.channel?.cancel(consumerTag).catch(() => { });
99
+ resolve(content.data);
100
+ }
101
+ }
102
+ catch {
103
+ }
104
+ };
105
+ this.channel.consume(replyQueue, onReply, { noAck: true })
106
+ .then(({ consumerTag: tag }) => {
107
+ consumerTag = tag;
108
+ const message = {
109
+ __event: topic,
110
+ __replyTo: replyQueue,
111
+ __correlationId: eventMeta.correlationId,
112
+ data: payload,
113
+ meta: eventMeta,
114
+ };
115
+ this.channel.publish(this.exchangeName, topic, Buffer.from(JSON.stringify(message)), {
116
+ contentType: 'application/json',
117
+ persistent: true,
118
+ });
119
+ });
120
+ const timer = setTimeout(() => {
121
+ this.channel?.cancel(consumerTag).catch(() => { });
122
+ reject(new Error(`RPC timeout for topic "${topic}" (correlationId: ${eventMeta.correlationId})`));
123
+ }, timeoutMs);
124
+ });
125
+ }
126
+ async reply(replyTo, correlationId, payload, meta) {
127
+ if (!this.channel) {
128
+ throw new Error('RabbitMQ adapter not connected');
129
+ }
130
+ const eventMeta = (0, envelope_1.ensureMeta)(meta);
131
+ const message = {
132
+ __correlationId: correlationId,
133
+ data: payload,
134
+ meta: eventMeta,
135
+ };
136
+ this.channel.sendToQueue(replyTo, Buffer.from(JSON.stringify(message)), {
137
+ contentType: 'application/json',
138
+ persistent: false,
139
+ });
140
+ }
141
+ async subscribe(topic, handler) {
142
+ if (!this.channel) {
143
+ throw new Error('RabbitMQ adapter not connected');
144
+ }
145
+ if (!this.handlers.has(topic)) {
146
+ this.handlers.set(topic, []);
147
+ await this.channel.bindQueue(this.queueName, this.exchangeName, topic);
148
+ this.logger.log(`Subscribed to topic: ${topic}`);
149
+ }
150
+ this.handlers.get(topic).push(handler);
151
+ }
152
+ async unsubscribe(topic) {
153
+ if (!this.channel) {
154
+ throw new Error('RabbitMQ adapter not connected');
155
+ }
156
+ if (this.handlers.has(topic)) {
157
+ await this.channel.unbindQueue(this.queueName, this.exchangeName, topic);
158
+ this.handlers.delete(topic);
159
+ this.logger.log(`Unsubscribed from topic: ${topic}`);
160
+ }
161
+ }
162
+ isConnected() {
163
+ return !!this.channel && !!this.connection;
164
+ }
165
+ async createExchange(name, type = 'topic') {
166
+ if (!this.channel) {
167
+ throw new Error('RabbitMQ adapter not connected');
168
+ }
169
+ await this.channel.assertExchange(name, type, { durable: true });
170
+ this.logger.log(`Exchange created: ${name} (type: ${type})`);
171
+ }
172
+ async handleMessage(msg) {
173
+ if (!msg || !this.channel) {
174
+ return;
175
+ }
176
+ let content;
177
+ try {
178
+ content = JSON.parse(msg.content.toString());
179
+ }
180
+ catch {
181
+ this.logger.error('Invalid JSON message; acking to avoid poison loop');
182
+ this.channel.ack(msg);
183
+ return;
184
+ }
185
+ const topic = content.__event;
186
+ if (!topic || typeof topic !== 'string') {
187
+ this.logger.warn('Message without __event topic; acking');
188
+ this.channel.ack(msg);
189
+ return;
190
+ }
191
+ const handlers = this.handlers.get(topic) ?? [];
192
+ if (handlers.length === 0) {
193
+ this.channel.ack(msg);
194
+ return;
195
+ }
196
+ const meta = (0, envelope_1.ensureMeta)(content?.meta);
197
+ const brokerMessage = {
198
+ topic,
199
+ payload: content?.data ?? content,
200
+ meta,
201
+ replyTo: content.__replyTo && content.__correlationId
202
+ ? {
203
+ destination: String(content.__replyTo),
204
+ correlationId: String(content.__correlationId),
205
+ }
206
+ : undefined,
207
+ ack: async () => {
208
+ if (this.channel) {
209
+ this.channel.ack(msg);
210
+ }
211
+ },
212
+ nack: async (requeue = false) => {
213
+ if (this.channel) {
214
+ this.channel.nack(msg, false, requeue);
215
+ }
216
+ },
217
+ };
218
+ try {
219
+ for (const handler of handlers) {
220
+ await handler(brokerMessage);
221
+ }
222
+ }
223
+ catch (error) {
224
+ this.logger.error(`Handler failed for topic="${topic}" (corr=${meta.correlationId}): ${error?.message || error}`);
225
+ if (this.channel) {
226
+ this.channel.nack(msg, false, false);
227
+ }
228
+ }
229
+ }
230
+ }
231
+ exports.RabbitMQAdapter = RabbitMQAdapter;
232
+ //# sourceMappingURL=rabbitmq.adapter.js.map