@boxitworld/commons-backend 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/classes/BacklogManager.d.ts +45 -0
- package/dist/classes/BacklogManager.d.ts.map +1 -0
- package/dist/classes/BacklogManager.js +193 -0
- package/dist/classes/BacklogManager.js.map +1 -0
- package/dist/classes/BatchEnhancedListener.d.ts +31 -0
- package/dist/classes/BatchEnhancedListener.d.ts.map +1 -0
- package/dist/classes/BatchEnhancedListener.js +168 -0
- package/dist/classes/BatchEnhancedListener.js.map +1 -0
- package/dist/classes/Publisher.d.ts +33 -0
- package/dist/classes/Publisher.d.ts.map +1 -0
- package/dist/classes/Publisher.js +171 -0
- package/dist/classes/Publisher.js.map +1 -0
- package/dist/config/kafka.config.d.ts +5 -0
- package/dist/config/kafka.config.d.ts.map +1 -0
- package/dist/config/kafka.config.js +40 -0
- package/dist/config/kafka.config.js.map +1 -0
- package/dist/interfaces/modules/boards.d.ts +27 -0
- package/dist/interfaces/modules/boards.d.ts.map +1 -0
- package/dist/interfaces/modules/boards.js +3 -0
- package/dist/interfaces/modules/boards.js.map +1 -0
- package/dist/interfaces/modules/company.d.ts +22 -0
- package/dist/interfaces/modules/company.d.ts.map +1 -0
- package/dist/interfaces/modules/company.js +3 -0
- package/dist/interfaces/modules/company.js.map +1 -0
- package/dist/interfaces/modules/customer.d.ts +29 -0
- package/dist/interfaces/modules/customer.d.ts.map +1 -0
- package/dist/interfaces/modules/customer.js +3 -0
- package/dist/interfaces/modules/customer.js.map +1 -0
- package/dist/interfaces/modules/device.d.ts +22 -0
- package/dist/interfaces/modules/device.d.ts.map +1 -0
- package/dist/interfaces/modules/device.js +3 -0
- package/dist/interfaces/modules/device.js.map +1 -0
- package/dist/interfaces/modules/door.d.ts +87 -0
- package/dist/interfaces/modules/door.d.ts.map +1 -0
- package/dist/interfaces/modules/door.js +3 -0
- package/dist/interfaces/modules/door.js.map +1 -0
- package/dist/interfaces/modules/package.d.ts +248 -0
- package/dist/interfaces/modules/package.d.ts.map +1 -0
- package/dist/interfaces/modules/package.js +3 -0
- package/dist/interfaces/modules/package.js.map +1 -0
- package/dist/interfaces/modules/platform.d.ts +50 -0
- package/dist/interfaces/modules/platform.d.ts.map +1 -0
- package/dist/interfaces/modules/platform.js +3 -0
- package/dist/interfaces/modules/platform.js.map +1 -0
- package/dist/interfaces/modules/setting.d.ts +61 -0
- package/dist/interfaces/modules/setting.d.ts.map +1 -0
- package/dist/interfaces/modules/setting.js +3 -0
- package/dist/interfaces/modules/setting.js.map +1 -0
- package/dist/interfaces/modules/sync.d.ts +27 -0
- package/dist/interfaces/modules/sync.d.ts.map +1 -0
- package/dist/interfaces/modules/sync.js +3 -0
- package/dist/interfaces/modules/sync.js.map +1 -0
- package/dist/interfaces/shared/batch.d.ts +35 -0
- package/dist/interfaces/shared/batch.d.ts.map +1 -0
- package/dist/interfaces/shared/batch.js +11 -0
- package/dist/interfaces/shared/batch.js.map +1 -0
- package/dist/interfaces/shared/lockerboard.d.ts +51 -0
- package/dist/interfaces/shared/lockerboard.d.ts.map +1 -0
- package/dist/interfaces/shared/lockerboard.js +3 -0
- package/dist/interfaces/shared/lockerboard.js.map +1 -0
- package/dist/interfaces/shared/logs.d.ts +50 -0
- package/dist/interfaces/shared/logs.d.ts.map +1 -0
- package/dist/interfaces/shared/logs.js +21 -0
- package/dist/interfaces/shared/logs.js.map +1 -0
- package/dist/interfaces/shared/rfid.d.ts +40 -0
- package/dist/interfaces/shared/rfid.d.ts.map +1 -0
- package/dist/interfaces/shared/rfid.js +3 -0
- package/dist/interfaces/shared/rfid.js.map +1 -0
- package/dist/interfaces/shared/system.d.ts +39 -0
- package/dist/interfaces/shared/system.d.ts.map +1 -0
- package/dist/interfaces/shared/system.js +3 -0
- package/dist/interfaces/shared/system.js.map +1 -0
- package/dist/topics/topics.d.ts +8 -0
- package/dist/topics/topics.d.ts.map +1 -0
- package/dist/topics/topics.js +12 -0
- package/dist/topics/topics.js.map +1 -0
- package/dist/types/actions.types.d.ts +79 -0
- package/dist/types/actions.types.d.ts.map +1 -0
- package/dist/types/actions.types.js +29 -0
- package/dist/types/actions.types.js.map +1 -0
- package/dist/types/kafka.types.d.ts +37 -0
- package/dist/types/kafka.types.d.ts.map +1 -0
- package/dist/types/kafka.types.js +3 -0
- package/dist/types/kafka.types.js.map +1 -0
- package/package.json +62 -0
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { BacklogConfig } from '../types/kafka.types';
|
|
2
|
+
interface MessageProcessor<T> {
|
|
3
|
+
(payload: T): Promise<void>;
|
|
4
|
+
}
|
|
5
|
+
export interface BacklogStats {
|
|
6
|
+
processed: number;
|
|
7
|
+
failed: number;
|
|
8
|
+
retries: number;
|
|
9
|
+
queueSize: number;
|
|
10
|
+
activeProcessing: number;
|
|
11
|
+
circuitBreakerOpen: boolean;
|
|
12
|
+
averageProcessingTime: number;
|
|
13
|
+
}
|
|
14
|
+
export declare class BacklogManager {
|
|
15
|
+
private config;
|
|
16
|
+
private queue;
|
|
17
|
+
private activeProcessing;
|
|
18
|
+
private stats;
|
|
19
|
+
private lastProcessTime;
|
|
20
|
+
private processingTimes;
|
|
21
|
+
private circuitBreakerOpen;
|
|
22
|
+
private circuitBreakerLastCheck;
|
|
23
|
+
private consecutiveFailures;
|
|
24
|
+
constructor(config?: Partial<BacklogConfig>);
|
|
25
|
+
processMessage<T>(payload: T, processor: MessageProcessor<T>, messageId: string): Promise<boolean>;
|
|
26
|
+
isAtMaxCapacity(): boolean;
|
|
27
|
+
getAvailableCapacity(): number;
|
|
28
|
+
waitForCapacity(requiredCapacity?: number, timeoutMs?: number): Promise<boolean>;
|
|
29
|
+
private addToQueue;
|
|
30
|
+
private executeMessage;
|
|
31
|
+
private startQueueProcessor;
|
|
32
|
+
private applyRateLimit;
|
|
33
|
+
private isCircuitBreakerOpen;
|
|
34
|
+
private recordSuccess;
|
|
35
|
+
private recordFailure;
|
|
36
|
+
private updateStats;
|
|
37
|
+
private sleep;
|
|
38
|
+
getStats(): BacklogStats;
|
|
39
|
+
resetStats(): void;
|
|
40
|
+
updateConfig(newConfig: Partial<BacklogConfig>): void;
|
|
41
|
+
forceCircuitBreakerOpen(): void;
|
|
42
|
+
forceCircuitBreakerClose(): void;
|
|
43
|
+
}
|
|
44
|
+
export {};
|
|
45
|
+
//# sourceMappingURL=BacklogManager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BacklogManager.d.ts","sourceRoot":"","sources":["../../src/classes/BacklogManager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAErD,UAAU,gBAAgB,CAAC,CAAC;IAC1B,CAAC,OAAO,EAAE,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;CAC7B;AAUD,MAAM,WAAW,YAAY;IAC3B,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,gBAAgB,EAAE,MAAM,CAAC;IACzB,kBAAkB,EAAE,OAAO,CAAC;IAC5B,qBAAqB,EAAE,MAAM,CAAC;CAC/B;AAaD,qBAAa,cAAc;IACzB,OAAO,CAAC,MAAM,CAA0B;IACxC,OAAO,CAAC,KAAK,CAAwB;IACrC,OAAO,CAAC,gBAAgB,CAAa;IACrC,OAAO,CAAC,KAAK,CAAe;IAC5B,OAAO,CAAC,eAAe,CAAsB;IAC7C,OAAO,CAAC,eAAe,CAAgB;IACvC,OAAO,CAAC,kBAAkB,CAAkB;IAC5C,OAAO,CAAC,uBAAuB,CAAsB;IACrD,OAAO,CAAC,mBAAmB,CAAa;gBAE5B,MAAM,GAAE,OAAO,CAAC,aAAa,CAAM;IA6BzC,cAAc,CAAC,CAAC,EACpB,OAAO,EAAE,CAAC,EACV,SAAS,EAAE,gBAAgB,CAAC,CAAC,CAAC,EAC9B,SAAS,EAAE,MAAM,GAChB,OAAO,CAAC,OAAO,CAAC;IAqBnB,eAAe,IAAI,OAAO;IAO1B,oBAAoB,IAAI,MAAM;IAOxB,eAAe,CAAC,gBAAgB,GAAE,MAAU,EAAE,SAAS,GAAE,MAAc,GAAG,OAAO,CAAC,OAAO,CAAC;IAgBhG,OAAO,CAAC,UAAU;YAcJ,cAAc;IAqD5B,OAAO,CAAC,mBAAmB;YA0Bb,cAAc;IAW5B,OAAO,CAAC,oBAAoB;IAe5B,OAAO,CAAC,aAAa;IAgBrB,OAAO,CAAC,aAAa;IAiBrB,OAAO,CAAC,WAAW;IAcnB,OAAO,CAAC,KAAK;IAOb,QAAQ,IAAI,YAAY;IAQxB,UAAU,IAAI,IAAI;IAgBlB,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,aAAa,CAAC,GAAG,IAAI;IAOrD,uBAAuB,IAAI,IAAI;IAK/B,wBAAwB,IAAI,IAAI;CAIjC"}
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.BacklogManager = void 0;
|
|
4
|
+
class BacklogManager {
|
|
5
|
+
constructor(config = {}) {
|
|
6
|
+
this.queue = [];
|
|
7
|
+
this.activeProcessing = 0;
|
|
8
|
+
this.lastProcessTime = Date.now();
|
|
9
|
+
this.processingTimes = [];
|
|
10
|
+
this.circuitBreakerOpen = false;
|
|
11
|
+
this.circuitBreakerLastCheck = Date.now();
|
|
12
|
+
this.consecutiveFailures = 0;
|
|
13
|
+
this.config = {
|
|
14
|
+
maxConcurrent: 5,
|
|
15
|
+
rateLimitMs: 100,
|
|
16
|
+
circuitBreakerThreshold: 10,
|
|
17
|
+
circuitBreakerResetTime: 30000,
|
|
18
|
+
maxRetries: 3,
|
|
19
|
+
backlogRecoveryBatchSize: 10,
|
|
20
|
+
enableBacklogRecovery: true,
|
|
21
|
+
...config
|
|
22
|
+
};
|
|
23
|
+
this.stats = {
|
|
24
|
+
processed: 0,
|
|
25
|
+
failed: 0,
|
|
26
|
+
retries: 0,
|
|
27
|
+
queueSize: 0,
|
|
28
|
+
activeProcessing: 0,
|
|
29
|
+
circuitBreakerOpen: false,
|
|
30
|
+
averageProcessingTime: 0
|
|
31
|
+
};
|
|
32
|
+
this.startQueueProcessor();
|
|
33
|
+
}
|
|
34
|
+
async processMessage(payload, processor, messageId) {
|
|
35
|
+
if (this.isCircuitBreakerOpen()) {
|
|
36
|
+
console.log(`🔴 Circuit breaker OPEN - rejecting message ${messageId}`);
|
|
37
|
+
return false;
|
|
38
|
+
}
|
|
39
|
+
if (this.isAtMaxCapacity()) {
|
|
40
|
+
console.log(`⏸️ Queue at max capacity - queueing message ${messageId}`);
|
|
41
|
+
this.addToQueue(payload, processor, messageId);
|
|
42
|
+
return true;
|
|
43
|
+
}
|
|
44
|
+
return await this.executeMessage(payload, processor, messageId, 0);
|
|
45
|
+
}
|
|
46
|
+
isAtMaxCapacity() {
|
|
47
|
+
return this.activeProcessing >= this.config.maxConcurrent;
|
|
48
|
+
}
|
|
49
|
+
getAvailableCapacity() {
|
|
50
|
+
return Math.max(0, this.config.maxConcurrent - this.activeProcessing);
|
|
51
|
+
}
|
|
52
|
+
async waitForCapacity(requiredCapacity = 1, timeoutMs = 10000) {
|
|
53
|
+
const startTime = Date.now();
|
|
54
|
+
while (this.getAvailableCapacity() < requiredCapacity) {
|
|
55
|
+
if (Date.now() - startTime > timeoutMs) {
|
|
56
|
+
return false;
|
|
57
|
+
}
|
|
58
|
+
await this.sleep(100);
|
|
59
|
+
}
|
|
60
|
+
return true;
|
|
61
|
+
}
|
|
62
|
+
addToQueue(payload, processor, messageId) {
|
|
63
|
+
this.queue.push({
|
|
64
|
+
id: messageId,
|
|
65
|
+
payload,
|
|
66
|
+
processor,
|
|
67
|
+
timestamp: Date.now(),
|
|
68
|
+
retries: 0
|
|
69
|
+
});
|
|
70
|
+
this.updateStats();
|
|
71
|
+
}
|
|
72
|
+
async executeMessage(payload, processor, messageId, retryCount) {
|
|
73
|
+
this.activeProcessing++;
|
|
74
|
+
this.updateStats();
|
|
75
|
+
const startTime = Date.now();
|
|
76
|
+
try {
|
|
77
|
+
await this.applyRateLimit();
|
|
78
|
+
await processor(payload);
|
|
79
|
+
const processingTime = Date.now() - startTime;
|
|
80
|
+
this.recordSuccess(processingTime);
|
|
81
|
+
console.log(`✅ Message ${messageId} processed in ${processingTime}ms`);
|
|
82
|
+
return true;
|
|
83
|
+
}
|
|
84
|
+
catch (error) {
|
|
85
|
+
console.error(`❌ Error processing message ${messageId}:`, error);
|
|
86
|
+
this.recordFailure();
|
|
87
|
+
if (retryCount < this.config.maxRetries) {
|
|
88
|
+
console.log(`🔄 Retrying message ${messageId} (attempt ${retryCount + 1}/${this.config.maxRetries})`);
|
|
89
|
+
const delay = Math.min(1000 * Math.pow(2, retryCount), 10000);
|
|
90
|
+
await this.sleep(delay);
|
|
91
|
+
this.stats.retries++;
|
|
92
|
+
return await this.executeMessage(payload, processor, messageId, retryCount + 1);
|
|
93
|
+
}
|
|
94
|
+
return false;
|
|
95
|
+
}
|
|
96
|
+
finally {
|
|
97
|
+
this.activeProcessing--;
|
|
98
|
+
this.updateStats();
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
startQueueProcessor() {
|
|
102
|
+
setInterval(async () => {
|
|
103
|
+
if (this.queue.length > 0 && !this.isAtMaxCapacity() && !this.isCircuitBreakerOpen()) {
|
|
104
|
+
const batchSize = Math.min(this.config.backlogRecoveryBatchSize, this.getAvailableCapacity(), this.queue.length);
|
|
105
|
+
for (let i = 0; i < batchSize; i++) {
|
|
106
|
+
const item = this.queue.shift();
|
|
107
|
+
if (item) {
|
|
108
|
+
this.executeMessage(item.payload, item.processor, item.id, item.retries)
|
|
109
|
+
.catch(error => {
|
|
110
|
+
console.error(`💥 Queue processor error for ${item.id}:`, error);
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
}, 500);
|
|
116
|
+
}
|
|
117
|
+
async applyRateLimit() {
|
|
118
|
+
const elapsed = Date.now() - this.lastProcessTime;
|
|
119
|
+
if (elapsed < this.config.rateLimitMs) {
|
|
120
|
+
await this.sleep(this.config.rateLimitMs - elapsed);
|
|
121
|
+
}
|
|
122
|
+
this.lastProcessTime = Date.now();
|
|
123
|
+
}
|
|
124
|
+
isCircuitBreakerOpen() {
|
|
125
|
+
if (this.circuitBreakerOpen &&
|
|
126
|
+
Date.now() - this.circuitBreakerLastCheck > this.config.circuitBreakerResetTime) {
|
|
127
|
+
console.log(`🟢 Circuit breaker reset attempt`);
|
|
128
|
+
this.circuitBreakerOpen = false;
|
|
129
|
+
this.consecutiveFailures = 0;
|
|
130
|
+
}
|
|
131
|
+
return this.circuitBreakerOpen;
|
|
132
|
+
}
|
|
133
|
+
recordSuccess(processingTime) {
|
|
134
|
+
this.stats.processed++;
|
|
135
|
+
this.consecutiveFailures = 0;
|
|
136
|
+
this.processingTimes.push(processingTime);
|
|
137
|
+
if (this.processingTimes.length > 100) {
|
|
138
|
+
this.processingTimes.shift();
|
|
139
|
+
}
|
|
140
|
+
this.updateStats();
|
|
141
|
+
}
|
|
142
|
+
recordFailure() {
|
|
143
|
+
this.stats.failed++;
|
|
144
|
+
this.consecutiveFailures++;
|
|
145
|
+
if (this.consecutiveFailures >= this.config.circuitBreakerThreshold) {
|
|
146
|
+
console.log(`🔴 Circuit breaker OPEN - ${this.consecutiveFailures} consecutive failures`);
|
|
147
|
+
this.circuitBreakerOpen = true;
|
|
148
|
+
this.circuitBreakerLastCheck = Date.now();
|
|
149
|
+
}
|
|
150
|
+
this.updateStats();
|
|
151
|
+
}
|
|
152
|
+
updateStats() {
|
|
153
|
+
this.stats.queueSize = this.queue.length;
|
|
154
|
+
this.stats.activeProcessing = this.activeProcessing;
|
|
155
|
+
this.stats.circuitBreakerOpen = this.circuitBreakerOpen;
|
|
156
|
+
if (this.processingTimes.length > 0) {
|
|
157
|
+
this.stats.averageProcessingTime =
|
|
158
|
+
this.processingTimes.reduce((a, b) => a + b, 0) / this.processingTimes.length;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
sleep(ms) {
|
|
162
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
163
|
+
}
|
|
164
|
+
getStats() {
|
|
165
|
+
this.updateStats();
|
|
166
|
+
return { ...this.stats };
|
|
167
|
+
}
|
|
168
|
+
resetStats() {
|
|
169
|
+
this.stats = {
|
|
170
|
+
processed: 0,
|
|
171
|
+
failed: 0,
|
|
172
|
+
retries: 0,
|
|
173
|
+
queueSize: this.queue.length,
|
|
174
|
+
activeProcessing: this.activeProcessing,
|
|
175
|
+
circuitBreakerOpen: this.circuitBreakerOpen,
|
|
176
|
+
averageProcessingTime: 0
|
|
177
|
+
};
|
|
178
|
+
this.processingTimes = [];
|
|
179
|
+
}
|
|
180
|
+
updateConfig(newConfig) {
|
|
181
|
+
this.config = { ...this.config, ...newConfig };
|
|
182
|
+
}
|
|
183
|
+
forceCircuitBreakerOpen() {
|
|
184
|
+
this.circuitBreakerOpen = true;
|
|
185
|
+
this.circuitBreakerLastCheck = Date.now();
|
|
186
|
+
}
|
|
187
|
+
forceCircuitBreakerClose() {
|
|
188
|
+
this.circuitBreakerOpen = false;
|
|
189
|
+
this.consecutiveFailures = 0;
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
exports.BacklogManager = BacklogManager;
|
|
193
|
+
//# sourceMappingURL=BacklogManager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BacklogManager.js","sourceRoot":"","sources":["../../src/classes/BacklogManager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { KafkaMessage, Kafka, Consumer } from "kafkajs";
|
|
2
|
+
import { BaseEvent, BacklogConfig, BatchConfig, KafkaStats } from '../types/kafka.types';
|
|
3
|
+
import { BacklogManager } from './BacklogManager';
|
|
4
|
+
export interface MonitorableListener {
|
|
5
|
+
getStats(): KafkaStats;
|
|
6
|
+
resetStats(): void;
|
|
7
|
+
updateBacklogConfig(config: Partial<BacklogConfig>): void;
|
|
8
|
+
forceCircuitBreaker(open: boolean): void;
|
|
9
|
+
}
|
|
10
|
+
export declare abstract class BatchEnhancedListener<T extends BaseEvent> implements MonitorableListener {
|
|
11
|
+
abstract topic: T['topic'];
|
|
12
|
+
abstract groupName: string;
|
|
13
|
+
protected kafka: Kafka;
|
|
14
|
+
protected consumer: Consumer | null;
|
|
15
|
+
protected backlogManager: BacklogManager;
|
|
16
|
+
protected listenerName: string;
|
|
17
|
+
protected batchConfig: Required<BatchConfig>;
|
|
18
|
+
protected isRunning: boolean;
|
|
19
|
+
abstract processMessage(data: T['data'], msg: KafkaMessage): Promise<void>;
|
|
20
|
+
constructor(kafka: Kafka, listenerName: string, backlogConfig?: Partial<BacklogConfig>, batchConfig?: Partial<BatchConfig>);
|
|
21
|
+
protected getDefaultBacklogConfig(): BacklogConfig;
|
|
22
|
+
listen(fromBeginning?: boolean): Promise<void>;
|
|
23
|
+
private processBatch;
|
|
24
|
+
private parseMessage;
|
|
25
|
+
stop(): Promise<void>;
|
|
26
|
+
getStats(): KafkaStats;
|
|
27
|
+
updateBacklogConfig(newConfig: Partial<BacklogConfig>): void;
|
|
28
|
+
resetStats(): void;
|
|
29
|
+
forceCircuitBreaker(open: boolean): void;
|
|
30
|
+
}
|
|
31
|
+
//# sourceMappingURL=BatchEnhancedListener.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BatchEnhancedListener.d.ts","sourceRoot":"","sources":["../../src/classes/BatchEnhancedListener.ts"],"names":[],"mappings":"AAAA,OAAO,EAAoB,YAAY,EAAE,KAAK,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAC1E,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AACzF,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAElD,MAAM,WAAW,mBAAmB;IAClC,QAAQ,IAAI,UAAU,CAAC;IACvB,UAAU,IAAI,IAAI,CAAC;IACnB,mBAAmB,CAAC,MAAM,EAAE,OAAO,CAAC,aAAa,CAAC,GAAG,IAAI,CAAC;IAC1D,mBAAmB,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAAC;CAC1C;AAED,8BAAsB,qBAAqB,CAAC,CAAC,SAAS,SAAS,CAAE,YAAW,mBAAmB;IAE3F,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC;IAC3B,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAE3B,SAAS,CAAC,KAAK,EAAE,KAAK,CAAC;IACvB,SAAS,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI,CAAQ;IAC3C,SAAS,CAAC,cAAc,EAAE,cAAc,CAAC;IACzC,SAAS,CAAC,YAAY,EAAE,MAAM,CAAC;IAC/B,SAAS,CAAC,WAAW,EAAE,QAAQ,CAAC,WAAW,CAAC,CAAC;IAC7C,SAAS,CAAC,SAAS,EAAE,OAAO,CAAS;IAErC,QAAQ,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,GAAG,EAAE,YAAY,GAAG,OAAO,CAAC,IAAI,CAAC;gBAGtE,KAAK,EAAE,KAAK,EACZ,YAAY,EAAE,MAAM,EACpB,aAAa,CAAC,EAAE,OAAO,CAAC,aAAa,CAAC,EACtC,WAAW,CAAC,EAAE,OAAO,CAAC,WAAW,CAAC;IAoBtC,SAAS,CAAC,uBAAuB,IAAI,aAAa;IAY5C,MAAM,CAAC,aAAa,GAAE,OAAe,GAAG,OAAO,CAAC,IAAI,CAAC;YAqC7C,YAAY;IAsD1B,OAAO,CAAC,YAAY;IAqBd,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAiB3B,QAAQ,IAAI,UAAU;IAWtB,mBAAmB,CAAC,SAAS,EAAE,OAAO,CAAC,aAAa,CAAC,GAAG,IAAI;IAK5D,UAAU,IAAI,IAAI;IAKlB,mBAAmB,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI;CAS3C"}
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.BatchEnhancedListener = void 0;
|
|
4
|
+
const BacklogManager_1 = require("./BacklogManager");
|
|
5
|
+
class BatchEnhancedListener {
|
|
6
|
+
constructor(kafka, listenerName, backlogConfig, batchConfig) {
|
|
7
|
+
this.consumer = null;
|
|
8
|
+
this.isRunning = false;
|
|
9
|
+
this.kafka = kafka;
|
|
10
|
+
this.listenerName = listenerName;
|
|
11
|
+
const defaultBacklogConfig = this.getDefaultBacklogConfig();
|
|
12
|
+
this.backlogManager = new BacklogManager_1.BacklogManager({
|
|
13
|
+
...defaultBacklogConfig,
|
|
14
|
+
...backlogConfig
|
|
15
|
+
});
|
|
16
|
+
this.batchConfig = {
|
|
17
|
+
maxBytesPerPartition: 10240,
|
|
18
|
+
maxBytes: 102400,
|
|
19
|
+
...batchConfig
|
|
20
|
+
};
|
|
21
|
+
console.log(`🚀 Batch Enhanced listener ${listenerName} initialized with flow control`);
|
|
22
|
+
}
|
|
23
|
+
getDefaultBacklogConfig() {
|
|
24
|
+
return {
|
|
25
|
+
maxConcurrent: 3,
|
|
26
|
+
rateLimitMs: 200,
|
|
27
|
+
circuitBreakerThreshold: 5,
|
|
28
|
+
circuitBreakerResetTime: 30000,
|
|
29
|
+
maxRetries: 2,
|
|
30
|
+
backlogRecoveryBatchSize: 25,
|
|
31
|
+
enableBacklogRecovery: true
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
async listen(fromBeginning = false) {
|
|
35
|
+
if (this.isRunning) {
|
|
36
|
+
console.log(`⚠️ Listener ${this.listenerName} already running`);
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
console.log(`🎧 Starting batch enhanced listener: ${this.listenerName}`);
|
|
40
|
+
try {
|
|
41
|
+
this.consumer = this.kafka.consumer({
|
|
42
|
+
groupId: this.groupName,
|
|
43
|
+
sessionTimeout: 30000,
|
|
44
|
+
heartbeatInterval: 3000,
|
|
45
|
+
maxWaitTimeInMs: 5000,
|
|
46
|
+
maxBytesPerPartition: this.batchConfig.maxBytesPerPartition,
|
|
47
|
+
maxBytes: this.batchConfig.maxBytes
|
|
48
|
+
});
|
|
49
|
+
await this.consumer.connect();
|
|
50
|
+
await this.consumer.subscribe({ topic: this.topic, fromBeginning });
|
|
51
|
+
this.isRunning = true;
|
|
52
|
+
await this.consumer.run({
|
|
53
|
+
eachBatchAutoResolve: false,
|
|
54
|
+
eachBatch: async (payload) => {
|
|
55
|
+
await this.processBatch(payload);
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
catch (error) {
|
|
60
|
+
console.error(`💥 KAFKA ERROR in ${this.listenerName}:`, error);
|
|
61
|
+
this.isRunning = false;
|
|
62
|
+
throw error;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
async processBatch(payload) {
|
|
66
|
+
const { batch, resolveOffset, heartbeat, isRunning, isStale } = payload;
|
|
67
|
+
console.log(`📦 Processing batch: ${batch.messages.length} messages for ${this.listenerName}`);
|
|
68
|
+
for (const message of batch.messages) {
|
|
69
|
+
while (this.backlogManager.isAtMaxCapacity()) {
|
|
70
|
+
console.log(`⏸️ ${this.listenerName} pausing batch - at max capacity`);
|
|
71
|
+
const hasCapacity = await this.backlogManager.waitForCapacity(1, 10000);
|
|
72
|
+
if (!hasCapacity) {
|
|
73
|
+
console.error(`❌ ${this.listenerName} timeout waiting for capacity`);
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
await heartbeat();
|
|
77
|
+
if (!isRunning() || isStale()) {
|
|
78
|
+
console.warn(`⚠️ ${this.listenerName} consumer not running or stale`);
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
try {
|
|
83
|
+
const parsedData = this.parseMessage(message);
|
|
84
|
+
const success = await this.backlogManager.processMessage({ data: parsedData, msg: message }, async ({ data, msg }) => {
|
|
85
|
+
await this.processMessage(data, msg);
|
|
86
|
+
}, `${this.listenerName}:${this.topic}:${message.offset}`);
|
|
87
|
+
if (success) {
|
|
88
|
+
resolveOffset(message.offset);
|
|
89
|
+
console.log(`✅ ${this.listenerName} processed message offset ${message.offset}`);
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
console.error(`❌ ${this.listenerName} failed to process message offset ${message.offset}`);
|
|
93
|
+
break;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
catch (error) {
|
|
97
|
+
console.error(`💥 ${this.listenerName} critical error processing message:`, error);
|
|
98
|
+
break;
|
|
99
|
+
}
|
|
100
|
+
await heartbeat();
|
|
101
|
+
}
|
|
102
|
+
console.log(`📦 Batch completed for ${this.listenerName}`);
|
|
103
|
+
}
|
|
104
|
+
parseMessage(msg) {
|
|
105
|
+
const data = msg.value;
|
|
106
|
+
if (!data) {
|
|
107
|
+
throw new Error('Message value is null or undefined');
|
|
108
|
+
}
|
|
109
|
+
try {
|
|
110
|
+
const dataString = typeof data === 'string' ? data : data.toString('utf8');
|
|
111
|
+
const parsed = JSON.parse(dataString);
|
|
112
|
+
if (parsed.data && parsed.messageId && parsed.topic) {
|
|
113
|
+
return parsed.data;
|
|
114
|
+
}
|
|
115
|
+
return parsed;
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
console.error('Failed to parse message:', error);
|
|
119
|
+
throw new Error('Invalid JSON in message');
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
async stop() {
|
|
123
|
+
console.log(`🛑 Stopping listener: ${this.listenerName}`);
|
|
124
|
+
this.isRunning = false;
|
|
125
|
+
if (this.consumer) {
|
|
126
|
+
try {
|
|
127
|
+
await this.consumer.disconnect();
|
|
128
|
+
console.log(`🔌 Consumer disconnected for ${this.listenerName}`);
|
|
129
|
+
}
|
|
130
|
+
catch (error) {
|
|
131
|
+
console.error(`💥 Error disconnecting consumer for ${this.listenerName}:`, error);
|
|
132
|
+
}
|
|
133
|
+
finally {
|
|
134
|
+
this.consumer = null;
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
getStats() {
|
|
139
|
+
return {
|
|
140
|
+
listenerName: this.listenerName,
|
|
141
|
+
topic: this.topic,
|
|
142
|
+
groupName: this.groupName,
|
|
143
|
+
...this.backlogManager.getStats(),
|
|
144
|
+
availableCapacity: this.backlogManager.getAvailableCapacity(),
|
|
145
|
+
isAtMaxCapacity: this.backlogManager.isAtMaxCapacity()
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
updateBacklogConfig(newConfig) {
|
|
149
|
+
this.backlogManager.updateConfig(newConfig);
|
|
150
|
+
console.log(`🔧 ${this.listenerName} batch backlog config updated`);
|
|
151
|
+
}
|
|
152
|
+
resetStats() {
|
|
153
|
+
this.backlogManager.resetStats();
|
|
154
|
+
console.log(`🔄 ${this.listenerName} batch stats reset`);
|
|
155
|
+
}
|
|
156
|
+
forceCircuitBreaker(open) {
|
|
157
|
+
if (open) {
|
|
158
|
+
this.backlogManager.forceCircuitBreakerOpen();
|
|
159
|
+
console.log(`🔴 ${this.listenerName} batch circuit breaker forced OPEN`);
|
|
160
|
+
}
|
|
161
|
+
else {
|
|
162
|
+
this.backlogManager.forceCircuitBreakerClose();
|
|
163
|
+
console.log(`🟢 ${this.listenerName} batch circuit breaker forced CLOSED`);
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
exports.BatchEnhancedListener = BatchEnhancedListener;
|
|
168
|
+
//# sourceMappingURL=BatchEnhancedListener.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BatchEnhancedListener.js","sourceRoot":"","sources":["../../src/classes/BatchEnhancedListener.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { Kafka } from 'kafkajs';
|
|
2
|
+
import { BaseEvent, TypedEvent } from '../types/kafka.types';
|
|
3
|
+
export declare abstract class Publisher<T extends BaseEvent> {
|
|
4
|
+
abstract topic: T['topic'];
|
|
5
|
+
private kafka;
|
|
6
|
+
private producer;
|
|
7
|
+
private isConnected;
|
|
8
|
+
private publishStats;
|
|
9
|
+
constructor(kafka: Kafka);
|
|
10
|
+
private ensureConnection;
|
|
11
|
+
publish(data: T['data'], keyKafka: T['keyKafka'], headers?: Record<string, string>): Promise<string>;
|
|
12
|
+
publishBatch(messages: Array<{
|
|
13
|
+
data: T['data'];
|
|
14
|
+
keyKafka: T['keyKafka'];
|
|
15
|
+
headers?: Record<string, string>;
|
|
16
|
+
}>): Promise<string[]>;
|
|
17
|
+
disconnect(): Promise<void>;
|
|
18
|
+
getPublishStats(): {
|
|
19
|
+
topic: T["topic"];
|
|
20
|
+
successful: number;
|
|
21
|
+
failed: number;
|
|
22
|
+
total: number;
|
|
23
|
+
successRate: string;
|
|
24
|
+
lastPublish: string;
|
|
25
|
+
isConnected: boolean;
|
|
26
|
+
};
|
|
27
|
+
resetStats(): void;
|
|
28
|
+
}
|
|
29
|
+
export declare class GenericPublisher extends Publisher<TypedEvent> {
|
|
30
|
+
topic: string;
|
|
31
|
+
constructor(kafka: Kafka, topic: string);
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=Publisher.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Publisher.d.ts","sourceRoot":"","sources":["../../src/classes/Publisher.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAY,MAAM,SAAS,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAY7D,8BAAsB,SAAS,CAAC,CAAC,SAAS,SAAS;IACjD,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC;IAE3B,OAAO,CAAC,KAAK,CAAQ;IACrB,OAAO,CAAC,QAAQ,CAAyB;IACzC,OAAO,CAAC,WAAW,CAAkB;IACrC,OAAO,CAAC,YAAY,CAIlB;gBAEU,KAAK,EAAE,KAAK;YAOV,gBAAgB;IAwBxB,OAAO,CACX,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,EACf,QAAQ,EAAE,CAAC,CAAC,UAAU,CAAC,EACvB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAC/B,OAAO,CAAC,MAAM,CAAC;IAuEZ,YAAY,CAChB,QAAQ,EAAE,KAAK,CAAC;QACd,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC;QAChB,QAAQ,EAAE,CAAC,CAAC,UAAU,CAAC,CAAC;QACxB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;KAClC,CAAC,GACD,OAAO,CAAC,MAAM,EAAE,CAAC;IAqDd,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAiBjC,eAAe;;;;;;;;;IAiBf,UAAU,IAAI,IAAI;CAOnB;AAMD,qBAAa,gBAAiB,SAAQ,SAAS,CAAC,UAAU,CAAC;IACzD,KAAK,EAAE,MAAM,CAAC;gBAEF,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM;CAIxC"}
|