mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-02 23:20:21 +00:00
352 lines
10 KiB
TypeScript
352 lines
10 KiB
TypeScript
import { describe, expect, it } from "vitest";
|
|
import { MissingEnvVarError, resolveConfigEnvVars } from "./env-substitution.js";
|
|
|
|
describe("resolveConfigEnvVars", () => {
|
|
describe("basic substitution", () => {
|
|
it("substitutes direct, inline, repeated, and multi-var patterns", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
expected: unknown;
|
|
}> = [
|
|
{
|
|
name: "single env var",
|
|
config: { key: "${FOO}" },
|
|
env: { FOO: "bar" },
|
|
expected: { key: "bar" },
|
|
},
|
|
{
|
|
name: "multiple env vars in same string",
|
|
config: { key: "${A}/${B}" },
|
|
env: { A: "x", B: "y" },
|
|
expected: { key: "x/y" },
|
|
},
|
|
{
|
|
name: "inline prefix/suffix",
|
|
config: { key: "prefix-${FOO}-suffix" },
|
|
env: { FOO: "bar" },
|
|
expected: { key: "prefix-bar-suffix" },
|
|
},
|
|
{
|
|
name: "same var repeated",
|
|
config: { key: "${FOO}:${FOO}" },
|
|
env: { FOO: "bar" },
|
|
expected: { key: "bar:bar" },
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
const result = resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect(result, scenario.name).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("nested structures", () => {
|
|
it("substitutes variables in nested objects and arrays", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
expected: unknown;
|
|
}> = [
|
|
{
|
|
name: "nested object",
|
|
config: { outer: { inner: { key: "${API_KEY}" } } },
|
|
env: { API_KEY: "secret123" },
|
|
expected: { outer: { inner: { key: "secret123" } } },
|
|
},
|
|
{
|
|
name: "flat array",
|
|
config: { items: ["${A}", "${B}", "${C}"] },
|
|
env: { A: "1", B: "2", C: "3" },
|
|
expected: { items: ["1", "2", "3"] },
|
|
},
|
|
{
|
|
name: "array of objects",
|
|
config: {
|
|
providers: [
|
|
{ name: "openai", apiKey: "${OPENAI_KEY}" },
|
|
{ name: "anthropic", apiKey: "${ANTHROPIC_KEY}" },
|
|
],
|
|
},
|
|
env: { OPENAI_KEY: "sk-xxx", ANTHROPIC_KEY: "sk-yyy" },
|
|
expected: {
|
|
providers: [
|
|
{ name: "openai", apiKey: "sk-xxx" },
|
|
{ name: "anthropic", apiKey: "sk-yyy" },
|
|
],
|
|
},
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
const result = resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect(result, scenario.name).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("missing env var handling", () => {
|
|
it("throws MissingEnvVarError with var name and config path details", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
varName: string;
|
|
configPath: string;
|
|
}> = [
|
|
{
|
|
name: "missing top-level var",
|
|
config: { key: "${MISSING}" },
|
|
env: {},
|
|
varName: "MISSING",
|
|
configPath: "key",
|
|
},
|
|
{
|
|
name: "missing nested var",
|
|
config: { outer: { inner: { key: "${MISSING_VAR}" } } },
|
|
env: {},
|
|
varName: "MISSING_VAR",
|
|
configPath: "outer.inner.key",
|
|
},
|
|
{
|
|
name: "missing var in array element",
|
|
config: { items: ["ok", "${MISSING}"] },
|
|
env: { OK: "val" },
|
|
varName: "MISSING",
|
|
configPath: "items[1]",
|
|
},
|
|
{
|
|
name: "empty string env value treated as missing",
|
|
config: { key: "${EMPTY}" },
|
|
env: { EMPTY: "" },
|
|
varName: "EMPTY",
|
|
configPath: "key",
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
try {
|
|
resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect.fail(`${scenario.name}: expected MissingEnvVarError`);
|
|
} catch (err) {
|
|
expect(err, scenario.name).toBeInstanceOf(MissingEnvVarError);
|
|
const error = err as MissingEnvVarError;
|
|
expect(error.varName, scenario.name).toBe(scenario.varName);
|
|
expect(error.configPath, scenario.name).toBe(scenario.configPath);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("escape syntax", () => {
|
|
it("handles escaped placeholders alongside regular substitutions", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
expected: unknown;
|
|
}> = [
|
|
{
|
|
name: "escaped placeholder stays literal",
|
|
config: { key: "$${VAR}" },
|
|
env: { VAR: "value" },
|
|
expected: { key: "${VAR}" },
|
|
},
|
|
{
|
|
name: "mix of escaped and unescaped vars",
|
|
config: { key: "${REAL}/$${LITERAL}" },
|
|
env: { REAL: "resolved" },
|
|
expected: { key: "resolved/${LITERAL}" },
|
|
},
|
|
{
|
|
name: "escaped first, unescaped second",
|
|
config: { key: "$${FOO} ${FOO}" },
|
|
env: { FOO: "bar" },
|
|
expected: { key: "${FOO} bar" },
|
|
},
|
|
{
|
|
name: "unescaped first, escaped second",
|
|
config: { key: "${FOO} $${FOO}" },
|
|
env: { FOO: "bar" },
|
|
expected: { key: "bar ${FOO}" },
|
|
},
|
|
{
|
|
name: "multiple escaped placeholders",
|
|
config: { key: "$${A}:$${B}" },
|
|
env: {},
|
|
expected: { key: "${A}:${B}" },
|
|
},
|
|
{
|
|
name: "env values are not unescaped",
|
|
config: { key: "${FOO}" },
|
|
env: { FOO: "$${BAR}" },
|
|
expected: { key: "$${BAR}" },
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
const result = resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect(result, scenario.name).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("pattern matching rules", () => {
|
|
it("leaves non-matching placeholders unchanged", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
expected: unknown;
|
|
}> = [
|
|
{
|
|
name: "$VAR (no braces)",
|
|
config: { key: "$VAR" },
|
|
env: { VAR: "value" },
|
|
expected: { key: "$VAR" },
|
|
},
|
|
{
|
|
name: "lowercase placeholder",
|
|
config: { key: "${lowercase}" },
|
|
env: { lowercase: "value" },
|
|
expected: { key: "${lowercase}" },
|
|
},
|
|
{
|
|
name: "mixed-case placeholder",
|
|
config: { key: "${MixedCase}" },
|
|
env: { MixedCase: "value" },
|
|
expected: { key: "${MixedCase}" },
|
|
},
|
|
{
|
|
name: "invalid numeric prefix",
|
|
config: { key: "${123INVALID}" },
|
|
env: {},
|
|
expected: { key: "${123INVALID}" },
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
const result = resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect(result, scenario.name).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
|
|
it("substitutes valid uppercase/underscore placeholder names", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
expected: unknown;
|
|
}> = [
|
|
{
|
|
name: "underscore-prefixed name",
|
|
config: { key: "${_UNDERSCORE_START}" },
|
|
env: { _UNDERSCORE_START: "valid" },
|
|
expected: { key: "valid" },
|
|
},
|
|
{
|
|
name: "name with numbers",
|
|
config: { key: "${VAR_WITH_NUMBERS_123}" },
|
|
env: { VAR_WITH_NUMBERS_123: "valid" },
|
|
expected: { key: "valid" },
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
const result = resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect(result, scenario.name).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("passthrough behavior", () => {
|
|
it("passes through primitives unchanged", () => {
|
|
for (const value of ["hello", 42, true, null]) {
|
|
expect(resolveConfigEnvVars(value, {})).toBe(value);
|
|
}
|
|
});
|
|
|
|
it("preserves empty and non-string containers", () => {
|
|
const scenarios: Array<{ config: unknown; expected: unknown }> = [
|
|
{ config: {}, expected: {} },
|
|
{ config: [], expected: [] },
|
|
{
|
|
config: { num: 42, bool: true, nil: null, arr: [1, 2] },
|
|
expected: { num: 42, bool: true, nil: null, arr: [1, 2] },
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
expect(resolveConfigEnvVars(scenario.config, {})).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("real-world config patterns", () => {
|
|
it("substitutes provider, gateway, and base URL config values", () => {
|
|
const scenarios: Array<{
|
|
name: string;
|
|
config: unknown;
|
|
env: Record<string, string>;
|
|
expected: unknown;
|
|
}> = [
|
|
{
|
|
name: "provider API keys",
|
|
config: {
|
|
models: {
|
|
providers: {
|
|
"vercel-gateway": { apiKey: "${VERCEL_GATEWAY_API_KEY}" },
|
|
openai: { apiKey: "${OPENAI_API_KEY}" },
|
|
},
|
|
},
|
|
},
|
|
env: {
|
|
VERCEL_GATEWAY_API_KEY: "vg_key_123",
|
|
OPENAI_API_KEY: "sk-xxx",
|
|
},
|
|
expected: {
|
|
models: {
|
|
providers: {
|
|
"vercel-gateway": { apiKey: "vg_key_123" },
|
|
openai: { apiKey: "sk-xxx" },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{
|
|
name: "gateway auth token",
|
|
config: { gateway: { auth: { token: "${OPENCLAW_GATEWAY_TOKEN}" } } },
|
|
env: { OPENCLAW_GATEWAY_TOKEN: "secret-token" },
|
|
expected: { gateway: { auth: { token: "secret-token" } } },
|
|
},
|
|
{
|
|
name: "provider base URL composition",
|
|
config: {
|
|
models: {
|
|
providers: {
|
|
custom: { baseUrl: "${CUSTOM_API_BASE}/v1" },
|
|
},
|
|
},
|
|
},
|
|
env: { CUSTOM_API_BASE: "https://api.example.com" },
|
|
expected: {
|
|
models: {
|
|
providers: {
|
|
custom: { baseUrl: "https://api.example.com/v1" },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
];
|
|
|
|
for (const scenario of scenarios) {
|
|
const result = resolveConfigEnvVars(scenario.config, scenario.env);
|
|
expect(result, scenario.name).toEqual(scenario.expected);
|
|
}
|
|
});
|
|
});
|
|
});
|