update outgoing call duration every 30 seconds until the call is over

This commit is contained in:
m5r
2021-08-30 20:53:21 +08:00
parent 6a2e76857b
commit ab004235f6
8 changed files with 131 additions and 157 deletions

View File

@ -1,7 +1,8 @@
import { Queue } from "quirrel/blitz";
import type { CallInstance } from "twilio/lib/rest/api/v2010/account/call";
import db, { Direction, CallStatus } from "../../../../db";
import db from "../../../../db";
import { translateCallDirection, translateCallStatus } from "../../../../integrations/twilio";
type Payload = {
organizationId: string;
@ -25,8 +26,8 @@ const insertCallsQueue = Queue<Payload>("api/queue/insert-calls", async ({ calls
id: call.sid,
from: call.from,
to: call.to,
direction: translateDirection(call.direction),
status: translateStatus(call.status),
direction: translateCallDirection(call.direction),
status: translateCallStatus(call.status),
duration: call.duration,
createdAt: new Date(call.dateCreated),
}))
@ -36,34 +37,3 @@ const insertCallsQueue = Queue<Payload>("api/queue/insert-calls", async ({ calls
});
export default insertCallsQueue;
function translateDirection(direction: CallInstance["direction"]): Direction {
switch (direction) {
case "inbound":
return Direction.Inbound;
case "outbound":
default:
return Direction.Outbound;
}
}
function translateStatus(status: CallInstance["status"]): CallStatus {
switch (status) {
case "busy":
return CallStatus.Busy;
case "canceled":
return CallStatus.Canceled;
case "completed":
return CallStatus.Completed;
case "failed":
return CallStatus.Failed;
case "in-progress":
return CallStatus.InProgress;
case "no-answer":
return CallStatus.NoAnswer;
case "queued":
return CallStatus.Queued;
case "ringing":
return CallStatus.Ringing;
}
}

View File

@ -0,0 +1,27 @@
import { Queue } from "quirrel/blitz";
import db from "../../../../db";
import getTwilioClient, { translateCallStatus } from "../../../../integrations/twilio";
type Payload = {
organizationId: string;
callId: string;
};
const updateCallDurationQueue = Queue<Payload>("api/queue/update-call-duration", async ({ organizationId, callId }) => {
const organization = await db.organization.findFirst({ where: { id: organizationId } });
const twilioClient = getTwilioClient(organization);
const call = await twilioClient.calls.get(callId).fetch();
await db.phoneCall.update({
where: { id: callId },
data: { duration: call.duration, status: translateCallStatus(call.status) },
});
const callHasFinished = ["busy", "no-answer", "canceled", "failed"].includes(call.status);
if (!callHasFinished) {
await updateCallDurationQueue.enqueue({ organizationId, callId }, { delay: "30s" });
}
});
export default updateCallDurationQueue;

View File

@ -1,13 +1,11 @@
import type { BlitzApiRequest, BlitzApiResponse } from "blitz";
import { getConfig } from "blitz";
import twilio from "twilio";
import type { CallInstance } from "twilio/lib/rest/api/v2010/account/call";
import db, { CallStatus, Direction } from "../../../../db";
import db, { Direction } from "../../../../db";
import appLogger from "../../../../integrations/logger";
import { voiceUrl } from "../../../../integrations/twilio";
import { translateCallStatus, voiceUrl } from "../../../../integrations/twilio";
import updateCallDurationQueue from "../queue/update-call-duration";
const { serverRuntimeConfig } = getConfig();
const logger = appLogger.child({ route: "/api/webhook/call" });
type ApiError = {
@ -60,13 +58,21 @@ export default async function incomingCallHandler(req: BlitzApiRequest, res: Bli
id: req.body.CallSid,
from: phoneNumber.number,
to: req.body.To,
status: translateStatus(req.body.CallStatus),
status: translateCallStatus(req.body.CallStatus),
direction: Direction.Outbound,
duration: "", // TODO
duration: "0",
organizationId: phoneNumber.organization.id,
phoneNumberId: phoneNumber.id,
},
});
await updateCallDurationQueue.enqueue(
{
organizationId: phoneNumber.organization.id,
callId: req.body.CallSid,
},
{ delay: "30s" },
);
const twiml = new twilio.twiml.VoiceResponse();
const dial = twiml.dial({
answerOnBridge: true,
@ -129,24 +135,3 @@ const outgoingBody = {
From: "client:95267d60-3d35-4c36-9905-8543ecb4f174__673b461a-11ba-43a4-89d7-9e29403053d4",
To: "+33613370787",
};
function translateStatus(status: CallInstance["status"]): CallStatus {
switch (status) {
case "busy":
return CallStatus.Busy;
case "canceled":
return CallStatus.Canceled;
case "completed":
return CallStatus.Completed;
case "failed":
return CallStatus.Failed;
case "in-progress":
return CallStatus.InProgress;
case "no-answer":
return CallStatus.NoAnswer;
case "queued":
return CallStatus.Queued;
case "ringing":
return CallStatus.Ringing;
}
}