219 lines
7.4 KiB
TypeScript
219 lines
7.4 KiB
TypeScript
import { describe, expect, it } from "vitest";
|
|
import { createManagerHarness, FakeProvider, markCallAnswered } from "./manager.test-harness.js";
|
|
|
|
describe("CallManager closed-loop turns", () => {
|
|
it("completes a closed-loop turn without live audio", async () => {
|
|
const { manager, provider } = await createManagerHarness({
|
|
transcriptTimeoutMs: 5000,
|
|
});
|
|
|
|
const started = await manager.initiateCall("+15550000003");
|
|
expect(started.success).toBe(true);
|
|
|
|
markCallAnswered(manager, started.callId, "evt-closed-loop-answered");
|
|
|
|
const turnPromise = manager.continueCall(started.callId, "How can I help?");
|
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
|
|
manager.processEvent({
|
|
id: "evt-closed-loop-speech",
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: "Please check status",
|
|
isFinal: true,
|
|
});
|
|
|
|
const turn = await turnPromise;
|
|
expect(turn.success).toBe(true);
|
|
expect(turn.transcript).toBe("Please check status");
|
|
expect(provider.startListeningCalls).toHaveLength(1);
|
|
expect(provider.stopListeningCalls).toHaveLength(1);
|
|
|
|
const call = manager.getCall(started.callId);
|
|
expect(call?.transcript.map((entry) => entry.text)).toEqual([
|
|
"How can I help?",
|
|
"Please check status",
|
|
]);
|
|
const metadata = (call?.metadata ?? {}) as Record<string, unknown>;
|
|
expect(typeof metadata.lastTurnLatencyMs).toBe("number");
|
|
expect(typeof metadata.lastTurnListenWaitMs).toBe("number");
|
|
expect(metadata.turnCount).toBe(1);
|
|
});
|
|
|
|
it("rejects overlapping continueCall requests for the same call", async () => {
|
|
const { manager, provider } = await createManagerHarness({
|
|
transcriptTimeoutMs: 5000,
|
|
});
|
|
|
|
const started = await manager.initiateCall("+15550000004");
|
|
expect(started.success).toBe(true);
|
|
|
|
markCallAnswered(manager, started.callId, "evt-overlap-answered");
|
|
|
|
const first = manager.continueCall(started.callId, "First prompt");
|
|
const second = await manager.continueCall(started.callId, "Second prompt");
|
|
expect(second.success).toBe(false);
|
|
expect(second.error).toBe("Already waiting for transcript");
|
|
|
|
manager.processEvent({
|
|
id: "evt-overlap-speech",
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: "Done",
|
|
isFinal: true,
|
|
});
|
|
|
|
const firstResult = await first;
|
|
expect(firstResult.success).toBe(true);
|
|
expect(firstResult.transcript).toBe("Done");
|
|
expect(provider.startListeningCalls).toHaveLength(1);
|
|
expect(provider.stopListeningCalls).toHaveLength(1);
|
|
});
|
|
|
|
it("ignores speech events with mismatched turnToken while waiting for transcript", async () => {
|
|
const { manager, provider } = await createManagerHarness(
|
|
{
|
|
transcriptTimeoutMs: 5000,
|
|
},
|
|
new FakeProvider("twilio"),
|
|
);
|
|
|
|
const started = await manager.initiateCall("+15550000004");
|
|
expect(started.success).toBe(true);
|
|
|
|
markCallAnswered(manager, started.callId, "evt-turn-token-answered");
|
|
|
|
const turnPromise = manager.continueCall(started.callId, "Prompt");
|
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
|
|
const expectedTurnToken = provider.startListeningCalls[0]?.turnToken;
|
|
expect(typeof expectedTurnToken).toBe("string");
|
|
|
|
manager.processEvent({
|
|
id: "evt-turn-token-bad",
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: "stale replay",
|
|
isFinal: true,
|
|
turnToken: "wrong-token",
|
|
});
|
|
|
|
const pendingState = await Promise.race([
|
|
turnPromise.then(() => "resolved"),
|
|
new Promise<"pending">((resolve) => setTimeout(() => resolve("pending"), 0)),
|
|
]);
|
|
expect(pendingState).toBe("pending");
|
|
|
|
manager.processEvent({
|
|
id: "evt-turn-token-good",
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: "final answer",
|
|
isFinal: true,
|
|
turnToken: expectedTurnToken,
|
|
});
|
|
|
|
const turnResult = await turnPromise;
|
|
expect(turnResult.success).toBe(true);
|
|
expect(turnResult.transcript).toBe("final answer");
|
|
|
|
const call = manager.getCall(started.callId);
|
|
expect(call?.transcript.map((entry) => entry.text)).toEqual(["Prompt", "final answer"]);
|
|
});
|
|
|
|
it("tracks latency metadata across multiple closed-loop turns", async () => {
|
|
const { manager, provider } = await createManagerHarness({
|
|
transcriptTimeoutMs: 5000,
|
|
});
|
|
|
|
const started = await manager.initiateCall("+15550000005");
|
|
expect(started.success).toBe(true);
|
|
|
|
markCallAnswered(manager, started.callId, "evt-multi-answered");
|
|
|
|
const firstTurn = manager.continueCall(started.callId, "First question");
|
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
manager.processEvent({
|
|
id: "evt-multi-speech-1",
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: "First answer",
|
|
isFinal: true,
|
|
});
|
|
await firstTurn;
|
|
|
|
const secondTurn = manager.continueCall(started.callId, "Second question");
|
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
manager.processEvent({
|
|
id: "evt-multi-speech-2",
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: "Second answer",
|
|
isFinal: true,
|
|
});
|
|
const secondResult = await secondTurn;
|
|
|
|
expect(secondResult.success).toBe(true);
|
|
|
|
const call = manager.getCall(started.callId);
|
|
expect(call?.transcript.map((entry) => entry.text)).toEqual([
|
|
"First question",
|
|
"First answer",
|
|
"Second question",
|
|
"Second answer",
|
|
]);
|
|
const metadata = (call?.metadata ?? {}) as Record<string, unknown>;
|
|
expect(metadata.turnCount).toBe(2);
|
|
expect(typeof metadata.lastTurnLatencyMs).toBe("number");
|
|
expect(typeof metadata.lastTurnListenWaitMs).toBe("number");
|
|
expect(provider.startListeningCalls).toHaveLength(2);
|
|
expect(provider.stopListeningCalls).toHaveLength(2);
|
|
});
|
|
|
|
it("handles repeated closed-loop turns without waiter churn", async () => {
|
|
const { manager, provider } = await createManagerHarness({
|
|
transcriptTimeoutMs: 5000,
|
|
});
|
|
|
|
const started = await manager.initiateCall("+15550000006");
|
|
expect(started.success).toBe(true);
|
|
|
|
markCallAnswered(manager, started.callId, "evt-loop-answered");
|
|
|
|
for (let i = 1; i <= 5; i++) {
|
|
const turnPromise = manager.continueCall(started.callId, `Prompt ${i}`);
|
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
manager.processEvent({
|
|
id: `evt-loop-speech-${i}`,
|
|
type: "call.speech",
|
|
callId: started.callId,
|
|
providerCallId: "request-uuid",
|
|
timestamp: Date.now(),
|
|
transcript: `Answer ${i}`,
|
|
isFinal: true,
|
|
});
|
|
const result = await turnPromise;
|
|
expect(result.success).toBe(true);
|
|
expect(result.transcript).toBe(`Answer ${i}`);
|
|
}
|
|
|
|
const call = manager.getCall(started.callId);
|
|
const metadata = (call?.metadata ?? {}) as Record<string, unknown>;
|
|
expect(metadata.turnCount).toBe(5);
|
|
expect(provider.startListeningCalls).toHaveLength(5);
|
|
expect(provider.stopListeningCalls).toHaveLength(5);
|
|
});
|
|
});
|