mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-02 04:41:11 +00:00
616 lines
17 KiB
TypeScript
616 lines
17 KiB
TypeScript
import { describe, expect, it } from "vitest";
|
|
import { formatPairingApproveHint } from "../channels/plugins/helpers.js";
|
|
import { DEFAULT_ACCOUNT_ID } from "../routing/session-key.js";
|
|
import {
|
|
adaptScopedAccountAccessor,
|
|
authorizeConfigWrite,
|
|
createScopedAccountConfigAccessors,
|
|
createScopedChannelConfigAdapter,
|
|
createScopedChannelConfigBase,
|
|
createScopedDmSecurityResolver,
|
|
createHybridChannelConfigAdapter,
|
|
createTopLevelChannelConfigAdapter,
|
|
createTopLevelChannelConfigBase,
|
|
createHybridChannelConfigBase,
|
|
mapAllowFromEntries,
|
|
resolveChannelConfigWrites,
|
|
resolveIMessageConfigAllowFrom,
|
|
resolveIMessageConfigDefaultTo,
|
|
resolveOptionalConfigString,
|
|
resolveWhatsAppConfigAllowFrom,
|
|
resolveWhatsAppConfigDefaultTo,
|
|
} from "./channel-config-helpers.js";
|
|
|
|
const resolveDefaultAccountId = () => DEFAULT_ACCOUNT_ID;
|
|
|
|
function createMergedReaderCfg(
|
|
channelId: "whatsapp" | "imessage",
|
|
accountConfig: { allowFrom: string[]; defaultTo: string },
|
|
) {
|
|
return {
|
|
channels: {
|
|
[channelId]: {
|
|
allowFrom: ["root"],
|
|
defaultTo: " root:chat ",
|
|
accounts: {
|
|
alt: accountConfig,
|
|
},
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
function createConfigWritesCfg() {
|
|
return {
|
|
channels: {
|
|
telegram: {
|
|
configWrites: true,
|
|
accounts: {
|
|
Work: { configWrites: false },
|
|
},
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
function expectAdapterAllowFromAndDefaultTo(adapter: unknown) {
|
|
const channelAdapter = adapter as {
|
|
resolveAllowFrom?: (params: { cfg: object; accountId: string }) => unknown;
|
|
resolveDefaultTo?: (params: { cfg: object; accountId: string }) => unknown;
|
|
setAccountEnabled?: (params: { cfg: object; accountId: string; enabled: boolean }) => {
|
|
channels?: {
|
|
demo?: unknown;
|
|
};
|
|
};
|
|
};
|
|
|
|
expect(channelAdapter.resolveAllowFrom?.({ cfg: {}, accountId: "alt" })).toEqual(["alt"]);
|
|
expect(channelAdapter.resolveDefaultTo?.({ cfg: {}, accountId: "alt" })).toBe("room:123");
|
|
expect(
|
|
channelAdapter.setAccountEnabled?.({
|
|
cfg: {},
|
|
accountId: "default",
|
|
enabled: true,
|
|
})?.channels?.demo,
|
|
).toEqual({ enabled: true });
|
|
}
|
|
|
|
describe("mapAllowFromEntries", () => {
|
|
it.each([
|
|
{
|
|
name: "coerces allowFrom entries to strings",
|
|
input: ["user", 42],
|
|
expected: ["user", "42"],
|
|
},
|
|
{
|
|
name: "returns empty list for missing input",
|
|
input: undefined,
|
|
expected: [],
|
|
},
|
|
])("$name", ({ input, expected }) => {
|
|
expect(mapAllowFromEntries(input)).toEqual(expected);
|
|
});
|
|
});
|
|
|
|
describe("resolveOptionalConfigString", () => {
|
|
it.each([
|
|
{
|
|
name: "trims and returns string values",
|
|
input: " room:123 ",
|
|
expected: "room:123",
|
|
},
|
|
{
|
|
name: "coerces numeric values",
|
|
input: 123,
|
|
expected: "123",
|
|
},
|
|
{
|
|
name: "returns undefined for empty string values",
|
|
input: " ",
|
|
expected: undefined,
|
|
},
|
|
{
|
|
name: "returns undefined for missing values",
|
|
input: undefined,
|
|
expected: undefined,
|
|
},
|
|
])("$name", ({ input, expected }) => {
|
|
expect(resolveOptionalConfigString(input)).toBe(expected);
|
|
});
|
|
});
|
|
|
|
describe("provider config readers", () => {
|
|
it.each([
|
|
{
|
|
name: "reads merged WhatsApp allowFrom/defaultTo without the channel registry",
|
|
cfg: createMergedReaderCfg("whatsapp", {
|
|
allowFrom: ["49123", "42"],
|
|
defaultTo: " alt:chat ",
|
|
}),
|
|
resolveAllowFrom: resolveWhatsAppConfigAllowFrom,
|
|
resolveDefaultTo: resolveWhatsAppConfigDefaultTo,
|
|
expectedAllowFrom: ["49123", "42"],
|
|
},
|
|
{
|
|
name: "reads merged iMessage allowFrom/defaultTo without the channel registry",
|
|
cfg: createMergedReaderCfg("imessage", {
|
|
allowFrom: ["chat_id:9", "user@example.com"],
|
|
defaultTo: " alt:chat ",
|
|
}),
|
|
resolveAllowFrom: resolveIMessageConfigAllowFrom,
|
|
resolveDefaultTo: resolveIMessageConfigDefaultTo,
|
|
expectedAllowFrom: ["chat_id:9", "user@example.com"],
|
|
},
|
|
])("$name", ({ cfg, resolveAllowFrom, resolveDefaultTo, expectedAllowFrom }) => {
|
|
expect(resolveAllowFrom({ cfg, accountId: "alt" })).toEqual(expectedAllowFrom);
|
|
expect(resolveDefaultTo({ cfg, accountId: "alt" })).toBe("alt:chat");
|
|
});
|
|
});
|
|
|
|
describe("config write helpers", () => {
|
|
it("matches account ids case-insensitively", () => {
|
|
expect(
|
|
resolveChannelConfigWrites({
|
|
cfg: createConfigWritesCfg(),
|
|
channelId: "telegram",
|
|
accountId: "work",
|
|
}),
|
|
).toBe(false);
|
|
});
|
|
|
|
it("blocks account-scoped writes when the configured account key differs only by case", () => {
|
|
expect(
|
|
authorizeConfigWrite({
|
|
cfg: createConfigWritesCfg(),
|
|
target: {
|
|
kind: "account",
|
|
scope: { channelId: "telegram", accountId: "work" },
|
|
},
|
|
}),
|
|
).toEqual({
|
|
allowed: false,
|
|
reason: "target-disabled",
|
|
blockedScope: {
|
|
kind: "target",
|
|
scope: { channelId: "telegram", accountId: "work" },
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("adaptScopedAccountAccessor", () => {
|
|
it("binds positional callback args into the shared account context object", () => {
|
|
const accessor = adaptScopedAccountAccessor(({ cfg, accountId }) => ({
|
|
channel: cfg.channels?.demo,
|
|
accountId: accountId ?? "default",
|
|
}));
|
|
|
|
expect(
|
|
accessor(
|
|
{
|
|
channels: {
|
|
demo: {
|
|
enabled: true,
|
|
},
|
|
},
|
|
},
|
|
"alt",
|
|
),
|
|
).toEqual({
|
|
channel: {
|
|
enabled: true,
|
|
},
|
|
accountId: "alt",
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("createScopedAccountConfigAccessors", () => {
|
|
it("maps allowFrom and defaultTo from the resolved account", () => {
|
|
const accessors = createScopedAccountConfigAccessors({
|
|
resolveAccount: ({ accountId }) => ({
|
|
allowFrom: accountId ? [accountId, 42] : ["fallback"],
|
|
defaultTo: " room:123 ",
|
|
}),
|
|
resolveAllowFrom: (account) => account.allowFrom,
|
|
formatAllowFrom: (allowFrom) => allowFrom.map((entry) => String(entry).toUpperCase()),
|
|
resolveDefaultTo: (account) => account.defaultTo,
|
|
});
|
|
|
|
expect(
|
|
accessors.resolveAllowFrom?.({
|
|
cfg: {},
|
|
accountId: "owner",
|
|
}),
|
|
).toEqual(["owner", "42"]);
|
|
expect(
|
|
accessors.formatAllowFrom?.({
|
|
cfg: {},
|
|
allowFrom: ["owner"],
|
|
}),
|
|
).toEqual(["OWNER"]);
|
|
expect(
|
|
accessors.resolveDefaultTo?.({
|
|
cfg: {},
|
|
accountId: "owner",
|
|
}),
|
|
).toBe("room:123");
|
|
});
|
|
|
|
it("omits resolveDefaultTo when no selector is provided", () => {
|
|
const accessors = createScopedAccountConfigAccessors({
|
|
resolveAccount: () => ({ allowFrom: ["owner"] }),
|
|
resolveAllowFrom: (account) => account.allowFrom,
|
|
formatAllowFrom: (allowFrom) => allowFrom.map((entry) => String(entry)),
|
|
});
|
|
|
|
expect(accessors.resolveDefaultTo).toBeUndefined();
|
|
});
|
|
});
|
|
|
|
describe("createScopedChannelConfigBase", () => {
|
|
it("wires shared account config CRUD through the section helper", () => {
|
|
const base = createScopedChannelConfigBase({
|
|
sectionKey: "demo",
|
|
listAccountIds: () => ["default", "alt"],
|
|
resolveAccount: (_cfg, accountId) => ({ accountId: accountId ?? "default" }),
|
|
defaultAccountId: resolveDefaultAccountId,
|
|
clearBaseFields: ["token"],
|
|
});
|
|
|
|
expect(base.listAccountIds({})).toEqual(["default", "alt"]);
|
|
expect(base.resolveAccount({}, "alt")).toEqual({ accountId: "alt" });
|
|
expect(base.defaultAccountId!({})).toBe("default");
|
|
expect(
|
|
base.setAccountEnabled!({
|
|
cfg: {},
|
|
accountId: "default",
|
|
enabled: true,
|
|
}).channels?.demo,
|
|
).toEqual({ enabled: true });
|
|
expect(
|
|
base.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels,
|
|
).toBeUndefined();
|
|
});
|
|
|
|
it("can force default account config into accounts.default", () => {
|
|
const base = createScopedChannelConfigBase({
|
|
sectionKey: "demo",
|
|
listAccountIds: () => ["default", "alt"],
|
|
resolveAccount: (_cfg, accountId) => ({ accountId: accountId ?? "default" }),
|
|
defaultAccountId: resolveDefaultAccountId,
|
|
clearBaseFields: [],
|
|
allowTopLevel: false,
|
|
});
|
|
|
|
expect(
|
|
base.setAccountEnabled!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
enabled: true,
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
token: "secret",
|
|
accounts: {
|
|
default: { enabled: true },
|
|
},
|
|
});
|
|
expect(
|
|
base.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
accounts: {
|
|
default: { enabled: true },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
token: "secret",
|
|
accounts: undefined,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("createScopedChannelConfigAdapter", () => {
|
|
it("combines scoped CRUD and allowFrom accessors", () => {
|
|
const adapter = createScopedChannelConfigAdapter({
|
|
sectionKey: "demo",
|
|
listAccountIds: () => ["default", "alt"],
|
|
resolveAccount: (_cfg, accountId) => ({
|
|
accountId: accountId ?? "default",
|
|
allowFrom: accountId ? [accountId] : ["fallback"],
|
|
defaultTo: " room:123 ",
|
|
}),
|
|
defaultAccountId: resolveDefaultAccountId,
|
|
clearBaseFields: ["token"],
|
|
resolveAllowFrom: (account) => account.allowFrom,
|
|
formatAllowFrom: (allowFrom) => allowFrom.map((entry) => String(entry).toUpperCase()),
|
|
resolveDefaultTo: (account) => account.defaultTo,
|
|
});
|
|
|
|
expect(adapter.listAccountIds({})).toEqual(["default", "alt"]);
|
|
expect(adapter.resolveAccount({}, "alt")).toEqual({
|
|
accountId: "alt",
|
|
allowFrom: ["alt"],
|
|
defaultTo: " room:123 ",
|
|
});
|
|
expectAdapterAllowFromAndDefaultTo(adapter);
|
|
});
|
|
});
|
|
|
|
describe("createScopedDmSecurityResolver", () => {
|
|
it("builds account-aware DM policy payloads", () => {
|
|
const resolveDmPolicy = createScopedDmSecurityResolver<{
|
|
accountId?: string | null;
|
|
dmPolicy?: string;
|
|
allowFrom?: string[];
|
|
}>({
|
|
channelKey: "demo",
|
|
resolvePolicy: (account) => account.dmPolicy,
|
|
resolveAllowFrom: (account) => account.allowFrom,
|
|
policyPathSuffix: "dmPolicy",
|
|
normalizeEntry: (raw) => raw.toLowerCase(),
|
|
});
|
|
|
|
expect(
|
|
resolveDmPolicy({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
accounts: {
|
|
alt: {},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
accountId: "alt",
|
|
account: {
|
|
accountId: "alt",
|
|
dmPolicy: "allowlist",
|
|
allowFrom: ["Owner"],
|
|
},
|
|
}),
|
|
).toEqual({
|
|
policy: "allowlist",
|
|
allowFrom: ["Owner"],
|
|
policyPath: "channels.demo.accounts.alt.dmPolicy",
|
|
allowFromPath: "channels.demo.accounts.alt.",
|
|
approveHint: formatPairingApproveHint("demo"),
|
|
normalizeEntry: expect.any(Function),
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("createTopLevelChannelConfigBase", () => {
|
|
it("wires top-level enable/delete semantics", () => {
|
|
const base = createTopLevelChannelConfigBase({
|
|
sectionKey: "demo",
|
|
resolveAccount: () => ({ accountId: "default" }),
|
|
});
|
|
|
|
expect(base.listAccountIds({})).toEqual(["default"]);
|
|
expect(base.defaultAccountId!({})).toBe("default");
|
|
expect(
|
|
base.setAccountEnabled!({
|
|
cfg: {},
|
|
accountId: "default",
|
|
enabled: true,
|
|
}).channels?.demo,
|
|
).toEqual({ enabled: true });
|
|
expect(
|
|
base.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
enabled: true,
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels,
|
|
).toBeUndefined();
|
|
});
|
|
|
|
it("can clear only account-scoped fields while preserving channel settings", () => {
|
|
const base = createTopLevelChannelConfigBase({
|
|
sectionKey: "demo",
|
|
resolveAccount: () => ({ accountId: "default" }),
|
|
deleteMode: "clear-fields",
|
|
clearBaseFields: ["token", "allowFrom"],
|
|
});
|
|
|
|
expect(
|
|
base.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
allowFrom: ["owner"],
|
|
markdown: { tables: false },
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
markdown: { tables: false },
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("createTopLevelChannelConfigAdapter", () => {
|
|
it("combines top-level CRUD with separate accessor account resolution", () => {
|
|
const adapter = createTopLevelChannelConfigAdapter<
|
|
{ accountId: string; enabled: boolean },
|
|
{ allowFrom: string[]; defaultTo: string }
|
|
>({
|
|
sectionKey: "demo",
|
|
resolveAccount: () => ({ accountId: "default", enabled: true }),
|
|
resolveAccessorAccount: () => ({ allowFrom: ["owner"], defaultTo: " chat:123 " }),
|
|
deleteMode: "clear-fields",
|
|
clearBaseFields: ["token"],
|
|
resolveAllowFrom: (account) => account.allowFrom,
|
|
formatAllowFrom: (allowFrom) => allowFrom.map((entry) => String(entry)),
|
|
resolveDefaultTo: (account) => account.defaultTo,
|
|
});
|
|
|
|
expect(adapter.resolveAccount({})).toEqual({ accountId: "default", enabled: true });
|
|
expect(adapter.resolveAllowFrom?.({ cfg: {} })).toEqual(["owner"]);
|
|
expect(adapter.resolveDefaultTo?.({ cfg: {} })).toBe("chat:123");
|
|
expect(
|
|
adapter.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
markdown: { tables: false },
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
markdown: { tables: false },
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("createHybridChannelConfigBase", () => {
|
|
it("writes default account enable at the channel root and named accounts under accounts", () => {
|
|
const base = createHybridChannelConfigBase({
|
|
sectionKey: "demo",
|
|
listAccountIds: () => ["default", "alt"],
|
|
resolveAccount: (_cfg, accountId) => ({ accountId: accountId ?? "default" }),
|
|
defaultAccountId: resolveDefaultAccountId,
|
|
clearBaseFields: ["token"],
|
|
});
|
|
|
|
expect(
|
|
base.setAccountEnabled!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
accounts: {
|
|
alt: { enabled: false },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
enabled: true,
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
accounts: {
|
|
alt: { enabled: false },
|
|
},
|
|
enabled: true,
|
|
});
|
|
expect(
|
|
base.setAccountEnabled!({
|
|
cfg: {},
|
|
accountId: "alt",
|
|
enabled: true,
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
accounts: {
|
|
alt: { enabled: true },
|
|
},
|
|
});
|
|
});
|
|
|
|
it("can preserve the section when deleting the default account", () => {
|
|
const base = createHybridChannelConfigBase({
|
|
sectionKey: "demo",
|
|
listAccountIds: () => ["default", "alt"],
|
|
resolveAccount: (_cfg, accountId) => ({ accountId: accountId ?? "default" }),
|
|
defaultAccountId: resolveDefaultAccountId,
|
|
clearBaseFields: ["token", "name"],
|
|
preserveSectionOnDefaultDelete: true,
|
|
});
|
|
|
|
expect(
|
|
base.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
name: "bot",
|
|
accounts: {
|
|
alt: { enabled: true },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
accounts: {
|
|
alt: { enabled: true },
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("createHybridChannelConfigAdapter", () => {
|
|
it("combines hybrid CRUD with allowFrom/defaultTo accessors", () => {
|
|
const adapter = createHybridChannelConfigAdapter<
|
|
{ accountId: string; enabled: boolean },
|
|
{ allowFrom: string[]; defaultTo: string }
|
|
>({
|
|
sectionKey: "demo",
|
|
listAccountIds: () => ["default", "alt"],
|
|
resolveAccount: (_cfg, accountId) => ({
|
|
accountId: accountId ?? "default",
|
|
enabled: true,
|
|
}),
|
|
resolveAccessorAccount: ({ accountId }) => ({
|
|
allowFrom: [accountId ?? "default"],
|
|
defaultTo: " room:123 ",
|
|
}),
|
|
defaultAccountId: resolveDefaultAccountId,
|
|
clearBaseFields: ["token"],
|
|
preserveSectionOnDefaultDelete: true,
|
|
resolveAllowFrom: (account) => account.allowFrom,
|
|
formatAllowFrom: (allowFrom) => allowFrom.map((entry) => String(entry).toUpperCase()),
|
|
resolveDefaultTo: (account) => account.defaultTo,
|
|
});
|
|
|
|
expectAdapterAllowFromAndDefaultTo(adapter);
|
|
expect(
|
|
adapter.deleteAccount!({
|
|
cfg: {
|
|
channels: {
|
|
demo: {
|
|
token: "secret",
|
|
markdown: { tables: false },
|
|
},
|
|
},
|
|
},
|
|
accountId: "default",
|
|
}).channels?.demo,
|
|
).toEqual({
|
|
markdown: { tables: false },
|
|
});
|
|
});
|
|
});
|