2026-01-16 23:12:50 +00:00
|
|
|
import { afterEach, describe, expect, it, vi } from "vitest";
|
2026-03-14 02:53:57 -07:00
|
|
|
import type { OpenClawConfig } from "../../../src/config/config.js";
|
2026-01-16 23:12:50 +00:00
|
|
|
import { resolveDiscordToken } from "./token.js";
|
|
|
|
|
|
|
|
|
|
describe("resolveDiscordToken", () => {
|
|
|
|
|
afterEach(() => {
|
|
|
|
|
vi.unstubAllEnvs();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("prefers config token over env", () => {
|
|
|
|
|
vi.stubEnv("DISCORD_BOT_TOKEN", "env-token");
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: { discord: { token: "cfg-token" } },
|
2026-01-30 03:15:10 +01:00
|
|
|
} as OpenClawConfig;
|
2026-01-16 23:12:50 +00:00
|
|
|
const res = resolveDiscordToken(cfg);
|
|
|
|
|
expect(res.token).toBe("cfg-token");
|
|
|
|
|
expect(res.source).toBe("config");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("uses env token when config is missing", () => {
|
|
|
|
|
vi.stubEnv("DISCORD_BOT_TOKEN", "env-token");
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: { discord: {} },
|
2026-01-30 03:15:10 +01:00
|
|
|
} as OpenClawConfig;
|
2026-01-16 23:12:50 +00:00
|
|
|
const res = resolveDiscordToken(cfg);
|
|
|
|
|
expect(res.token).toBe("env-token");
|
|
|
|
|
expect(res.source).toBe("env");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("prefers account token for non-default accounts", () => {
|
|
|
|
|
vi.stubEnv("DISCORD_BOT_TOKEN", "env-token");
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: {
|
|
|
|
|
discord: {
|
|
|
|
|
token: "base-token",
|
|
|
|
|
accounts: {
|
|
|
|
|
work: { token: "acct-token" },
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
2026-01-30 03:15:10 +01:00
|
|
|
} as OpenClawConfig;
|
2026-01-16 23:12:50 +00:00
|
|
|
const res = resolveDiscordToken(cfg, { accountId: "work" });
|
|
|
|
|
expect(res.token).toBe("acct-token");
|
|
|
|
|
expect(res.source).toBe("config");
|
|
|
|
|
});
|
2026-03-02 20:58:20 -06:00
|
|
|
|
|
|
|
|
it("falls back to top-level token for non-default accounts without account token", () => {
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: {
|
|
|
|
|
discord: {
|
|
|
|
|
token: "base-token",
|
|
|
|
|
accounts: {
|
|
|
|
|
work: {},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
} as OpenClawConfig;
|
|
|
|
|
const res = resolveDiscordToken(cfg, { accountId: "work" });
|
|
|
|
|
expect(res.token).toBe("base-token");
|
|
|
|
|
expect(res.source).toBe("config");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("does not inherit top-level token when account token is explicitly blank", () => {
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: {
|
|
|
|
|
discord: {
|
|
|
|
|
token: "base-token",
|
|
|
|
|
accounts: {
|
|
|
|
|
work: { token: "" },
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
} as OpenClawConfig;
|
|
|
|
|
const res = resolveDiscordToken(cfg, { accountId: "work" });
|
|
|
|
|
expect(res.token).toBe("");
|
|
|
|
|
expect(res.source).toBe("none");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("resolves account token when account key casing differs from normalized id", () => {
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: {
|
|
|
|
|
discord: {
|
|
|
|
|
accounts: {
|
|
|
|
|
Work: { token: "acct-token" },
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
} as OpenClawConfig;
|
|
|
|
|
const res = resolveDiscordToken(cfg, { accountId: "work" });
|
|
|
|
|
expect(res.token).toBe("acct-token");
|
|
|
|
|
expect(res.source).toBe("config");
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it("throws when token is an unresolved SecretRef object", () => {
|
|
|
|
|
const cfg = {
|
|
|
|
|
channels: {
|
|
|
|
|
discord: {
|
|
|
|
|
token: { source: "env", provider: "default", id: "DISCORD_BOT_TOKEN" },
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
} as unknown as OpenClawConfig;
|
|
|
|
|
|
|
|
|
|
expect(() => resolveDiscordToken(cfg)).toThrow(
|
|
|
|
|
/channels\.discord\.token: unresolved SecretRef/i,
|
|
|
|
|
);
|
|
|
|
|
});
|
2026-01-16 23:12:50 +00:00
|
|
|
});
|