Files
openclaw/extensions/xai/tts.test.ts
Jason Zhou bfd540bcdf [codex] refresh plugin regression fixtures
Summary:
- Refresh plugin regression fixtures and test-support mocks for guarded network resolution, progress streaming windows, staged TTS output, QQBot STT, and CLI runner assertions.
- Resolve current-main conflicts in Discord, Google video, QQBot STT, and CLI runner tests without changing runtime code.

Verification:
- pnpm check:test-types
- pnpm vitest run $(git diff --name-only origin/main...HEAD)
- git diff --check
- GitHub CI passed, including Real behavior proof, auto-response, ClawSweeper dispatch, CodeQL, and full CI checks.

Co-authored-by: Jason Zhou <22532527+JayZeeDesign@users.noreply.github.com>
2026-05-10 23:44:50 -05:00

98 lines
2.9 KiB
TypeScript

import { mockPinnedHostnameResolution } from "openclaw/plugin-sdk/test-env";
import { beforeEach, afterEach, describe, expect, it, vi } from "vitest";
import { isValidXaiTtsVoice, XAI_BASE_URL, XAI_TTS_VOICES, xaiTTS } from "./tts.js";
describe("xai tts", () => {
const originalFetch = globalThis.fetch;
let ssrfMock: { mockRestore: () => void } | undefined;
beforeEach(() => {
ssrfMock = mockPinnedHostnameResolution();
});
afterEach(() => {
ssrfMock?.mockRestore();
ssrfMock = undefined;
globalThis.fetch = originalFetch;
vi.restoreAllMocks();
});
describe("isValidXaiTtsVoice", () => {
it("accepts all valid voices", () => {
for (const voice of XAI_TTS_VOICES) {
expect(isValidXaiTtsVoice(voice)).toBe(true);
}
});
it("rejects invalid voice names", () => {
expect(isValidXaiTtsVoice("invalid")).toBe(false);
expect(isValidXaiTtsVoice("")).toBe(false);
expect(isValidXaiTtsVoice("ALLOY")).toBe(false);
expect(isValidXaiTtsVoice("alloy ")).toBe(false);
expect(isValidXaiTtsVoice(" alloy")).toBe(false);
});
it("treats custom endpoints as permissive", () => {
expect(isValidXaiTtsVoice("grok-voice-custom", "https://custom.api.x.ai/v1")).toBe(true);
});
});
describe("xaiTTS diagnostics", () => {
it("includes parsed provider detail and request id for JSON API errors", async () => {
const fetchMock = vi.fn(
async () =>
new Response(
JSON.stringify({
error: {
message: "Invalid API key",
type: "invalid_request_error",
code: "invalid_api_key",
},
}),
{
status: 401,
headers: {
"Content-Type": "application/json",
"x-request-id": "req_123",
},
},
),
);
globalThis.fetch = fetchMock as unknown as typeof fetch;
await expect(
xaiTTS({
text: "hello",
apiKey: "bad-key",
baseUrl: XAI_BASE_URL,
voiceId: "eve",
language: "en",
responseFormat: "mp3",
timeoutMs: 5_000,
}),
).rejects.toThrow(
"xAI TTS API error (401): Invalid API key [type=invalid_request_error, code=invalid_api_key] [request_id=req_123]",
);
});
it("falls back to raw body text when the error body is non-JSON", async () => {
const fetchMock = vi.fn(
async () => new Response("temporary upstream outage", { status: 503 }),
);
globalThis.fetch = fetchMock as unknown as typeof fetch;
await expect(
xaiTTS({
text: "hello",
apiKey: "test-key",
baseUrl: XAI_BASE_URL,
voiceId: "eve",
language: "en",
responseFormat: "mp3",
timeoutMs: 5_000,
}),
).rejects.toThrow("xAI TTS API error (503): temporary upstream outage");
});
});
});