Files
openclaw/extensions/deepinfra/image-generation-provider.ts
2026-04-28 04:01:43 +01:00

90 lines
3.1 KiB
TypeScript

import {
createOpenAiCompatibleImageGenerationProvider,
imageSourceUploadFileName,
type ImageGenerationProvider,
} from "openclaw/plugin-sdk/image-generation";
import { normalizeOptionalString } from "openclaw/plugin-sdk/text-runtime";
import {
DEEPINFRA_BASE_URL,
DEEPINFRA_IMAGE_MODELS,
DEFAULT_DEEPINFRA_IMAGE_MODEL,
DEFAULT_DEEPINFRA_IMAGE_SIZE,
normalizeDeepInfraBaseUrl,
normalizeDeepInfraModelRef,
} from "./media-models.js";
const DEEPINFRA_IMAGE_SIZES = ["512x512", "1024x1024", "1024x1792", "1792x1024"] as const;
const MAX_DEEPINFRA_INPUT_IMAGES = 1;
export function buildDeepInfraImageGenerationProvider(): ImageGenerationProvider {
return createOpenAiCompatibleImageGenerationProvider({
id: "deepinfra",
label: "DeepInfra",
defaultModel: DEFAULT_DEEPINFRA_IMAGE_MODEL,
models: [...DEEPINFRA_IMAGE_MODELS],
capabilities: {
generate: {
maxCount: 4,
supportsSize: true,
supportsAspectRatio: false,
supportsResolution: false,
},
edit: {
enabled: true,
maxCount: 1,
maxInputImages: MAX_DEEPINFRA_INPUT_IMAGES,
supportsSize: true,
supportsAspectRatio: false,
supportsResolution: false,
},
geometry: {
sizes: [...DEEPINFRA_IMAGE_SIZES],
},
},
defaultBaseUrl: DEEPINFRA_BASE_URL,
normalizeModel: normalizeDeepInfraModelRef,
resolveBaseUrl: ({ providerConfig }) =>
normalizeDeepInfraBaseUrl(providerConfig?.baseUrl, DEEPINFRA_BASE_URL),
resolveAllowPrivateNetwork: () => false,
useConfiguredRequest: true,
resolveCount: ({ req, mode }) => (mode === "edit" ? 1 : (req.count ?? 1)),
buildGenerateRequest: ({ req, model, count }) => ({
kind: "json",
body: {
model,
prompt: req.prompt,
n: count,
size: normalizeOptionalString(req.size) ?? DEFAULT_DEEPINFRA_IMAGE_SIZE,
response_format: "b64_json",
},
}),
buildEditRequest: ({ req, inputImages, model, count }) => {
const image = inputImages[0];
if (!image) {
throw new Error("DeepInfra image edit missing reference image.");
}
const form = new FormData();
form.set("model", model);
form.set("prompt", req.prompt);
form.set("n", String(count));
form.set("size", normalizeOptionalString(req.size) ?? DEFAULT_DEEPINFRA_IMAGE_SIZE);
form.set("response_format", "b64_json");
const mimeType = normalizeOptionalString(image.mimeType) ?? "image/png";
form.append(
"image",
new Blob([new Uint8Array(image.buffer)], { type: mimeType }),
imageSourceUploadFileName({ image, index: 0 }),
);
return { kind: "multipart", form };
},
response: { defaultMimeType: "image/jpeg", sniffMimeType: true },
tooManyInputImagesError: "DeepInfra image editing supports one reference image.",
missingApiKeyError: "DeepInfra API key missing",
emptyResponseError: "DeepInfra image response did not include generated image data",
failureLabels: {
generate: "DeepInfra image generation failed",
edit: "DeepInfra image edit failed",
},
});
}