@absolutejs/voice 0.0.22-beta.205 → 0.0.22-beta.206
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/README.md +24 -1
- package/dist/index.js +45 -7
- package/dist/phoneAgent.d.ts +14 -0
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -851,7 +851,30 @@ The wrapper mounts selected carrier routes plus two proof surfaces:
|
|
|
851
851
|
- `/api/voice/phone/smoke-contract?sessionId=...`: trace-backed production smoke contract.
|
|
852
852
|
- `/voice/phone/smoke-contract?sessionId=...`: HTML production smoke contract.
|
|
853
853
|
|
|
854
|
-
The setup
|
|
854
|
+
The setup JSON includes `setupInstructions`, so your own admin UI can render copy-ready carrier fields without scraping HTML:
|
|
855
|
+
|
|
856
|
+
```ts
|
|
857
|
+
const setup = await fetch('/api/voice/phone/setup').then((response) =>
|
|
858
|
+
response.json()
|
|
859
|
+
);
|
|
860
|
+
|
|
861
|
+
for (const carrier of setup.setupInstructions) {
|
|
862
|
+
console.log(carrier.carrierName, carrier.status);
|
|
863
|
+
console.log(carrier.steps.join('\n'));
|
|
864
|
+
}
|
|
865
|
+
```
|
|
866
|
+
|
|
867
|
+
Each instruction includes:
|
|
868
|
+
|
|
869
|
+
- `answerLabel`: `TwiML URL`, `TeXML URL`, or `Answer URL`.
|
|
870
|
+
- `answerUrl`: the URL to paste into the carrier's inbound voice/answer field.
|
|
871
|
+
- `webhookUrl`: the status callback or webhook URL.
|
|
872
|
+
- `streamUrl`: the `wss://` media stream URL the carrier must reach.
|
|
873
|
+
- `setupPath` and `smokePath`: package-mounted proof pages for that carrier.
|
|
874
|
+
- `steps`: ordered copy/paste guidance for the carrier dashboard.
|
|
875
|
+
- `issues`: contract errors or warnings from the carrier matrix.
|
|
876
|
+
|
|
877
|
+
The setup page renders the same instructions and tells you exactly what to copy into the carrier dashboard:
|
|
855
878
|
|
|
856
879
|
- Twilio: set the phone number voice webhook/TwiML URL to the reported TwiML URL, set the status callback to the reported webhook URL, and allow the reported `wss://` media stream.
|
|
857
880
|
- Telnyx: set the connection TeXML URL to the reported TeXML URL, set the status webhook to the reported webhook URL, and allow the reported `wss://` media stream.
|
package/dist/index.js
CHANGED
|
@@ -19096,6 +19096,41 @@ var loadCarrierMatrixInputs = async (input) => {
|
|
|
19096
19096
|
}
|
|
19097
19097
|
return entries;
|
|
19098
19098
|
};
|
|
19099
|
+
var carrierAnswerLabel = (provider) => provider === "telnyx" ? "TeXML URL" : provider === "plivo" ? "Answer URL" : "TwiML URL";
|
|
19100
|
+
var findCarrierMatrixEntry = (matrix, carrier) => matrix?.entries.find((candidate) => candidate.provider === carrier.provider && (candidate.name === carrier.name || candidate.name === (carrier.name ?? carrier.provider)));
|
|
19101
|
+
var buildVoicePhoneAgentSetupInstructions = (input) => input.carriers.map((carrier) => {
|
|
19102
|
+
const entry = findCarrierMatrixEntry(input.matrix, carrier);
|
|
19103
|
+
const urls = entry?.setup.urls;
|
|
19104
|
+
const answerLabel = carrierAnswerLabel(carrier.provider);
|
|
19105
|
+
const urlRecord = urls ?? {};
|
|
19106
|
+
const answerUrl = urlRecord.twiml ?? urlRecord.texml ?? urlRecord.answer;
|
|
19107
|
+
const webhookUrl = urls?.webhook;
|
|
19108
|
+
const streamUrl = urls?.stream;
|
|
19109
|
+
const setupPath = typeof carrier.setupPath === "string" ? `${carrier.setupPath}?format=html` : undefined;
|
|
19110
|
+
const smokePath = typeof carrier.smokePath === "string" ? `${carrier.smokePath}?format=html` : undefined;
|
|
19111
|
+
const productionSmokePath = input.productionSmokePath ? `${input.productionSmokePath.replace("/api/", "/")}?sessionId=` : undefined;
|
|
19112
|
+
const steps = [
|
|
19113
|
+
`Set ${answerLabel} to ${answerUrl ?? "missing"}.`,
|
|
19114
|
+
`Set status webhook to ${webhookUrl ?? "missing"}.`,
|
|
19115
|
+
`Allow media stream URL ${streamUrl ?? "missing"}.`,
|
|
19116
|
+
...setupPath ? [`Open setup report ${setupPath}.`] : [],
|
|
19117
|
+
...smokePath ? [`Run carrier smoke ${smokePath}.`] : [],
|
|
19118
|
+
...productionSmokePath ? [`Certify production smoke traces at ${productionSmokePath}.`] : []
|
|
19119
|
+
];
|
|
19120
|
+
return {
|
|
19121
|
+
answerLabel,
|
|
19122
|
+
answerUrl,
|
|
19123
|
+
carrierName: carrier.name ?? carrier.provider,
|
|
19124
|
+
issues: entry?.issues.map((issue) => `${issue.severity}: ${issue.message}`) ?? [],
|
|
19125
|
+
provider: carrier.provider,
|
|
19126
|
+
setupPath: carrier.setupPath,
|
|
19127
|
+
smokePath: carrier.smokePath,
|
|
19128
|
+
status: entry?.status ?? "unknown",
|
|
19129
|
+
steps,
|
|
19130
|
+
streamUrl,
|
|
19131
|
+
webhookUrl
|
|
19132
|
+
};
|
|
19133
|
+
});
|
|
19099
19134
|
var resolveCarrierContract = async (input) => {
|
|
19100
19135
|
const matrixInputs = await loadCarrierMatrixInputs({
|
|
19101
19136
|
app: input.app,
|
|
@@ -19111,7 +19146,7 @@ var resolveCarrierContract = async (input) => {
|
|
|
19111
19146
|
};
|
|
19112
19147
|
var renderVoicePhoneAgentSetupHTML = (report) => {
|
|
19113
19148
|
const carrierRows = report.carriers.map((carrier) => {
|
|
19114
|
-
const entry = report.matrix
|
|
19149
|
+
const entry = findCarrierMatrixEntry(report.matrix, carrier);
|
|
19115
19150
|
const urls = entry?.setup.urls;
|
|
19116
19151
|
const primaryUrl = carrier.provider === "plivo" ? urls?.twiml : urls?.twiml;
|
|
19117
19152
|
return `<tr><td>${escapeHtml33(carrier.name ?? carrier.provider)}</td><td>${escapeHtml33(carrier.provider)}</td><td><code>${escapeHtml33(carrier.setupPath || "disabled")}</code></td><td><code>${escapeHtml33(carrier.smokePath || "disabled")}</code></td><td>${entry ? `<span class="${escapeHtml33(entry.status)}">${escapeHtml33(entry.status.toUpperCase())}</span>` : "unknown"}</td><td>${primaryUrl ? `<code>${escapeHtml33(primaryUrl)}</code>` : '<span class="muted">missing</span>'}</td><td>${urls?.webhook ? `<code>${escapeHtml33(urls.webhook)}</code>` : '<span class="muted">missing</span>'}</td><td>${urls?.stream ? `<code>${escapeHtml33(urls.stream)}</code>` : '<span class="muted">missing</span>'}</td></tr>`;
|
|
@@ -19150,12 +19185,10 @@ app.use(
|
|
|
19150
19185
|
})
|
|
19151
19186
|
);`);
|
|
19152
19187
|
const checklist = report.carriers.map((carrier) => {
|
|
19153
|
-
const
|
|
19154
|
-
const
|
|
19155
|
-
const
|
|
19156
|
-
|
|
19157
|
-
const issueList = entry?.issues.map((issue) => `<li>${escapeHtml33(issue.severity)}: ${escapeHtml33(issue.message)}</li>`).join("") ?? "";
|
|
19158
|
-
return `<article><h3>${escapeHtml33(carrier.name ?? carrier.provider)}</h3><ol><li>Set ${escapeHtml33(answerLabel)} to <code>${escapeHtml33(answerUrl ?? "missing")}</code>.</li><li>Set status webhook to <code>${escapeHtml33(urls?.webhook ?? "missing")}</code>.</li><li>Allow media stream URL <code>${escapeHtml33(urls?.stream ?? "missing")}</code>.</li><li>Open setup: ${carrier.setupPath ? `<a href="${escapeHtml33(carrier.setupPath)}?format=html">${escapeHtml33(carrier.setupPath)}</a>` : '<span class="muted">disabled</span>'}.</li><li>Run smoke: ${carrier.smokePath ? `<a href="${escapeHtml33(carrier.smokePath)}?format=html">${escapeHtml33(carrier.smokePath)}</a>` : '<span class="muted">disabled</span>'}.</li>${report.productionSmokePath ? `<li>Certify production smoke traces: <a href="${escapeHtml33(report.productionSmokePath.replace("/api/", "/"))}?sessionId=">${escapeHtml33(report.productionSmokePath)}</a>.</li>` : ""}</ol>${issueList ? `<ul class="issues">${issueList}</ul>` : '<p class="pass">No carrier contract issues.</p>'}</article>`;
|
|
19188
|
+
const instruction = report.setupInstructions.find((candidate) => candidate.provider === carrier.provider && candidate.carrierName === (carrier.name ?? carrier.provider));
|
|
19189
|
+
const issueList = instruction?.issues.map((issue) => `<li>${escapeHtml33(issue)}</li>`).join("") ?? "";
|
|
19190
|
+
const steps = instruction?.steps.map((step) => `<li>${escapeHtml33(step)}</li>`).join("") ?? "";
|
|
19191
|
+
return `<article><h3>${escapeHtml33(carrier.name ?? carrier.provider)}</h3><ol>${steps}</ol>${issueList ? `<ul class="issues">${issueList}</ul>` : '<p class="pass">No carrier contract issues.</p>'}</article>`;
|
|
19159
19192
|
}).join("");
|
|
19160
19193
|
return `<!doctype html><html lang="en"><head><meta charset="utf-8" /><meta name="viewport" content="width=device-width, initial-scale=1" /><title>${escapeHtml33(report.title)}</title><style>body{background:#10151c;color:#f8f3e7;font-family:ui-sans-serif,system-ui,sans-serif;margin:0}main{margin:auto;max-width:1180px;padding:32px}.hero,.primitive{background:linear-gradient(135deg,rgba(20,184,166,.2),rgba(245,158,11,.12));border:1px solid #283544;border-radius:28px;margin-bottom:18px;padding:28px}.primitive{background:#151d27;border-color:#365a60}.eyebrow{color:#5eead4;font-weight:900;letter-spacing:.12em;text-transform:uppercase}h1{font-size:clamp(2.3rem,6vw,4.8rem);line-height:.92;margin:.2rem 0 1rem}.badge{border:1px solid #3f3f46;border-radius:999px;display:inline-flex;padding:8px 12px}.pass{color:#86efac}.fail{color:#fca5a5}.warn{color:#fde68a}.muted{color:#aab5c0}table{background:#151d27;border:1px solid #283544;border-collapse:collapse;border-radius:18px;display:block;overflow:auto;width:100%}td,th{border-bottom:1px solid #283544;padding:12px;text-align:left;vertical-align:top}code{color:#fde68a;overflow-wrap:anywhere}.primitive p{color:#cbd5de;line-height:1.55}.primitive pre{background:#0b1118;border:1px solid #283544;border-radius:18px;color:#fef3c7;overflow:auto;padding:16px}.checklist{display:grid;gap:14px;grid-template-columns:repeat(auto-fit,minmax(280px,1fr));margin:18px 0}.checklist article{background:#151d27;border:1px solid #283544;border-radius:18px;padding:18px}.checklist ol{padding-left:20px}.issues{color:#fca5a5}.stages{display:grid;gap:8px;grid-template-columns:repeat(auto-fit,minmax(180px,1fr));padding-left:18px}a{color:#5eead4}</style></head><body><main><section class="hero"><p class="eyebrow">Phone agent setup</p><h1>${escapeHtml33(report.title)}</h1><p>One self-hosted entrypoint for carrier routes, setup reports, smoke checks, and normalized call lifecycle stages.</p><p class="badge ${report.ready ? "pass" : "fail"}">Ready: ${String(report.ready)}</p>${report.matrixPath ? `<p><a href="${escapeHtml33(report.matrixPath)}?format=html">Open carrier matrix</a></p>` : ""}</section><section class="primitive"><p class="eyebrow">Copy into your app</p><h2><code>createVoicePhoneAgent(...)</code> builds this carrier control plane</h2><p>Mount carrier routes once, expose setup and smoke proof, then feed the same carrier matrix and phone-agent smoke reports into production readiness so carrier regressions block deploys.</p><pre><code>${snippet}</code></pre></section><h2>Carrier Setup Checklist</h2><section class="checklist">${checklist}</section><h2>Carrier URLs</h2><table><thead><tr><th>Name</th><th>Provider</th><th>Setup</th><th>Smoke</th><th>Status</th><th>Answer/TwiML/TeXML</th><th>Webhook</th><th>Stream</th></tr></thead><tbody>${carrierRows}</tbody></table><h2>Lifecycle Schema</h2><ul class="stages">${stageList}</ul></main></body></html>`;
|
|
19161
19194
|
};
|
|
@@ -19230,6 +19263,11 @@ var createVoicePhoneAgent = (options) => {
|
|
|
19230
19263
|
matrixPath,
|
|
19231
19264
|
productionSmokePath,
|
|
19232
19265
|
ready: matrix ? matrix.pass : carrierSummaries.length > 0,
|
|
19266
|
+
setupInstructions: buildVoicePhoneAgentSetupInstructions({
|
|
19267
|
+
carriers: carrierSummaries,
|
|
19268
|
+
matrix,
|
|
19269
|
+
productionSmokePath
|
|
19270
|
+
}),
|
|
19233
19271
|
setupPath,
|
|
19234
19272
|
title: setupTitle
|
|
19235
19273
|
};
|
package/dist/phoneAgent.d.ts
CHANGED
|
@@ -40,6 +40,19 @@ export type VoicePhoneAgentCarrierSummary = {
|
|
|
40
40
|
setupPath?: false | string;
|
|
41
41
|
smokePath?: false | string;
|
|
42
42
|
};
|
|
43
|
+
export type VoicePhoneAgentSetupInstruction = {
|
|
44
|
+
answerLabel: string;
|
|
45
|
+
answerUrl?: string;
|
|
46
|
+
carrierName: string;
|
|
47
|
+
issues: string[];
|
|
48
|
+
provider: 'plivo' | 'telnyx' | 'twilio';
|
|
49
|
+
setupPath?: false | string;
|
|
50
|
+
smokePath?: false | string;
|
|
51
|
+
status: 'fail' | 'pass' | 'unknown' | 'warn';
|
|
52
|
+
steps: string[];
|
|
53
|
+
streamUrl?: string;
|
|
54
|
+
webhookUrl?: string;
|
|
55
|
+
};
|
|
43
56
|
export type VoicePhoneAgentRoutes = {
|
|
44
57
|
carriers: VoicePhoneAgentCarrierSummary[];
|
|
45
58
|
matrixPath?: string;
|
|
@@ -55,6 +68,7 @@ export type VoicePhoneAgentSetupReport = {
|
|
|
55
68
|
matrixPath?: string;
|
|
56
69
|
productionSmokePath?: string;
|
|
57
70
|
ready: boolean;
|
|
71
|
+
setupInstructions: VoicePhoneAgentSetupInstruction[];
|
|
58
72
|
setupPath?: string;
|
|
59
73
|
title: string;
|
|
60
74
|
};
|