mirror of https://github.com/openclaw/openclaw.git
219 lines
6.7 KiB
TypeScript
219 lines
6.7 KiB
TypeScript
import { assertOkOrThrowHttpError, fetchWithTimeout } from "openclaw/plugin-sdk/provider-http";
|
|
|
|
export const DEFAULT_VYDRA_BASE_URL = "https://www.vydra.ai/api/v1";
|
|
export const DEFAULT_VYDRA_IMAGE_MODEL = "grok-imagine";
|
|
export const DEFAULT_VYDRA_VIDEO_MODEL = "veo3";
|
|
export const DEFAULT_VYDRA_SPEECH_MODEL = "elevenlabs/tts";
|
|
export const DEFAULT_VYDRA_VOICE_ID = "21m00Tcm4TlvDq8ikWAM";
|
|
export const DEFAULT_HTTP_TIMEOUT_MS = 120_000;
|
|
const POLL_INTERVAL_MS = 2_500;
|
|
const MAX_POLL_ATTEMPTS = 120;
|
|
|
|
type VydraMediaKind = "audio" | "image" | "video";
|
|
|
|
type VydraJobPayload = {
|
|
id?: string;
|
|
jobId?: string;
|
|
status?: string;
|
|
message?: string;
|
|
error?: string | { message?: string; detail?: string } | null;
|
|
};
|
|
|
|
function asObject(value: unknown): Record<string, unknown> | undefined {
|
|
return typeof value === "object" && value !== null && !Array.isArray(value)
|
|
? (value as Record<string, unknown>)
|
|
: undefined;
|
|
}
|
|
|
|
function addUrlValue(value: unknown, urls: Set<string>): void {
|
|
if (typeof value === "string") {
|
|
const trimmed = value.trim();
|
|
if (/^https?:\/\//iu.test(trimmed)) {
|
|
urls.add(trimmed);
|
|
}
|
|
return;
|
|
}
|
|
if (Array.isArray(value)) {
|
|
for (const entry of value) {
|
|
addUrlValue(entry, urls);
|
|
}
|
|
}
|
|
}
|
|
|
|
export function trimToUndefined(value: unknown): string | undefined {
|
|
return typeof value === "string" && value.trim() ? value.trim() : undefined;
|
|
}
|
|
|
|
export function normalizeVydraBaseUrl(value: string | undefined): string {
|
|
const fallback = DEFAULT_VYDRA_BASE_URL;
|
|
const trimmed = trimToUndefined(value);
|
|
if (!trimmed) {
|
|
return fallback;
|
|
}
|
|
try {
|
|
const url = new URL(trimmed);
|
|
if (url.hostname === "vydra.ai") {
|
|
url.hostname = "www.vydra.ai";
|
|
}
|
|
const pathname = url.pathname.replace(/\/+$/u, "");
|
|
if (!pathname) {
|
|
url.pathname = "/api/v1";
|
|
} else {
|
|
url.pathname = pathname;
|
|
}
|
|
return url.toString().replace(/\/$/u, "");
|
|
} catch {
|
|
return fallback;
|
|
}
|
|
}
|
|
|
|
export function resolveVydraBaseUrlFromConfig(cfg: unknown): string {
|
|
const models = asObject(asObject(cfg)?.models);
|
|
const providers = asObject(models?.providers);
|
|
const vydra = asObject(providers?.vydra);
|
|
return normalizeVydraBaseUrl(trimToUndefined(vydra?.baseUrl));
|
|
}
|
|
|
|
export function resolveVydraResponseJobId(payload: unknown): string | undefined {
|
|
const object = asObject(payload) as VydraJobPayload | undefined;
|
|
return trimToUndefined(object?.jobId) ?? trimToUndefined(object?.id);
|
|
}
|
|
|
|
export function resolveVydraResponseStatus(payload: unknown): string | undefined {
|
|
return trimToUndefined(asObject(payload)?.status)?.toLowerCase();
|
|
}
|
|
|
|
export function resolveVydraErrorMessage(payload: unknown): string | undefined {
|
|
const object = asObject(payload) as VydraJobPayload | undefined;
|
|
const error = object?.error;
|
|
if (typeof error === "string" && error.trim()) {
|
|
return error.trim();
|
|
}
|
|
const errorObject = asObject(error);
|
|
return (
|
|
trimToUndefined(errorObject?.message) ??
|
|
trimToUndefined(errorObject?.detail) ??
|
|
trimToUndefined(object?.message)
|
|
);
|
|
}
|
|
|
|
export function extractVydraResultUrls(payload: unknown, kind: VydraMediaKind): string[] {
|
|
const urls = new Set<string>();
|
|
const preferredKeys =
|
|
kind === "audio"
|
|
? ["audioUrl", "audioUrls"]
|
|
: kind === "image"
|
|
? ["imageUrl", "imageUrls"]
|
|
: ["videoUrl", "videoUrls"];
|
|
const sharedKeys = ["resultUrl", "resultUrls", "outputUrl", "outputUrls", "url", "urls"];
|
|
const recurseKeys = ["output", "outputs", "result", "results", "data", "asset", "assets"];
|
|
|
|
const visit = (value: unknown, depth = 0) => {
|
|
if (depth > 5) {
|
|
return;
|
|
}
|
|
if (Array.isArray(value)) {
|
|
for (const entry of value) {
|
|
visit(entry, depth + 1);
|
|
}
|
|
return;
|
|
}
|
|
const object = asObject(value);
|
|
if (!object) {
|
|
return;
|
|
}
|
|
for (const key of [...preferredKeys, ...sharedKeys]) {
|
|
addUrlValue(object[key], urls);
|
|
}
|
|
for (const key of recurseKeys) {
|
|
if (key in object) {
|
|
visit(object[key], depth + 1);
|
|
}
|
|
}
|
|
};
|
|
|
|
visit(payload);
|
|
return [...urls];
|
|
}
|
|
|
|
function inferExtension(kind: VydraMediaKind, mimeType: string): string {
|
|
const normalized = mimeType.toLowerCase();
|
|
if (normalized.includes("jpeg")) {
|
|
return "jpg";
|
|
}
|
|
if (normalized.includes("webp")) {
|
|
return "webp";
|
|
}
|
|
if (normalized.includes("wav")) {
|
|
return "wav";
|
|
}
|
|
if (normalized.includes("mpeg") || normalized.includes("mp3")) {
|
|
return "mp3";
|
|
}
|
|
if (normalized.includes("webm")) {
|
|
return "webm";
|
|
}
|
|
if (normalized.includes("quicktime")) {
|
|
return "mov";
|
|
}
|
|
return kind === "image" ? "png" : kind === "audio" ? "mp3" : "mp4";
|
|
}
|
|
|
|
export async function downloadVydraAsset(params: {
|
|
url: string;
|
|
kind: VydraMediaKind;
|
|
timeoutMs?: number;
|
|
fetchFn: typeof fetch;
|
|
}): Promise<{ buffer: Buffer; mimeType: string; fileName: string }> {
|
|
const response = await fetchWithTimeout(
|
|
params.url,
|
|
{ method: "GET" },
|
|
params.timeoutMs ?? DEFAULT_HTTP_TIMEOUT_MS,
|
|
params.fetchFn,
|
|
);
|
|
await assertOkOrThrowHttpError(response, `Vydra ${params.kind} download failed`);
|
|
const mimeType =
|
|
response.headers.get("content-type")?.trim() ||
|
|
(params.kind === "image" ? "image/png" : params.kind === "audio" ? "audio/mpeg" : "video/mp4");
|
|
const arrayBuffer = await response.arrayBuffer();
|
|
const extension = inferExtension(params.kind, mimeType);
|
|
const fileStem = params.kind === "image" ? "image" : params.kind === "audio" ? "audio" : "video";
|
|
return {
|
|
buffer: Buffer.from(arrayBuffer),
|
|
mimeType,
|
|
fileName: `${fileStem}-1.${extension}`,
|
|
};
|
|
}
|
|
|
|
export async function waitForVydraJob(params: {
|
|
baseUrl: string;
|
|
jobId: string;
|
|
headers: Headers;
|
|
timeoutMs?: number;
|
|
fetchFn: typeof fetch;
|
|
kind: VydraMediaKind;
|
|
}): Promise<unknown> {
|
|
for (let attempt = 0; attempt < MAX_POLL_ATTEMPTS; attempt += 1) {
|
|
const response = await fetchWithTimeout(
|
|
`${params.baseUrl}/jobs/${params.jobId}`,
|
|
{
|
|
method: "GET",
|
|
headers: params.headers,
|
|
},
|
|
params.timeoutMs ?? DEFAULT_HTTP_TIMEOUT_MS,
|
|
params.fetchFn,
|
|
);
|
|
await assertOkOrThrowHttpError(response, "Vydra job status request failed");
|
|
const payload = await response.json();
|
|
const status = resolveVydraResponseStatus(payload);
|
|
if (status === "completed" || extractVydraResultUrls(payload, params.kind).length > 0) {
|
|
return payload;
|
|
}
|
|
if (status === "failed" || status === "error" || status === "cancelled") {
|
|
throw new Error(resolveVydraErrorMessage(payload) ?? `Vydra job ${params.jobId} failed`);
|
|
}
|
|
await new Promise((resolve) => setTimeout(resolve, POLL_INTERVAL_MS));
|
|
}
|
|
throw new Error(`Vydra job ${params.jobId} did not finish in time`);
|
|
}
|