Files
openclaw/src/infra/clawhub.test.ts
2026-03-23 18:52:37 -07:00

168 lines
6.2 KiB
TypeScript

import fs from "node:fs/promises";
import os from "node:os";
import path from "node:path";
import { afterEach, describe, expect, it, vi } from "vitest";
import {
parseClawHubPluginSpec,
resolveClawHubAuthToken,
searchClawHubSkills,
resolveLatestVersionFromPackage,
satisfiesGatewayMinimum,
satisfiesPluginApiRange,
} from "./clawhub.js";
describe("clawhub helpers", () => {
const originalHome = process.env.HOME;
afterEach(() => {
delete process.env.OPENCLAW_CLAWHUB_TOKEN;
delete process.env.CLAWHUB_TOKEN;
delete process.env.CLAWHUB_AUTH_TOKEN;
delete process.env.OPENCLAW_CLAWHUB_CONFIG_PATH;
delete process.env.CLAWHUB_CONFIG_PATH;
delete process.env.CLAWDHUB_CONFIG_PATH;
delete process.env.XDG_CONFIG_HOME;
if (originalHome == null) {
delete process.env.HOME;
} else {
process.env.HOME = originalHome;
}
});
it("parses explicit ClawHub package specs", () => {
expect(parseClawHubPluginSpec("clawhub:demo")).toEqual({
name: "demo",
});
expect(parseClawHubPluginSpec("clawhub:demo@1.2.3")).toEqual({
name: "demo",
version: "1.2.3",
});
expect(parseClawHubPluginSpec("@scope/pkg")).toBeNull();
});
it("resolves latest versions from latestVersion before tags", () => {
expect(
resolveLatestVersionFromPackage({
package: {
name: "demo",
displayName: "Demo",
family: "code-plugin",
channel: "official",
isOfficial: true,
createdAt: 0,
updatedAt: 0,
latestVersion: "1.2.3",
tags: { latest: "1.2.2" },
},
}),
).toBe("1.2.3");
expect(
resolveLatestVersionFromPackage({
package: {
name: "demo",
displayName: "Demo",
family: "code-plugin",
channel: "official",
isOfficial: true,
createdAt: 0,
updatedAt: 0,
tags: { latest: "1.2.2" },
},
}),
).toBe("1.2.2");
});
it("checks plugin api ranges without semver dependency", () => {
expect(satisfiesPluginApiRange("1.2.3", "^1.2.0")).toBe(true);
expect(satisfiesPluginApiRange("1.9.0", ">=1.2.0 <2.0.0")).toBe(true);
expect(satisfiesPluginApiRange("2.0.0", "^1.2.0")).toBe(false);
expect(satisfiesPluginApiRange("1.1.9", ">=1.2.0")).toBe(false);
expect(satisfiesPluginApiRange("2026.3.22", ">=2026.3.22")).toBe(true);
expect(satisfiesPluginApiRange("2026.3.21", ">=2026.3.22")).toBe(false);
expect(satisfiesPluginApiRange("invalid", "^1.2.0")).toBe(false);
});
it("checks min gateway versions with loose host labels", () => {
expect(satisfiesGatewayMinimum("2026.3.22", "2026.3.0")).toBe(true);
expect(satisfiesGatewayMinimum("OpenClaw 2026.3.22", "2026.3.0")).toBe(true);
expect(satisfiesGatewayMinimum("2026.2.9", "2026.3.0")).toBe(false);
expect(satisfiesGatewayMinimum("unknown", "2026.3.0")).toBe(false);
});
it("resolves ClawHub auth token from config.json", async () => {
const configRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-clawhub-config-"));
const configPath = path.join(configRoot, "clawhub", "config.json");
process.env.OPENCLAW_CLAWHUB_CONFIG_PATH = configPath;
await fs.mkdir(path.dirname(configPath), { recursive: true });
await fs.writeFile(configPath, JSON.stringify({ auth: { token: "cfg-token-123" } }), "utf8");
await expect(resolveClawHubAuthToken()).resolves.toBe("cfg-token-123");
});
it("resolves ClawHub auth token from the legacy config path override", async () => {
const configRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-clawdhub-config-"));
const configPath = path.join(configRoot, "config.json");
process.env.CLAWDHUB_CONFIG_PATH = configPath;
await fs.writeFile(configPath, JSON.stringify({ token: "legacy-token-123" }), "utf8");
await expect(resolveClawHubAuthToken()).resolves.toBe("legacy-token-123");
});
it.runIf(process.platform === "darwin")(
"resolves ClawHub auth token from the macOS Application Support path",
async () => {
const fakeHome = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-clawhub-home-"));
const configPath = path.join(
fakeHome,
"Library",
"Application Support",
"clawhub",
"config.json",
);
const homedirSpy = vi.spyOn(os, "homedir").mockReturnValue(fakeHome);
try {
await fs.mkdir(path.dirname(configPath), { recursive: true });
await fs.writeFile(configPath, JSON.stringify({ token: "macos-token-123" }), "utf8");
await expect(resolveClawHubAuthToken()).resolves.toBe("macos-token-123");
} finally {
homedirSpy.mockRestore();
}
},
);
it.runIf(process.platform === "darwin")(
"falls back to XDG_CONFIG_HOME on macOS when Application Support has no config",
async () => {
const fakeHome = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-clawhub-home-"));
const xdgRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-clawhub-xdg-"));
const configPath = path.join(xdgRoot, "clawhub", "config.json");
const homedirSpy = vi.spyOn(os, "homedir").mockReturnValue(fakeHome);
process.env.XDG_CONFIG_HOME = xdgRoot;
try {
await fs.mkdir(path.dirname(configPath), { recursive: true });
await fs.writeFile(configPath, JSON.stringify({ token: "xdg-token-123" }), "utf8");
await expect(resolveClawHubAuthToken()).resolves.toBe("xdg-token-123");
} finally {
homedirSpy.mockRestore();
}
},
);
it("injects resolved auth token into ClawHub requests", async () => {
process.env.OPENCLAW_CLAWHUB_TOKEN = "env-token-123";
const fetchImpl = async (input: string | URL | Request, init?: RequestInit) => {
const url = input instanceof Request ? input.url : String(input);
expect(url).toContain("/api/v1/search");
expect(new Headers(init?.headers).get("Authorization")).toBe("Bearer env-token-123");
return new Response(JSON.stringify({ results: [] }), {
status: 200,
headers: { "content-type": "application/json" },
});
};
await expect(searchClawHubSkills({ query: "calendar", fetchImpl })).resolves.toEqual([]);
});
});