mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-06 05:51:15 +08:00
326 lines
10 KiB
TypeScript
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");
|
|
});
|
|
});
|