forked from nathydre21/nepa
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathAsyncEventProcessor.ts
More file actions
59 lines (50 loc) · 1.6 KB
/
AsyncEventProcessor.ts
File metadata and controls
59 lines (50 loc) · 1.6 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
import { DomainEvent } from './EventBus';
import MessageBroker from './MessageBroker';
interface ProcessorConfig {
maxRetries: number;
retryDelay: number;
deadLetterQueue: string;
}
class AsyncEventProcessor {
private config: ProcessorConfig;
constructor(config: Partial<ProcessorConfig> = {}) {
this.config = {
maxRetries: config.maxRetries || 3,
retryDelay: config.retryDelay || 1000,
deadLetterQueue: config.deadLetterQueue || 'dlq',
};
}
async process(
event: DomainEvent,
handler: (event: DomainEvent) => Promise<void>,
attempt = 0
): Promise<void> {
try {
await handler(event);
} catch (error) {
if (attempt < this.config.maxRetries) {
console.log(`⚠️ Retry ${attempt + 1}/${this.config.maxRetries} for ${event.eventType}`);
await this.delay(this.config.retryDelay * Math.pow(2, attempt));
return this.process(event, handler, attempt + 1);
}
console.error(`❌ Failed after ${this.config.maxRetries} retries:`, error);
await this.sendToDeadLetterQueue(event, error);
}
}
private async sendToDeadLetterQueue(event: DomainEvent, error: any): Promise<void> {
const dlqEvent = {
...event,
metadata: {
...event.metadata,
error: error.message,
failedAt: new Date().toISOString(),
},
};
console.log(`📮 Sending to DLQ: ${event.eventType}`);
// Store in DLQ for manual processing
}
private delay(ms: number): Promise<void> {
return new Promise(resolve => setTimeout(resolve, ms));
}
}
export default AsyncEventProcessor;