mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-13 18:21:27 +00:00
153 lines
5.0 KiB
TypeScript
153 lines
5.0 KiB
TypeScript
import type { AuthProfileStore } from "../agents/auth-profiles.js";
|
|
import { describeFailoverError, isFailoverError } from "../agents/failover-error.js";
|
|
import type { FallbackAttempt } from "../agents/model-fallback.types.js";
|
|
import type { OpenClawConfig } from "../config/config.js";
|
|
import { formatErrorMessage } from "../infra/errors.js";
|
|
import { createSubsystemLogger } from "../logging/subsystem.js";
|
|
import {
|
|
buildMediaGenerationNormalizationMetadata,
|
|
buildNoCapabilityModelConfiguredMessage,
|
|
resolveCapabilityModelCandidates,
|
|
throwCapabilityGenerationFailure,
|
|
} from "../media-generation/runtime-shared.js";
|
|
import { parseImageGenerationModelRef } from "./model-ref.js";
|
|
import { resolveImageGenerationOverrides } from "./normalization.js";
|
|
import { getImageGenerationProvider, listImageGenerationProviders } from "./provider-registry.js";
|
|
import type {
|
|
GeneratedImageAsset,
|
|
ImageGenerationIgnoredOverride,
|
|
ImageGenerationNormalization,
|
|
ImageGenerationResolution,
|
|
ImageGenerationResult,
|
|
ImageGenerationSourceImage,
|
|
} from "./types.js";
|
|
|
|
const log = createSubsystemLogger("image-generation");
|
|
|
|
export type GenerateImageParams = {
|
|
cfg: OpenClawConfig;
|
|
prompt: string;
|
|
agentDir?: string;
|
|
authStore?: AuthProfileStore;
|
|
modelOverride?: string;
|
|
count?: number;
|
|
size?: string;
|
|
aspectRatio?: string;
|
|
resolution?: ImageGenerationResolution;
|
|
inputImages?: ImageGenerationSourceImage[];
|
|
};
|
|
|
|
export type GenerateImageRuntimeResult = {
|
|
images: GeneratedImageAsset[];
|
|
provider: string;
|
|
model: string;
|
|
attempts: FallbackAttempt[];
|
|
normalization?: ImageGenerationNormalization;
|
|
metadata?: Record<string, unknown>;
|
|
ignoredOverrides: ImageGenerationIgnoredOverride[];
|
|
};
|
|
|
|
function buildNoImageGenerationModelConfiguredMessage(cfg: OpenClawConfig): string {
|
|
return buildNoCapabilityModelConfiguredMessage({
|
|
capabilityLabel: "image-generation",
|
|
modelConfigKey: "imageGenerationModel",
|
|
providers: listImageGenerationProviders(cfg),
|
|
});
|
|
}
|
|
|
|
export function listRuntimeImageGenerationProviders(params?: { config?: OpenClawConfig }) {
|
|
return listImageGenerationProviders(params?.config);
|
|
}
|
|
|
|
export async function generateImage(
|
|
params: GenerateImageParams,
|
|
): Promise<GenerateImageRuntimeResult> {
|
|
const candidates = resolveCapabilityModelCandidates({
|
|
cfg: params.cfg,
|
|
modelConfig: params.cfg.agents?.defaults?.imageGenerationModel,
|
|
modelOverride: params.modelOverride,
|
|
parseModelRef: parseImageGenerationModelRef,
|
|
agentDir: params.agentDir,
|
|
listProviders: listImageGenerationProviders,
|
|
});
|
|
if (candidates.length === 0) {
|
|
throw new Error(buildNoImageGenerationModelConfiguredMessage(params.cfg));
|
|
}
|
|
|
|
const attempts: FallbackAttempt[] = [];
|
|
let lastError: unknown;
|
|
|
|
for (const candidate of candidates) {
|
|
const provider = getImageGenerationProvider(candidate.provider, params.cfg);
|
|
if (!provider) {
|
|
const error = `No image-generation provider registered for ${candidate.provider}`;
|
|
attempts.push({
|
|
provider: candidate.provider,
|
|
model: candidate.model,
|
|
error,
|
|
});
|
|
lastError = new Error(error);
|
|
continue;
|
|
}
|
|
|
|
try {
|
|
const sanitized = resolveImageGenerationOverrides({
|
|
provider,
|
|
size: params.size,
|
|
aspectRatio: params.aspectRatio,
|
|
resolution: params.resolution,
|
|
inputImages: params.inputImages,
|
|
});
|
|
const result: ImageGenerationResult = await provider.generateImage({
|
|
provider: candidate.provider,
|
|
model: candidate.model,
|
|
prompt: params.prompt,
|
|
cfg: params.cfg,
|
|
agentDir: params.agentDir,
|
|
authStore: params.authStore,
|
|
count: params.count,
|
|
size: sanitized.size,
|
|
aspectRatio: sanitized.aspectRatio,
|
|
resolution: sanitized.resolution,
|
|
inputImages: params.inputImages,
|
|
});
|
|
if (!Array.isArray(result.images) || result.images.length === 0) {
|
|
throw new Error("Image generation provider returned no images.");
|
|
}
|
|
return {
|
|
images: result.images,
|
|
provider: candidate.provider,
|
|
model: result.model ?? candidate.model,
|
|
attempts,
|
|
normalization: sanitized.normalization,
|
|
metadata: {
|
|
...result.metadata,
|
|
...buildMediaGenerationNormalizationMetadata({
|
|
normalization: sanitized.normalization,
|
|
requestedSizeForDerivedAspectRatio: params.size,
|
|
}),
|
|
},
|
|
ignoredOverrides: sanitized.ignoredOverrides,
|
|
};
|
|
} catch (err) {
|
|
lastError = err;
|
|
const described = isFailoverError(err) ? describeFailoverError(err) : undefined;
|
|
attempts.push({
|
|
provider: candidate.provider,
|
|
model: candidate.model,
|
|
error: described?.message ?? formatErrorMessage(err),
|
|
reason: described?.reason,
|
|
status: described?.status,
|
|
code: described?.code,
|
|
});
|
|
log.debug(`image-generation candidate failed: ${candidate.provider}/${candidate.model}`);
|
|
}
|
|
}
|
|
|
|
throwCapabilityGenerationFailure({
|
|
capabilityLabel: "image generation",
|
|
attempts,
|
|
lastError,
|
|
});
|
|
}
|