mirror of
https://dev.azure.com/globalhealthx/EMR/_git/helix-engage-server
synced 2026-04-11 18:08:16 +00:00
feat: agent summary, AHT, and performance aggregation endpoint
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -202,6 +202,55 @@ export class OzonetelAgentController {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Get('performance')
|
||||||
|
async performance(@Query('date') date?: string) {
|
||||||
|
const targetDate = date ?? new Date().toISOString().split('T')[0];
|
||||||
|
this.logger.log(`Performance: date=${targetDate} agent=${this.defaultAgentId}`);
|
||||||
|
|
||||||
|
const [cdr, summary, aht] = await Promise.all([
|
||||||
|
this.ozonetelAgent.fetchCDR({ date: targetDate }),
|
||||||
|
this.ozonetelAgent.getAgentSummary(this.defaultAgentId, targetDate),
|
||||||
|
this.ozonetelAgent.getAHT(this.defaultAgentId),
|
||||||
|
]);
|
||||||
|
|
||||||
|
const totalCalls = cdr.length;
|
||||||
|
const inbound = cdr.filter((c: any) => c.Type === 'InBound').length;
|
||||||
|
const outbound = cdr.filter((c: any) => c.Type === 'Manual' || c.Type === 'Progressive').length;
|
||||||
|
const answered = cdr.filter((c: any) => c.Status === 'Answered').length;
|
||||||
|
const missed = cdr.filter((c: any) => c.Status === 'Unanswered' || c.Status === 'NotAnswered').length;
|
||||||
|
|
||||||
|
const talkTimes = cdr
|
||||||
|
.filter((c: any) => c.TalkTime && c.TalkTime !== '00:00:00')
|
||||||
|
.map((c: any) => {
|
||||||
|
const parts = c.TalkTime.split(':').map(Number);
|
||||||
|
return parts[0] * 3600 + parts[1] * 60 + parts[2];
|
||||||
|
});
|
||||||
|
const avgTalkTimeSec = talkTimes.length > 0
|
||||||
|
? Math.round(talkTimes.reduce((a: number, b: number) => a + b, 0) / talkTimes.length)
|
||||||
|
: 0;
|
||||||
|
|
||||||
|
const dispositions: Record<string, number> = {};
|
||||||
|
for (const c of cdr) {
|
||||||
|
const d = (c as any).Disposition || 'No Disposition';
|
||||||
|
dispositions[d] = (dispositions[d] ?? 0) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
const appointmentsBooked = cdr.filter((c: any) =>
|
||||||
|
c.Disposition?.toLowerCase().includes('appointment'),
|
||||||
|
).length;
|
||||||
|
|
||||||
|
return {
|
||||||
|
date: targetDate,
|
||||||
|
calls: { total: totalCalls, inbound, outbound, answered, missed },
|
||||||
|
avgTalkTimeSec,
|
||||||
|
avgHandlingTime: aht,
|
||||||
|
conversionRate: totalCalls > 0 ? Math.round((appointmentsBooked / totalCalls) * 100) : 0,
|
||||||
|
appointmentsBooked,
|
||||||
|
timeUtilization: summary,
|
||||||
|
dispositions,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
private mapToOzonetelDisposition(disposition: string): string {
|
private mapToOzonetelDisposition(disposition: string): string {
|
||||||
// Campaign only has 'General Enquiry' configured currently
|
// Campaign only has 'General Enquiry' configured currently
|
||||||
const map: Record<string, string> = {
|
const map: Record<string, string> = {
|
||||||
|
|||||||
@@ -352,6 +352,81 @@ export class OzonetelAgentService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getAgentSummary(agentId: string, date: string): Promise<{
|
||||||
|
totalLoginDuration: string;
|
||||||
|
totalBusyTime: string;
|
||||||
|
totalIdleTime: string;
|
||||||
|
totalPauseTime: string;
|
||||||
|
totalWrapupTime: string;
|
||||||
|
totalDialTime: string;
|
||||||
|
} | null> {
|
||||||
|
const url = `https://${this.apiDomain}/ca_reports/summaryReport`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const token = await this.getToken();
|
||||||
|
const response = await axios({
|
||||||
|
method: 'GET',
|
||||||
|
url,
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${token}`,
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
data: JSON.stringify({
|
||||||
|
userName: this.accountId,
|
||||||
|
agentId,
|
||||||
|
fromDate: `${date} 00:00:00`,
|
||||||
|
toDate: `${date} 23:59:59`,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = response.data;
|
||||||
|
if (data.status === 'success' && data.message) {
|
||||||
|
const record = Array.isArray(data.message) ? data.message[0] : data.message;
|
||||||
|
return {
|
||||||
|
totalLoginDuration: record.TotalLoginDuration ?? '00:00:00',
|
||||||
|
totalBusyTime: record.TotalBusyTime ?? '00:00:00',
|
||||||
|
totalIdleTime: record.TotalIdleTime ?? '00:00:00',
|
||||||
|
totalPauseTime: record.TotalPauseTime ?? '00:00:00',
|
||||||
|
totalWrapupTime: record.TotalWrapupTime ?? '00:00:00',
|
||||||
|
totalDialTime: record.TotalDialTime ?? '00:00:00',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
} catch (error: any) {
|
||||||
|
this.logger.error(`Agent summary failed: ${error.message}`);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async getAHT(agentId: string): Promise<string> {
|
||||||
|
const url = `https://${this.apiDomain}/ca_apis/aht`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const token = await this.getToken();
|
||||||
|
const response = await axios({
|
||||||
|
method: 'GET',
|
||||||
|
url,
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${token}`,
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
data: JSON.stringify({
|
||||||
|
userName: this.accountId,
|
||||||
|
agentId,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = response.data;
|
||||||
|
if (data.status === 'success') {
|
||||||
|
return data.AHT ?? '00:00:00';
|
||||||
|
}
|
||||||
|
return '00:00:00';
|
||||||
|
} catch (error: any) {
|
||||||
|
this.logger.error(`AHT failed: ${error.message}`);
|
||||||
|
return '00:00:00';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async logoutAgent(params: {
|
async logoutAgent(params: {
|
||||||
agentId: string;
|
agentId: string;
|
||||||
password: string;
|
password: string;
|
||||||
|
|||||||
Reference in New Issue
Block a user