Files
openclaw/extensions/runway/video-generation-provider.test.ts
2026-06-04 21:02:07 -04:00

326 lines
10 KiB
TypeScript

// Runway tests cover video generation provider plugin behavior.
import {
getProviderHttpMocks,
installProviderHttpMockCleanup,
} from "openclaw/plugin-sdk/provider-http-test-mocks";
import { expectExplicitVideoGenerationCapabilities } from "openclaw/plugin-sdk/provider-test-contracts";
import { beforeAll, describe, expect, it, vi } from "vitest";
const { postJsonRequestMock, fetchWithTimeoutMock } = getProviderHttpMocks();
let buildRunwayVideoGenerationProvider: typeof import("./video-generation-provider.js").buildRunwayVideoGenerationProvider;
beforeAll(async () => {
({ buildRunwayVideoGenerationProvider } = await import("./video-generation-provider.js"));
});
installProviderHttpMockCleanup();
function firstPostJsonRequest() {
const [call] = postJsonRequestMock.mock.calls;
if (!call) {
throw new Error("expected Runway create request");
}
const [request] = call;
if (!request || typeof request !== "object") {
throw new Error("expected Runway create request options");
}
return request as { url?: string; body?: Record<string, unknown> };
}
function firstFetchWithTimeoutCall() {
const [call] = fetchWithTimeoutMock.mock.calls;
if (!call) {
throw new Error("expected Runway poll request");
}
const [url, init, timeoutMs, requestFetch] = call;
if (typeof url !== "string") {
throw new Error("expected Runway poll request URL");
}
if (!init || typeof init !== "object" || Array.isArray(init)) {
throw new Error("expected Runway poll request init");
}
if (typeof timeoutMs !== "number") {
throw new Error("expected Runway poll request timeout");
}
return {
init: init as { method?: string; headers?: unknown },
requestFetch,
timeoutMs,
url,
};
}
function streamedVideoResponse(bytes: string): Response {
return new Response(
new ReadableStream({
start(controller) {
controller.enqueue(new TextEncoder().encode(bytes));
controller.close();
},
}),
{ headers: { "content-type": "video/mp4" } },
);
}
describe("runway video generation provider", () => {
it("declares explicit mode capabilities", () => {
expectExplicitVideoGenerationCapabilities(buildRunwayVideoGenerationProvider());
});
it("submits a text-to-video task, polls it, and downloads the output", async () => {
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => ({
id: "task-1",
}),
},
release: vi.fn(async () => {}),
});
fetchWithTimeoutMock
.mockResolvedValueOnce({
json: async () => ({
id: "task-1",
status: "SUCCEEDED",
output: ["https://example.com/out.mp4"],
}),
headers: new Headers(),
})
.mockResolvedValueOnce({
arrayBuffer: async () => Buffer.from("mp4-bytes"),
headers: new Headers({ "content-type": "video/webm" }),
});
const provider = buildRunwayVideoGenerationProvider();
const result = await provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "a tiny lobster DJ under neon lights",
cfg: {},
durationSeconds: 4,
aspectRatio: "16:9",
});
expect(postJsonRequestMock).toHaveBeenCalledTimes(1);
const createRequest = firstPostJsonRequest();
expect(createRequest.url).toBe("https://api.dev.runwayml.com/v1/text_to_video");
expect(createRequest.body).toEqual({
model: "gen4.5",
promptText: "a tiny lobster DJ under neon lights",
ratio: "1280:720",
duration: 4,
});
const pollCall = firstFetchWithTimeoutCall();
expect(pollCall.url).toBe("https://api.dev.runwayml.com/v1/tasks/task-1");
expect(pollCall.init.method).toBe("GET");
expect(pollCall.init.headers).toBeInstanceOf(Headers);
expect(pollCall.timeoutMs).toBe(120000);
expect(pollCall.requestFetch).toBe(fetch);
expect(result.videos).toHaveLength(1);
const video = result.videos[0];
if (!video) {
throw new Error("expected Runway generated video");
}
expect(video.fileName).toBe("video-1.webm");
const metadata = result.metadata as Record<string, unknown>;
expect(metadata.taskId).toBe("task-1");
expect(metadata.status).toBe("SUCCEEDED");
expect(metadata.endpoint).toBe("/v1/text_to_video");
});
it("rejects generated video downloads that exceed the configured media cap", async () => {
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => ({ id: "task-too-large" }),
},
release: vi.fn(async () => {}),
});
fetchWithTimeoutMock
.mockResolvedValueOnce({
json: async () => ({
id: "task-too-large",
status: "SUCCEEDED",
output: ["https://example.com/out.mp4"],
}),
headers: new Headers(),
})
.mockResolvedValueOnce(streamedVideoResponse("too-large"));
const provider = buildRunwayVideoGenerationProvider();
await expect(
provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "short video",
cfg: { agents: { defaults: { mediaMaxMb: 0.000001 } } },
}),
).rejects.toThrow("Runway generated video download exceeds 1 bytes");
});
it("does not round malformed duration values into create requests", async () => {
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => ({ id: "task-duration" }),
},
release: vi.fn(async () => {}),
});
fetchWithTimeoutMock
.mockResolvedValueOnce({
json: async () => ({
id: "task-duration",
status: "SUCCEEDED",
output: ["https://example.com/out.mp4"],
}),
headers: new Headers(),
})
.mockResolvedValueOnce({
arrayBuffer: async () => Buffer.from("mp4-bytes"),
headers: new Headers({ "content-type": "video/mp4" }),
});
const provider = buildRunwayVideoGenerationProvider();
await provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "a tiny lobster DJ under neon lights",
cfg: {},
durationSeconds: 4.5,
aspectRatio: "16:9",
});
expect(postJsonRequestMock).toHaveBeenCalledTimes(1);
expect(firstPostJsonRequest().body?.duration).toBe(5);
});
it("accepts local image buffers by converting them into data URIs", async () => {
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => ({ id: "task-2" }),
},
release: vi.fn(async () => {}),
});
fetchWithTimeoutMock
.mockResolvedValueOnce({
json: async () => ({
id: "task-2",
status: "SUCCEEDED",
output: ["https://example.com/out.mp4"],
}),
headers: new Headers(),
})
.mockResolvedValueOnce({
arrayBuffer: async () => Buffer.from("mp4-bytes"),
headers: new Headers({ "content-type": "video/mp4" }),
});
const provider = buildRunwayVideoGenerationProvider();
await provider.generateVideo({
provider: "runway",
model: "gen4_turbo",
prompt: "animate this frame",
cfg: {},
inputImages: [{ buffer: Buffer.from("png-bytes"), mimeType: "image/png" }],
aspectRatio: "1:1",
durationSeconds: 6,
});
expect(postJsonRequestMock).toHaveBeenCalledTimes(1);
const request = firstPostJsonRequest();
expect(request.url).toBe("https://api.dev.runwayml.com/v1/image_to_video");
expect(request.body?.promptImage).toMatch(/^data:image\/png;base64,/u);
expect(request.body?.ratio).toBe("960:960");
expect(request.body?.duration).toBe(6);
});
it("requires gen4_aleph for video-to-video", async () => {
const provider = buildRunwayVideoGenerationProvider();
await expect(
provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "restyle this clip",
cfg: {},
inputVideos: [{ url: "https://example.com/input.mp4" }],
}),
).rejects.toThrow("Runway video-to-video currently requires model gen4_aleph.");
expect(postJsonRequestMock).not.toHaveBeenCalled();
});
it("reports malformed create JSON with a provider-owned error", async () => {
const release = vi.fn(async () => {});
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => {
throw new SyntaxError("bad json");
},
},
release,
});
const provider = buildRunwayVideoGenerationProvider();
await expect(
provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "bad create response",
cfg: {},
}),
).rejects.toThrow("Runway video generation failed: malformed JSON response");
expect(release).toHaveBeenCalledOnce();
});
it("rejects status responses missing a task status", async () => {
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => ({ id: "task-missing-status" }),
},
release: vi.fn(async () => {}),
});
fetchWithTimeoutMock.mockResolvedValueOnce({
json: async () => ({
id: "task-missing-status",
output: ["https://example.com/out.mp4"],
}),
headers: new Headers(),
});
const provider = buildRunwayVideoGenerationProvider();
await expect(
provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "missing status",
cfg: {},
}),
).rejects.toThrow("Runway video status response missing task status");
});
it("rejects malformed completed output URLs", async () => {
postJsonRequestMock.mockResolvedValue({
response: {
json: async () => ({ id: "task-malformed-output" }),
},
release: vi.fn(async () => {}),
});
fetchWithTimeoutMock.mockResolvedValueOnce({
json: async () => ({
id: "task-malformed-output",
status: "SUCCEEDED",
output: "https://example.com/out.mp4",
}),
headers: new Headers(),
});
const provider = buildRunwayVideoGenerationProvider();
await expect(
provider.generateVideo({
provider: "runway",
model: "gen4.5",
prompt: "malformed output",
cfg: {},
}),
).rejects.toThrow("Runway video generation completed with malformed output URLs");
});
});