mirror of
https://dev.azure.com/globalhealthx/EMR/_git/helix-engage-server
synced 2026-04-12 02:18:18 +00:00
feat: add call events orchestrator with WebSocket gateway, wire Exotel → lookup → enrich → push flow
- CallEventsService orchestrates: Exotel webhook → lead lookup → AI enrichment → WebSocket push - CallEventsGateway (Socket.IO /call-events namespace) with agent room registration and disposition handling - EnrichedCallEvent/DispositionPayload types for frontend contract - Disposition flow: creates Call record, updates lead status, logs lead activity - Wired ExotelController to forward answered/ended events to CallEventsService - forwardRef used to resolve circular dependency between gateway and service Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
76
src/call-events/call-events.gateway.ts
Normal file
76
src/call-events/call-events.gateway.ts
Normal file
@@ -0,0 +1,76 @@
|
||||
import {
|
||||
WebSocketGateway,
|
||||
WebSocketServer,
|
||||
SubscribeMessage,
|
||||
MessageBody,
|
||||
ConnectedSocket,
|
||||
} from '@nestjs/websockets';
|
||||
import { Logger, Inject, forwardRef } from '@nestjs/common';
|
||||
import { Server, Socket } from 'socket.io';
|
||||
import type { EnrichedCallEvent, DispositionPayload } from './call-events.types';
|
||||
import { CallEventsService } from './call-events.service';
|
||||
|
||||
@WebSocketGateway({
|
||||
cors: {
|
||||
origin: process.env.CORS_ORIGIN ?? 'http://localhost:5173',
|
||||
credentials: true,
|
||||
},
|
||||
namespace: '/call-events',
|
||||
})
|
||||
export class CallEventsGateway {
|
||||
@WebSocketServer()
|
||||
server: Server;
|
||||
|
||||
private readonly logger = new Logger(CallEventsGateway.name);
|
||||
|
||||
constructor(
|
||||
@Inject(forwardRef(() => CallEventsService))
|
||||
private readonly callEventsService: CallEventsService,
|
||||
) {}
|
||||
|
||||
// Push enriched call event to a specific agent's room
|
||||
pushCallEvent(agentName: string, event: EnrichedCallEvent) {
|
||||
const room = `agent:${agentName}`;
|
||||
this.logger.log(`Pushing ${event.eventType} event to room ${room}`);
|
||||
this.server.to(room).emit('call:incoming', event);
|
||||
}
|
||||
|
||||
// Agent registers when they open the Call Desk page
|
||||
@SubscribeMessage('agent:register')
|
||||
handleAgentRegister(
|
||||
@ConnectedSocket() client: Socket,
|
||||
@MessageBody() agentName: string,
|
||||
) {
|
||||
const room = `agent:${agentName}`;
|
||||
client.join(room);
|
||||
this.logger.log(
|
||||
`Agent ${agentName} registered in room ${room} (socket: ${client.id})`,
|
||||
);
|
||||
client.emit('agent:registered', { agentName, room });
|
||||
}
|
||||
|
||||
// Agent sends disposition after a call
|
||||
@SubscribeMessage('call:disposition')
|
||||
async handleDisposition(
|
||||
@ConnectedSocket() client: Socket,
|
||||
@MessageBody() payload: DispositionPayload,
|
||||
) {
|
||||
this.logger.log(
|
||||
`Disposition received from ${payload.agentName}: ${payload.disposition}`,
|
||||
);
|
||||
await this.callEventsService.handleDisposition(payload);
|
||||
client.emit('call:disposition:ack', {
|
||||
status: 'saved',
|
||||
callSid: payload.callSid,
|
||||
});
|
||||
return payload;
|
||||
}
|
||||
|
||||
handleConnection(client: Socket) {
|
||||
this.logger.log(`Client connected: ${client.id}`);
|
||||
}
|
||||
|
||||
handleDisconnect(client: Socket) {
|
||||
this.logger.log(`Client disconnected: ${client.id}`);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user