mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-28 06:49:37 +00:00
* refactor: extract agent core package Introduce packages/agent-core as the OpenClaw-owned home for reusable agent loop, harness, session, prompt, and runtime dependency contracts. * refactor: extract shared llm runtime Move provider model registries, stream wrappers, OAuth helpers, and LLM utilities into src/llm with plugin-sdk barrels instead of depending on the old embedded runtime layout. * refactor: remove pi runtime internals Rename remaining Pi-shaped agent surfaces to OpenClaw agent runtime names, delete obsolete Pi docs and package graph checks, and add the third-party notice for incorporated code. * refactor: tighten agent session runtime Make agent-core/runtime dependencies explicit, consolidate compaction and session transcript helpers, and move model/session helpers behind OpenClaw-owned contracts. * refactor: remove static model and pi auth paths Drop static model catalogs and Pi auth bridges, move model/provider facts to manifest-owned runtime contracts, and harden internal embedded-agent utilities. * refactor: remove legacy provider compat paths * docs: remove agent parity notes * fix: skip provider wildcard metadata parsing * refactor: share session extension sdk loading * refactor: inline acpx proxy error formatter * refactor: fold edit recovery into edit tool * fix: accept extension batch separator * test: align startup provider plugin expectations * fix: restore provider-scoped release discovery * test: align static asset packaging expectations * fix: run static provider catalogs during scoped discovery * fix: add provider entry catalogs for scoped live discovery * fix: load lightweight provider catalog entries * fix: refresh provider-scoped plugin metadata * fix: keep provider catalog entries on release live path * fix: keep static manifest models in release live checks * fix: harden release model discovery * fix: reduce OpenAI live cache probe reasoning * fix: disable OpenAI cache probe reasoning * ci: extend OpenAI gateway live timeout * fix: extend live gateway model budget * fix: stabilize release validation regressions * fix: honor provider aliases in model rows * fix: stabilize release validation lanes * fix: stabilize release memory qa * ci: stabilize release validation lanes * ci: prefer ipv4 for live docker node calls * fix: restore shared tool-call stream wrapper * ci: remove legacy pi test shard alias * fix: clean up embedded agent test drift * fix: stabilize runtime alias status * fix: clean up embedded agent ci drift * fix: restore release ci invariants * fix: clean up post-rebase runtime drift * fix: restore release ci checks * fix: restore release ci after rebase * fix: remove stale pi runtime path * test: align compaction runtime expectations * test: update plugin prerelease expectations * fix: handle claude live tool approvals * fix: stabilize release validation gates * fix: finish agent runtime import * test: finish post-rebase agent runtime mocks * fix: keep codex compaction native * fix: stabilize codex app-server hook tests * test: isolate codex diagnostic active run * test: remove codex diagnostic completion race # Conflicts: # extensions/codex/src/app-server/run-attempt.test.ts * ci: fix full release manifest performance run id * refactor: narrow llm plugin sdk boundary * chore: drop generated google boundary stamps * fix: repair rebase fallout * fix: clean up rebased runtime references * fix: decode codex jwt payloads as base64url * fix: preserve shipped pi runtime alias * fix: add scoped sdk virtual modules * fix: decode llm codex oauth jwt as base64url * fix: avoid stale vertex adc negative cache * fix: harden tool arg decoding and codeql path * fix: keep vertex adc negative checks live * refactor: consolidate codex jwt and edit helpers * fix: await codex oauth node runtime imports * fix: preserve sdk tool and notice contracts * fix: preserve shipped compat config boundaries * fix: align codex oauth callback host * fix: terminate agent-core loop streams on failure * fix: keep codex oauth callback alive during fallback * ci: include session tools in critical codeql scans * fix: keep Cloudflare Anthropic provider auth header * docs: redirect legacy pi runtime pages * fix: honor bundled web provider compat discovery * fix: protect session output spill files * fix: keep legacy agent dir env blocked * fix: contain auto-discovered skill symlinks * fix: harden agent core sdk proxy surfaces * fix: restore approval reaction sdk compat * fix: keep live docker runs bounded * fix: keep codex oauth redirect host aligned * fix: resolve post-rebase agent runtime drift * fix: redact anthropic oauth parse failures * fix: preserve responses strict tool shaping * fix: repair agent runtime rebase cleanup * docs: redirect retired parity pages * fix: bound auto-discovered resources to roots * fix: repair post-rebase agent test drift * fix: preserve bundled provider allowlist migration * fix: preserve manifest-owned provider aliases * fix: declare photon image dependency * fix: keep provider headers out of proxy body * fix: preserve shipped env aliases * fix: refresh control ui i18n generated state * fix: quote read fallback paths * fix: preview edits through configured backend * test: satisfy core test typecheck * fix: preserve ZAI usage auth fallback * test: repair codex diagnostic test * fix: repair agent runtime rebase drift * test: finish embedded runner import rename * fix: repair agent runtime rebase integrations * test: align compaction oauth fallback expectations * fix: allow sdk-auth session models * fix: update doctor tool schema import * fix: preserve bedrock plugin region * fix: stream harmony-like prose immediately * ci: include session runtime in codeql shards * fix: repair latest rebase integrations * fix: honor explicit codex websocket transport * fix: keep openai-compatible credentials provider-scoped * fix: refresh sdk api baseline after rebase * fix: route cli runtime aliases through openclaw harness * test: rename stale harness mock expectation * test: rename embedded agent overflow calls * test: clean embedded auth test wording * test: use openclaw stream types in deepinfra cache test * fix: refresh sdk api baseline on latest main * fix: honor bundled discovery compat allowlists * fix: refresh sdk api baseline after latest rebase * fix: remove stale rebase imports * test: rename stale model catalog mock * test: mock renamed doctor runtime modules * fix: map canonical kimi env auth * fix: use internal model registry in bench script * fix: migrate deepinfra provider catalog entry * fix: enforce builtin tool suppression * fix: route compaction auth and proxy payloads safely * refactor: prune unused llm registry leftovers * test: update codex hooks session import * test: fix model picker ci coverage * test: align model picker auth mock types
312 lines
10 KiB
TypeScript
312 lines
10 KiB
TypeScript
import {
|
|
registerProviderPlugin,
|
|
requireRegisteredProvider,
|
|
} from "openclaw/plugin-sdk/plugin-test-runtime";
|
|
import { describe, expect, it } from "vitest";
|
|
import { resolveDefaultAgentDir } from "../src/agents/agent-scope.js";
|
|
import { isBillingErrorMessage } from "../src/agents/embedded-agent-helpers/failover-matches.js";
|
|
import { collectProviderApiKeys } from "../src/agents/live-auth-keys.js";
|
|
import { isLiveProfileKeyModeEnabled, isLiveTestEnabled } from "../src/agents/live-test-helpers.js";
|
|
import { resolveApiKeyForProvider } from "../src/agents/model-auth.js";
|
|
import { loadConfig, type OpenClawConfig } from "../src/config/config.js";
|
|
import {
|
|
DEFAULT_LIVE_IMAGE_MODELS,
|
|
parseCaseFilter,
|
|
parseCsvFilter,
|
|
parseProviderModelMap,
|
|
redactLiveApiKey,
|
|
resolveConfiguredLiveImageModels,
|
|
resolveLiveImageAuthStore,
|
|
} from "../src/image-generation/live-test-helpers.js";
|
|
import { isTruthyEnvValue } from "../src/infra/env.js";
|
|
import { getShellEnvAppliedKeys } from "../src/infra/shell-env.js";
|
|
import { encodePngRgba, fillPixel } from "../src/media/png-encode.js";
|
|
import { maybeLoadShellEnvForGenerationProviders } from "../src/test-utils/generation-live-test-helpers.js";
|
|
import { loadBundledProviderPlugin as loadBundledProviderPluginFromTestHelper } from "./helpers/media-generation/bundled-provider-builders.js";
|
|
|
|
const LIVE = isLiveTestEnabled();
|
|
const REQUIRE_PROFILE_KEYS =
|
|
isLiveProfileKeyModeEnabled() || isTruthyEnvValue(process.env.OPENCLAW_LIVE_REQUIRE_PROFILE_KEYS);
|
|
const describeLive = LIVE ? describe : describe.skip;
|
|
const providerFilter = parseCsvFilter(process.env.OPENCLAW_LIVE_IMAGE_GENERATION_PROVIDERS);
|
|
const caseFilter = parseCaseFilter(process.env.OPENCLAW_LIVE_IMAGE_GENERATION_CASES);
|
|
const envModelMap = parseProviderModelMap(process.env.OPENCLAW_LIVE_IMAGE_GENERATION_MODELS);
|
|
const DEFAULT_LIVE_IMAGE_GENERATION_TIMEOUT_MS = 120_000;
|
|
const LIVE_IMAGE_GENERATION_TIMEOUT_MS = resolvePositiveIntegerEnv(
|
|
process.env.OPENCLAW_LIVE_IMAGE_GENERATION_TIMEOUT_MS,
|
|
DEFAULT_LIVE_IMAGE_GENERATION_TIMEOUT_MS,
|
|
);
|
|
|
|
type LiveProviderCase = {
|
|
pluginId: string;
|
|
pluginName: string;
|
|
providerId: string;
|
|
};
|
|
|
|
type LiveImageCase = {
|
|
id: string;
|
|
providerId: string;
|
|
modelRef: string;
|
|
prompt: string;
|
|
size?: string;
|
|
resolution?: "1K" | "2K" | "4K";
|
|
inputImages?: Array<{ buffer: Buffer; mimeType: string; fileName?: string }>;
|
|
};
|
|
|
|
function resolvePositiveIntegerEnv(raw: string | undefined, fallback: number): number {
|
|
if (!raw) {
|
|
return fallback;
|
|
}
|
|
const parsed = Number(raw);
|
|
return Number.isFinite(parsed) && parsed > 0 ? Math.floor(parsed) : fallback;
|
|
}
|
|
|
|
function loadBundledProviderPlugin(
|
|
pluginId: string,
|
|
): ReturnType<typeof loadBundledProviderPluginFromTestHelper> {
|
|
return loadBundledProviderPluginFromTestHelper(pluginId);
|
|
}
|
|
|
|
const PROVIDER_CASES: LiveProviderCase[] = [
|
|
{
|
|
pluginId: "deepinfra",
|
|
pluginName: "DeepInfra Provider",
|
|
providerId: "deepinfra",
|
|
},
|
|
{
|
|
pluginId: "fal",
|
|
pluginName: "fal Provider",
|
|
providerId: "fal",
|
|
},
|
|
{
|
|
pluginId: "google",
|
|
pluginName: "Google Provider",
|
|
providerId: "google",
|
|
},
|
|
{
|
|
pluginId: "minimax",
|
|
pluginName: "MiniMax Provider",
|
|
providerId: "minimax",
|
|
},
|
|
{
|
|
pluginId: "openai",
|
|
pluginName: "OpenAI Provider",
|
|
providerId: "openai",
|
|
},
|
|
{
|
|
pluginId: "openrouter",
|
|
pluginName: "OpenRouter Provider",
|
|
providerId: "openrouter",
|
|
},
|
|
{
|
|
pluginId: "vydra",
|
|
pluginName: "Vydra Provider",
|
|
providerId: "vydra",
|
|
},
|
|
{
|
|
pluginId: "xai",
|
|
pluginName: "xAI Provider",
|
|
providerId: "xai",
|
|
},
|
|
]
|
|
.filter((entry) => (providerFilter ? providerFilter.has(entry.providerId) : true))
|
|
.toSorted((left, right) => left.providerId.localeCompare(right.providerId));
|
|
|
|
function createEditReferencePng(): Buffer {
|
|
const width = 192;
|
|
const height = 192;
|
|
const buf = Buffer.alloc(width * height * 4, 255);
|
|
|
|
for (let y = 0; y < height; y += 1) {
|
|
for (let x = 0; x < width; x += 1) {
|
|
fillPixel(buf, x, y, width, 245, 248, 255, 255);
|
|
}
|
|
}
|
|
|
|
for (let y = 24; y < 168; y += 1) {
|
|
for (let x = 24; x < 168; x += 1) {
|
|
fillPixel(buf, x, y, width, 255, 189, 89, 255);
|
|
}
|
|
}
|
|
|
|
for (let y = 48; y < 144; y += 1) {
|
|
for (let x = 48; x < 144; x += 1) {
|
|
fillPixel(buf, x, y, width, 41, 47, 54, 255);
|
|
}
|
|
}
|
|
|
|
return encodePngRgba(buf, width, height);
|
|
}
|
|
|
|
function withPluginsEnabled(cfg: OpenClawConfig): OpenClawConfig {
|
|
return {
|
|
...cfg,
|
|
plugins: {
|
|
...cfg.plugins,
|
|
enabled: true,
|
|
},
|
|
};
|
|
}
|
|
|
|
function resolveProviderModelForLiveTest(providerId: string, modelRef: string): string {
|
|
const slash = modelRef.indexOf("/");
|
|
if (slash <= 0 || slash === modelRef.length - 1) {
|
|
return modelRef;
|
|
}
|
|
return modelRef.slice(0, slash) === providerId ? modelRef.slice(slash + 1) : modelRef;
|
|
}
|
|
|
|
function buildLiveCases(params: {
|
|
providerId: string;
|
|
modelRef: string;
|
|
editEnabled: boolean;
|
|
}): LiveImageCase[] {
|
|
const generatePrompt =
|
|
"Create a minimal flat illustration of an orange cat face sticker on a white background.";
|
|
const editPrompt =
|
|
"Change ONLY the background to a pale blue gradient. Keep the subject, framing, and style identical.";
|
|
const cases: LiveImageCase[] = [
|
|
{
|
|
id: `${params.providerId}:generate`,
|
|
providerId: params.providerId,
|
|
modelRef: params.modelRef,
|
|
prompt: generatePrompt,
|
|
size: "1024x1024",
|
|
},
|
|
];
|
|
if (params.editEnabled) {
|
|
cases.push({
|
|
id: `${params.providerId}:edit`,
|
|
providerId: params.providerId,
|
|
modelRef: params.modelRef,
|
|
prompt: editPrompt,
|
|
resolution: "1K",
|
|
inputImages: [
|
|
{
|
|
buffer: createEditReferencePng(),
|
|
mimeType: "image/png",
|
|
fileName: "reference.png",
|
|
},
|
|
],
|
|
});
|
|
}
|
|
return cases;
|
|
}
|
|
|
|
describeLive("image generation live (provider sweep)", () => {
|
|
it(
|
|
"generates images for every configured image-generation variant with available auth",
|
|
async () => {
|
|
const cfg = withPluginsEnabled(loadConfig());
|
|
const configuredModels = resolveConfiguredLiveImageModels(cfg);
|
|
const agentDir = resolveDefaultAgentDir(cfg);
|
|
const attempted: string[] = [];
|
|
const skipped: string[] = [];
|
|
const failures: string[] = [];
|
|
|
|
maybeLoadShellEnvForGenerationProviders(PROVIDER_CASES.map((entry) => entry.providerId));
|
|
|
|
for (const providerCase of PROVIDER_CASES) {
|
|
const modelRef =
|
|
envModelMap.get(providerCase.providerId) ??
|
|
configuredModels.get(providerCase.providerId) ??
|
|
DEFAULT_LIVE_IMAGE_MODELS[providerCase.providerId];
|
|
if (!modelRef) {
|
|
skipped.push(`${providerCase.providerId}: no model configured`);
|
|
continue;
|
|
}
|
|
|
|
const hasLiveKeys = collectProviderApiKeys(providerCase.providerId).length > 0;
|
|
const authStore = resolveLiveImageAuthStore({
|
|
requireProfileKeys: REQUIRE_PROFILE_KEYS,
|
|
hasLiveKeys,
|
|
});
|
|
let authLabel = "unresolved";
|
|
try {
|
|
const auth = await resolveApiKeyForProvider({
|
|
provider: providerCase.providerId,
|
|
cfg,
|
|
agentDir,
|
|
store: authStore,
|
|
});
|
|
authLabel = `${auth.source} ${redactLiveApiKey(auth.apiKey)}`;
|
|
} catch {
|
|
skipped.push(`${providerCase.providerId}: no usable auth`);
|
|
continue;
|
|
}
|
|
|
|
const { imageProviders } = await registerProviderPlugin({
|
|
plugin: loadBundledProviderPlugin(providerCase.pluginId),
|
|
id: providerCase.pluginId,
|
|
name: providerCase.pluginName,
|
|
});
|
|
const provider = requireRegisteredProvider(
|
|
imageProviders,
|
|
providerCase.providerId,
|
|
"image provider",
|
|
);
|
|
const providerModel = resolveProviderModelForLiveTest(providerCase.providerId, modelRef);
|
|
const liveCases = buildLiveCases({
|
|
providerId: providerCase.providerId,
|
|
modelRef,
|
|
editEnabled: provider.capabilities.edit?.enabled ?? false,
|
|
}).filter((entry) => (caseFilter ? caseFilter.has(entry.id.toLowerCase()) : true));
|
|
|
|
for (const testCase of liveCases) {
|
|
const startedAt = Date.now();
|
|
console.error(
|
|
`[live:image-generation] starting ${testCase.id} model=${providerModel} auth=${authLabel}`,
|
|
);
|
|
try {
|
|
const result = await provider.generateImage({
|
|
provider: providerCase.providerId,
|
|
model: providerModel,
|
|
prompt: testCase.prompt,
|
|
cfg,
|
|
agentDir,
|
|
authStore,
|
|
size: testCase.size,
|
|
resolution: testCase.resolution,
|
|
inputImages: testCase.inputImages,
|
|
timeoutMs: LIVE_IMAGE_GENERATION_TIMEOUT_MS,
|
|
});
|
|
|
|
expect(result.images.length).toBeGreaterThan(0);
|
|
expect(result.images[0]?.mimeType.startsWith("image/")).toBe(true);
|
|
expect(result.images[0]?.buffer.byteLength).toBeGreaterThan(512);
|
|
attempted.push(`${testCase.id}:${result.model} (${authLabel})`);
|
|
console.error(
|
|
`[live:image-generation] done ${testCase.id} ms=${Date.now() - startedAt} images=${result.images.length}`,
|
|
);
|
|
} catch (error) {
|
|
const message = error instanceof Error ? error.message : String(error);
|
|
if (isBillingErrorMessage(message)) {
|
|
skipped.push(`${testCase.id} (${authLabel}): billing drift`);
|
|
console.warn(
|
|
`[live:image-generation] skip ${testCase.id} ms=${Date.now() - startedAt} reason=billing drift error=${message}`,
|
|
);
|
|
continue;
|
|
}
|
|
failures.push(`${testCase.id} (${authLabel}): ${message}`);
|
|
console.error(
|
|
`[live:image-generation] failed ${testCase.id} ms=${Date.now() - startedAt} error=${message}`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
console.log(
|
|
`[live:image-generation] attempted=${attempted.join(", ") || "none"} skipped=${skipped.join(", ") || "none"} failures=${failures.join(" | ") || "none"} shellEnv=${getShellEnvAppliedKeys().join(", ") || "none"}`,
|
|
);
|
|
|
|
if (attempted.length === 0) {
|
|
expect(failures).toStrictEqual([]);
|
|
console.warn("[live:image-generation] no provider had usable auth; skipping assertions");
|
|
return;
|
|
}
|
|
expect(failures).toStrictEqual([]);
|
|
},
|
|
15 * 60_000,
|
|
);
|
|
});
|