mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-13 19:10:39 +00:00
431 lines
11 KiB
TypeScript
431 lines
11 KiB
TypeScript
import path from "node:path";
|
|
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
import type { OpenClawConfig } from "../config/config.js";
|
|
import {
|
|
hasConfiguredModelFallbacks,
|
|
resolveAgentConfig,
|
|
resolveAgentDir,
|
|
resolveAgentEffectiveModelPrimary,
|
|
resolveAgentExplicitModelPrimary,
|
|
resolveFallbackAgentId,
|
|
resolveEffectiveModelFallbacks,
|
|
resolveAgentModelFallbacksOverride,
|
|
resolveAgentModelPrimary,
|
|
resolveRunModelFallbacksOverride,
|
|
resolveAgentWorkspaceDir,
|
|
} from "./agent-scope.js";
|
|
|
|
afterEach(() => {
|
|
vi.unstubAllEnvs();
|
|
});
|
|
|
|
describe("resolveAgentConfig", () => {
|
|
it("should return undefined when no agents config exists", () => {
|
|
const cfg: OpenClawConfig = {};
|
|
const result = resolveAgentConfig(cfg, "main");
|
|
expect(result).toBeUndefined();
|
|
});
|
|
|
|
it("should return undefined when agent id does not exist", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
list: [{ id: "main", workspace: "~/openclaw" }],
|
|
},
|
|
};
|
|
const result = resolveAgentConfig(cfg, "nonexistent");
|
|
expect(result).toBeUndefined();
|
|
});
|
|
|
|
it("should return basic agent config", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
list: [
|
|
{
|
|
id: "main",
|
|
name: "Main Agent",
|
|
workspace: "~/openclaw",
|
|
agentDir: "~/.openclaw/agents/main",
|
|
model: "anthropic/claude-opus-4",
|
|
},
|
|
],
|
|
},
|
|
};
|
|
const result = resolveAgentConfig(cfg, "main");
|
|
expect(result).toEqual({
|
|
name: "Main Agent",
|
|
workspace: "~/openclaw",
|
|
agentDir: "~/.openclaw/agents/main",
|
|
model: "anthropic/claude-opus-4",
|
|
identity: undefined,
|
|
groupChat: undefined,
|
|
subagents: undefined,
|
|
sandbox: undefined,
|
|
tools: undefined,
|
|
});
|
|
});
|
|
|
|
it("resolves explicit and effective model primary separately", () => {
|
|
const cfgWithStringDefault = {
|
|
agents: {
|
|
defaults: {
|
|
model: "anthropic/claude-sonnet-4",
|
|
},
|
|
list: [{ id: "main" }],
|
|
},
|
|
} as unknown as OpenClawConfig;
|
|
expect(resolveAgentExplicitModelPrimary(cfgWithStringDefault, "main")).toBeUndefined();
|
|
expect(resolveAgentEffectiveModelPrimary(cfgWithStringDefault, "main")).toBe(
|
|
"anthropic/claude-sonnet-4",
|
|
);
|
|
|
|
const cfgWithObjectDefault: OpenClawConfig = {
|
|
agents: {
|
|
defaults: {
|
|
model: {
|
|
primary: "openai/gpt-5.2",
|
|
fallbacks: ["anthropic/claude-sonnet-4"],
|
|
},
|
|
},
|
|
list: [{ id: "main" }],
|
|
},
|
|
};
|
|
expect(resolveAgentExplicitModelPrimary(cfgWithObjectDefault, "main")).toBeUndefined();
|
|
expect(resolveAgentEffectiveModelPrimary(cfgWithObjectDefault, "main")).toBe("openai/gpt-5.2");
|
|
|
|
const cfgNoDefaults: OpenClawConfig = {
|
|
agents: {
|
|
list: [{ id: "main" }],
|
|
},
|
|
};
|
|
expect(resolveAgentExplicitModelPrimary(cfgNoDefaults, "main")).toBeUndefined();
|
|
expect(resolveAgentEffectiveModelPrimary(cfgNoDefaults, "main")).toBeUndefined();
|
|
});
|
|
|
|
it("supports per-agent model primary+fallbacks", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
defaults: {
|
|
model: {
|
|
primary: "anthropic/claude-sonnet-4",
|
|
fallbacks: ["openai/gpt-4.1"],
|
|
},
|
|
},
|
|
list: [
|
|
{
|
|
id: "linus",
|
|
model: {
|
|
primary: "anthropic/claude-opus-4",
|
|
fallbacks: ["openai/gpt-5.2"],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
|
|
expect(resolveAgentModelPrimary(cfg, "linus")).toBe("anthropic/claude-opus-4");
|
|
expect(resolveAgentExplicitModelPrimary(cfg, "linus")).toBe("anthropic/claude-opus-4");
|
|
expect(resolveAgentEffectiveModelPrimary(cfg, "linus")).toBe("anthropic/claude-opus-4");
|
|
expect(resolveAgentModelFallbacksOverride(cfg, "linus")).toEqual(["openai/gpt-5.2"]);
|
|
|
|
// If fallbacks isn't present, we don't override the global fallbacks.
|
|
const cfgNoOverride: OpenClawConfig = {
|
|
agents: {
|
|
list: [
|
|
{
|
|
id: "linus",
|
|
model: {
|
|
primary: "anthropic/claude-opus-4",
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
expect(resolveAgentModelFallbacksOverride(cfgNoOverride, "linus")).toBe(undefined);
|
|
|
|
// Explicit empty list disables global fallbacks for that agent.
|
|
const cfgDisable: OpenClawConfig = {
|
|
agents: {
|
|
list: [
|
|
{
|
|
id: "linus",
|
|
model: {
|
|
primary: "anthropic/claude-opus-4",
|
|
fallbacks: [],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
expect(resolveAgentModelFallbacksOverride(cfgDisable, "linus")).toEqual([]);
|
|
|
|
expect(
|
|
resolveEffectiveModelFallbacks({
|
|
cfg,
|
|
agentId: "linus",
|
|
hasSessionModelOverride: false,
|
|
}),
|
|
).toEqual(["openai/gpt-5.2"]);
|
|
expect(
|
|
resolveEffectiveModelFallbacks({
|
|
cfg,
|
|
agentId: "linus",
|
|
hasSessionModelOverride: true,
|
|
}),
|
|
).toEqual(["openai/gpt-5.2"]);
|
|
expect(
|
|
resolveEffectiveModelFallbacks({
|
|
cfg: cfgNoOverride,
|
|
agentId: "linus",
|
|
hasSessionModelOverride: true,
|
|
}),
|
|
).toEqual([]);
|
|
|
|
const cfgInheritDefaults: OpenClawConfig = {
|
|
agents: {
|
|
defaults: {
|
|
model: {
|
|
fallbacks: ["openai/gpt-4.1"],
|
|
},
|
|
},
|
|
list: [
|
|
{
|
|
id: "linus",
|
|
model: {
|
|
primary: "anthropic/claude-opus-4",
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
expect(
|
|
resolveEffectiveModelFallbacks({
|
|
cfg: cfgInheritDefaults,
|
|
agentId: "linus",
|
|
hasSessionModelOverride: true,
|
|
}),
|
|
).toEqual(["openai/gpt-4.1"]);
|
|
expect(
|
|
resolveEffectiveModelFallbacks({
|
|
cfg: cfgDisable,
|
|
agentId: "linus",
|
|
hasSessionModelOverride: true,
|
|
}),
|
|
).toEqual([]);
|
|
});
|
|
|
|
it("resolves fallback agent id from explicit agent id first", () => {
|
|
expect(
|
|
resolveFallbackAgentId({
|
|
agentId: "Support",
|
|
sessionKey: "agent:main:session",
|
|
}),
|
|
).toBe("support");
|
|
});
|
|
|
|
it("resolves fallback agent id from session key when explicit id is missing", () => {
|
|
expect(
|
|
resolveFallbackAgentId({
|
|
sessionKey: "agent:worker:session",
|
|
}),
|
|
).toBe("worker");
|
|
});
|
|
|
|
it("resolves run fallback overrides via shared helper", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
defaults: {
|
|
model: {
|
|
fallbacks: ["openai/gpt-4.1"],
|
|
},
|
|
},
|
|
list: [
|
|
{
|
|
id: "support",
|
|
model: {
|
|
fallbacks: ["openai/gpt-5.2"],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
|
|
expect(
|
|
resolveRunModelFallbacksOverride({
|
|
cfg,
|
|
agentId: "support",
|
|
sessionKey: "agent:main:session",
|
|
}),
|
|
).toEqual(["openai/gpt-5.2"]);
|
|
expect(
|
|
resolveRunModelFallbacksOverride({
|
|
cfg,
|
|
agentId: undefined,
|
|
sessionKey: "agent:support:session",
|
|
}),
|
|
).toEqual(["openai/gpt-5.2"]);
|
|
});
|
|
|
|
it("computes whether any model fallbacks are configured via shared helper", () => {
|
|
const cfgDefaultsOnly: OpenClawConfig = {
|
|
agents: {
|
|
defaults: {
|
|
model: {
|
|
fallbacks: ["openai/gpt-4.1"],
|
|
},
|
|
},
|
|
list: [{ id: "main" }],
|
|
},
|
|
};
|
|
expect(
|
|
hasConfiguredModelFallbacks({
|
|
cfg: cfgDefaultsOnly,
|
|
sessionKey: "agent:main:session",
|
|
}),
|
|
).toBe(true);
|
|
|
|
const cfgAgentOverrideOnly: OpenClawConfig = {
|
|
agents: {
|
|
defaults: {
|
|
model: {
|
|
fallbacks: [],
|
|
},
|
|
},
|
|
list: [
|
|
{
|
|
id: "support",
|
|
model: {
|
|
fallbacks: ["openai/gpt-5.2"],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
expect(
|
|
hasConfiguredModelFallbacks({
|
|
cfg: cfgAgentOverrideOnly,
|
|
agentId: "support",
|
|
sessionKey: "agent:support:session",
|
|
}),
|
|
).toBe(true);
|
|
expect(
|
|
hasConfiguredModelFallbacks({
|
|
cfg: cfgAgentOverrideOnly,
|
|
agentId: "main",
|
|
sessionKey: "agent:main:session",
|
|
}),
|
|
).toBe(false);
|
|
});
|
|
|
|
it("should return agent-specific sandbox config", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
list: [
|
|
{
|
|
id: "work",
|
|
workspace: "~/openclaw-work",
|
|
sandbox: {
|
|
mode: "all",
|
|
scope: "agent",
|
|
perSession: false,
|
|
workspaceAccess: "ro",
|
|
workspaceRoot: "~/sandboxes",
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
const result = resolveAgentConfig(cfg, "work");
|
|
expect(result?.sandbox).toEqual({
|
|
mode: "all",
|
|
scope: "agent",
|
|
perSession: false,
|
|
workspaceAccess: "ro",
|
|
workspaceRoot: "~/sandboxes",
|
|
});
|
|
});
|
|
|
|
it("should return agent-specific tools config", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
list: [
|
|
{
|
|
id: "restricted",
|
|
workspace: "~/openclaw-restricted",
|
|
tools: {
|
|
allow: ["read"],
|
|
deny: ["exec", "write", "edit"],
|
|
elevated: {
|
|
enabled: false,
|
|
allowFrom: { whatsapp: ["+15555550123"] },
|
|
},
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
const result = resolveAgentConfig(cfg, "restricted");
|
|
expect(result?.tools).toEqual({
|
|
allow: ["read"],
|
|
deny: ["exec", "write", "edit"],
|
|
elevated: {
|
|
enabled: false,
|
|
allowFrom: { whatsapp: ["+15555550123"] },
|
|
},
|
|
});
|
|
});
|
|
|
|
it("should return both sandbox and tools config", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
list: [
|
|
{
|
|
id: "family",
|
|
workspace: "~/openclaw-family",
|
|
sandbox: {
|
|
mode: "all",
|
|
scope: "agent",
|
|
},
|
|
tools: {
|
|
allow: ["read"],
|
|
deny: ["exec"],
|
|
},
|
|
},
|
|
],
|
|
},
|
|
};
|
|
const result = resolveAgentConfig(cfg, "family");
|
|
expect(result?.sandbox?.mode).toBe("all");
|
|
expect(result?.tools?.allow).toEqual(["read"]);
|
|
});
|
|
|
|
it("should normalize agent id", () => {
|
|
const cfg: OpenClawConfig = {
|
|
agents: {
|
|
list: [{ id: "main", workspace: "~/openclaw" }],
|
|
},
|
|
};
|
|
// Should normalize to "main" (default)
|
|
const result = resolveAgentConfig(cfg, "");
|
|
expect(result).toBeDefined();
|
|
expect(result?.workspace).toBe("~/openclaw");
|
|
});
|
|
|
|
it("uses OPENCLAW_HOME for default agent workspace", () => {
|
|
const home = path.join(path.sep, "srv", "openclaw-home");
|
|
vi.stubEnv("OPENCLAW_HOME", home);
|
|
|
|
const workspace = resolveAgentWorkspaceDir({} as OpenClawConfig, "main");
|
|
expect(workspace).toBe(path.join(path.resolve(home), ".openclaw", "workspace"));
|
|
});
|
|
|
|
it("uses OPENCLAW_HOME for default agentDir", () => {
|
|
const home = path.join(path.sep, "srv", "openclaw-home");
|
|
vi.stubEnv("OPENCLAW_HOME", home);
|
|
// Clear state dir so it falls back to OPENCLAW_HOME
|
|
vi.stubEnv("OPENCLAW_STATE_DIR", "");
|
|
|
|
const agentDir = resolveAgentDir({} as OpenClawConfig, "main");
|
|
expect(agentDir).toBe(path.join(path.resolve(home), ".openclaw", "agents", "main", "agent"));
|
|
});
|
|
});
|