@kodelyth/voice-call 2026.5.42 → 2026.6.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +18 -6
- package/api.ts +0 -16
- package/cli-metadata.ts +0 -10
- package/config-api.ts +0 -12
- package/index.test.ts +0 -1075
- package/index.ts +0 -863
- package/runtime-api.ts +0 -20
- package/runtime-entry.ts +0 -1
- package/setup-api.ts +0 -47
- package/src/allowlist.test.ts +0 -18
- package/src/allowlist.ts +0 -19
- package/src/cli.test.ts +0 -12
- package/src/cli.ts +0 -866
- package/src/config-compat.test.ts +0 -130
- package/src/config-compat.ts +0 -227
- package/src/config.test.ts +0 -542
- package/src/config.ts +0 -883
- package/src/core-bridge.ts +0 -14
- package/src/deep-merge.test.ts +0 -40
- package/src/deep-merge.ts +0 -23
- package/src/gateway-continue-operation.ts +0 -200
- package/src/http-headers.test.ts +0 -16
- package/src/http-headers.ts +0 -15
- package/src/manager/context.ts +0 -50
- package/src/manager/events.test.ts +0 -578
- package/src/manager/events.ts +0 -332
- package/src/manager/lifecycle.ts +0 -53
- package/src/manager/lookup.test.ts +0 -52
- package/src/manager/lookup.ts +0 -35
- package/src/manager/outbound.test.ts +0 -629
- package/src/manager/outbound.ts +0 -508
- package/src/manager/state.ts +0 -48
- package/src/manager/store.ts +0 -107
- package/src/manager/timers.test.ts +0 -127
- package/src/manager/timers.ts +0 -113
- package/src/manager/twiml.test.ts +0 -13
- package/src/manager/twiml.ts +0 -17
- package/src/manager.closed-loop.test.ts +0 -259
- package/src/manager.inbound-allowlist.test.ts +0 -183
- package/src/manager.notify.test.ts +0 -390
- package/src/manager.restore.test.ts +0 -310
- package/src/manager.test-harness.ts +0 -127
- package/src/manager.ts +0 -441
- package/src/media-stream.test.ts +0 -953
- package/src/media-stream.ts +0 -876
- package/src/providers/base.ts +0 -99
- package/src/providers/mock.test.ts +0 -86
- package/src/providers/mock.ts +0 -185
- package/src/providers/plivo.test.ts +0 -93
- package/src/providers/plivo.ts +0 -601
- package/src/providers/shared/call-status.test.ts +0 -24
- package/src/providers/shared/call-status.ts +0 -24
- package/src/providers/shared/guarded-json-api.test.ts +0 -127
- package/src/providers/shared/guarded-json-api.ts +0 -49
- package/src/providers/telnyx.test.ts +0 -489
- package/src/providers/telnyx.ts +0 -419
- package/src/providers/twilio/api.test.ts +0 -184
- package/src/providers/twilio/api.ts +0 -100
- package/src/providers/twilio/twiml-policy.test.ts +0 -84
- package/src/providers/twilio/twiml-policy.ts +0 -87
- package/src/providers/twilio/webhook.ts +0 -34
- package/src/providers/twilio.test.ts +0 -607
- package/src/providers/twilio.ts +0 -861
- package/src/providers/twilio.types.ts +0 -17
- package/src/realtime-agent-context.test.ts +0 -101
- package/src/realtime-agent-context.ts +0 -149
- package/src/realtime-defaults.ts +0 -3
- package/src/realtime-fast-context.test.ts +0 -74
- package/src/realtime-fast-context.ts +0 -27
- package/src/realtime-transcription.runtime.ts +0 -4
- package/src/realtime-voice.runtime.ts +0 -5
- package/src/response-generator.test.ts +0 -385
- package/src/response-generator.ts +0 -348
- package/src/response-model.test.ts +0 -71
- package/src/response-model.ts +0 -23
- package/src/runtime.test.ts +0 -625
- package/src/runtime.ts +0 -528
- package/src/telephony-audio.test.ts +0 -61
- package/src/telephony-audio.ts +0 -12
- package/src/telephony-tts.test.ts +0 -196
- package/src/telephony-tts.ts +0 -235
- package/src/test-fixtures.ts +0 -82
- package/src/tts-provider-voice.test.ts +0 -34
- package/src/tts-provider-voice.ts +0 -21
- package/src/tunnel.test.ts +0 -173
- package/src/tunnel.ts +0 -314
- package/src/types.ts +0 -311
- package/src/utils.test.ts +0 -17
- package/src/utils.ts +0 -14
- package/src/voice-mapping.test.ts +0 -32
- package/src/voice-mapping.ts +0 -65
- package/src/webhook/realtime-audio-pacer.test.ts +0 -146
- package/src/webhook/realtime-audio-pacer.ts +0 -204
- package/src/webhook/realtime-handler.test.ts +0 -1450
- package/src/webhook/realtime-handler.ts +0 -1382
- package/src/webhook/stale-call-reaper.test.ts +0 -89
- package/src/webhook/stale-call-reaper.ts +0 -38
- package/src/webhook/stream-frame-adapter.test.ts +0 -187
- package/src/webhook/stream-frame-adapter.ts +0 -219
- package/src/webhook/tailscale.test.ts +0 -216
- package/src/webhook/tailscale.ts +0 -129
- package/src/webhook-exposure.test.ts +0 -33
- package/src/webhook-exposure.ts +0 -84
- package/src/webhook-security.test.ts +0 -813
- package/src/webhook-security.ts +0 -982
- package/src/webhook.hangup-once.lifecycle.test.ts +0 -179
- package/src/webhook.test.ts +0 -1615
- package/src/webhook.ts +0 -933
- package/src/webhook.types.ts +0 -5
- package/src/websocket-test-support.ts +0 -72
- package/tsconfig.json +0 -16
|
@@ -1,179 +0,0 @@
|
|
|
1
|
-
import { afterEach, describe, expect, it } from "vitest";
|
|
2
|
-
import { VoiceCallConfigSchema, type VoiceCallConfig } from "./config.js";
|
|
3
|
-
import { CallManager } from "./manager.js";
|
|
4
|
-
import { createTestStorePath, FakeProvider } from "./manager.test-harness.js";
|
|
5
|
-
import { flushPendingCallRecordWritesForTest } from "./manager/store.js";
|
|
6
|
-
import type { WebhookContext, WebhookParseOptions } from "./types.js";
|
|
7
|
-
import { VoiceCallWebhookServer } from "./webhook.js";
|
|
8
|
-
|
|
9
|
-
const createConfig = (overrides: Partial<VoiceCallConfig> = {}): VoiceCallConfig => {
|
|
10
|
-
const base = VoiceCallConfigSchema.parse({
|
|
11
|
-
enabled: true,
|
|
12
|
-
provider: "plivo",
|
|
13
|
-
fromNumber: "+15550000000",
|
|
14
|
-
inboundPolicy: "disabled",
|
|
15
|
-
});
|
|
16
|
-
base.serve.port = 0;
|
|
17
|
-
|
|
18
|
-
return {
|
|
19
|
-
...base,
|
|
20
|
-
...overrides,
|
|
21
|
-
serve: {
|
|
22
|
-
...base.serve,
|
|
23
|
-
...overrides.serve,
|
|
24
|
-
},
|
|
25
|
-
};
|
|
26
|
-
};
|
|
27
|
-
|
|
28
|
-
async function postWebhookForm(server: VoiceCallWebhookServer, baseUrl: string, body: string) {
|
|
29
|
-
const address = (
|
|
30
|
-
server as unknown as { server?: { address?: () => unknown } }
|
|
31
|
-
).server?.address?.();
|
|
32
|
-
const requestUrl = new URL(baseUrl);
|
|
33
|
-
if (
|
|
34
|
-
!address ||
|
|
35
|
-
typeof address !== "object" ||
|
|
36
|
-
!("port" in address) ||
|
|
37
|
-
(typeof address.port !== "number" && typeof address.port !== "string") ||
|
|
38
|
-
!address.port
|
|
39
|
-
) {
|
|
40
|
-
throw new Error("voice webhook server did not expose a bound port");
|
|
41
|
-
}
|
|
42
|
-
requestUrl.port = String(address.port);
|
|
43
|
-
return await fetch(requestUrl.toString(), {
|
|
44
|
-
method: "POST",
|
|
45
|
-
headers: {
|
|
46
|
-
"content-type": "application/x-www-form-urlencoded",
|
|
47
|
-
"x-plivo-signature-v2": "sig",
|
|
48
|
-
"x-plivo-signature-v2-nonce": "nonce",
|
|
49
|
-
},
|
|
50
|
-
body,
|
|
51
|
-
});
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
async function runDuplicateInboundReplayLifecycleTest(provider: FakeProvider) {
|
|
55
|
-
const config = createConfig();
|
|
56
|
-
const manager = new CallManager(config, createTestStorePath());
|
|
57
|
-
await manager.initialize(provider, "https://example.com/voice/webhook");
|
|
58
|
-
const server = new VoiceCallWebhookServer(config, manager, provider);
|
|
59
|
-
|
|
60
|
-
try {
|
|
61
|
-
const baseUrl = await server.start();
|
|
62
|
-
const first = await postWebhookForm(server, baseUrl, "CallSid=CA123&From=%2B15552222222");
|
|
63
|
-
const second = await postWebhookForm(server, baseUrl, "CallSid=CA123&From=%2B15552222222");
|
|
64
|
-
return { first, second, manager };
|
|
65
|
-
} finally {
|
|
66
|
-
await server.stop();
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
function expectSingleRejectedReplayHangup(params: {
|
|
71
|
-
first: Response;
|
|
72
|
-
second: Response;
|
|
73
|
-
provider: FakeProvider;
|
|
74
|
-
manager: CallManager;
|
|
75
|
-
}) {
|
|
76
|
-
expect(params.first.status).toBe(200);
|
|
77
|
-
expect(params.second.status).toBe(200);
|
|
78
|
-
expect(params.provider.hangupCalls).toHaveLength(1);
|
|
79
|
-
const [hangupCall] = params.provider.hangupCalls;
|
|
80
|
-
if (!hangupCall) {
|
|
81
|
-
throw new Error("Expected rejected replay hangup call");
|
|
82
|
-
}
|
|
83
|
-
expect(hangupCall.providerCallId).toBe("provider-inbound-1");
|
|
84
|
-
expect(hangupCall.reason).toBe("hangup-bot");
|
|
85
|
-
expect(params.manager.getCallByProviderCallId("provider-inbound-1")).toBeUndefined();
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
class RejectInboundReplayProvider extends FakeProvider {
|
|
89
|
-
override verifyWebhook() {
|
|
90
|
-
return { ok: true, verifiedRequestKey: "verified:req:reject-once" };
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
override parseWebhookEvent(_ctx: WebhookContext, options?: WebhookParseOptions) {
|
|
94
|
-
return {
|
|
95
|
-
statusCode: 200,
|
|
96
|
-
events: [
|
|
97
|
-
{
|
|
98
|
-
id: "evt-reject-once",
|
|
99
|
-
dedupeKey: options?.verifiedRequestKey,
|
|
100
|
-
type: "call.initiated" as const,
|
|
101
|
-
callId: "provider-inbound-1",
|
|
102
|
-
providerCallId: "provider-inbound-1",
|
|
103
|
-
timestamp: Date.now(),
|
|
104
|
-
direction: "inbound" as const,
|
|
105
|
-
from: "+15552222222",
|
|
106
|
-
to: "+15550000000",
|
|
107
|
-
},
|
|
108
|
-
],
|
|
109
|
-
};
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
class RejectInboundReplayWithHangupFailureProvider extends RejectInboundReplayProvider {
|
|
114
|
-
override async hangupCall(input: Parameters<FakeProvider["hangupCall"]>[0]): Promise<void> {
|
|
115
|
-
this.hangupCalls.push(input);
|
|
116
|
-
throw new Error("hangup failed");
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
describe("Voice-call webhook hangup-once lifecycle", () => {
|
|
121
|
-
afterEach(() => {
|
|
122
|
-
// Each test uses an isolated store path, so only server cleanup is needed.
|
|
123
|
-
});
|
|
124
|
-
|
|
125
|
-
it("hangs up a rejected inbound replay only once across duplicate webhook delivery", async () => {
|
|
126
|
-
const provider = new RejectInboundReplayProvider("plivo");
|
|
127
|
-
const { first, second, manager } = await runDuplicateInboundReplayLifecycleTest(provider);
|
|
128
|
-
expectSingleRejectedReplayHangup({ first, second, provider, manager });
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
it("does not attempt a second hangup when replay arrives after the first hangup fails", async () => {
|
|
132
|
-
const provider = new RejectInboundReplayWithHangupFailureProvider("plivo");
|
|
133
|
-
const { first, second, manager } = await runDuplicateInboundReplayLifecycleTest(provider);
|
|
134
|
-
expectSingleRejectedReplayHangup({ first, second, provider, manager });
|
|
135
|
-
});
|
|
136
|
-
|
|
137
|
-
it("keeps rejected inbound replay keys after manager restart", async () => {
|
|
138
|
-
const storePath = createTestStorePath();
|
|
139
|
-
const config = createConfig();
|
|
140
|
-
const firstProvider = new RejectInboundReplayProvider("plivo");
|
|
141
|
-
const firstManager = new CallManager(config, storePath);
|
|
142
|
-
await firstManager.initialize(firstProvider, "https://example.com/voice/webhook");
|
|
143
|
-
const firstServer = new VoiceCallWebhookServer(config, firstManager, firstProvider);
|
|
144
|
-
|
|
145
|
-
try {
|
|
146
|
-
const baseUrl = await firstServer.start();
|
|
147
|
-
const first = await postWebhookForm(
|
|
148
|
-
firstServer,
|
|
149
|
-
baseUrl,
|
|
150
|
-
"CallSid=CA123&From=%2B15552222222",
|
|
151
|
-
);
|
|
152
|
-
expect(first.status).toBe(200);
|
|
153
|
-
} finally {
|
|
154
|
-
await firstServer.stop();
|
|
155
|
-
}
|
|
156
|
-
await flushPendingCallRecordWritesForTest();
|
|
157
|
-
expect(firstProvider.hangupCalls).toHaveLength(1);
|
|
158
|
-
|
|
159
|
-
const secondProvider = new RejectInboundReplayProvider("plivo");
|
|
160
|
-
const secondManager = new CallManager(config, storePath);
|
|
161
|
-
await secondManager.initialize(secondProvider, "https://example.com/voice/webhook");
|
|
162
|
-
const secondServer = new VoiceCallWebhookServer(config, secondManager, secondProvider);
|
|
163
|
-
|
|
164
|
-
try {
|
|
165
|
-
const baseUrl = await secondServer.start();
|
|
166
|
-
const replay = await postWebhookForm(
|
|
167
|
-
secondServer,
|
|
168
|
-
baseUrl,
|
|
169
|
-
"CallSid=CA123&From=%2B15552222222",
|
|
170
|
-
);
|
|
171
|
-
expect(replay.status).toBe(200);
|
|
172
|
-
} finally {
|
|
173
|
-
await secondServer.stop();
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
expect(secondProvider.hangupCalls).toHaveLength(0);
|
|
177
|
-
expect(secondManager.getCallByProviderCallId("provider-inbound-1")).toBeUndefined();
|
|
178
|
-
});
|
|
179
|
-
});
|