mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-17 04:50:51 +00:00
* fix: prefer bundled channel plugins over npm duplicates * fix: tighten bundled plugin review follow-ups * fix: address check gate follow-ups * docs: add changelog for bundled plugin install fix * fix: align lifecycle test formatting with CI oxfmt
116 lines
3.7 KiB
TypeScript
116 lines
3.7 KiB
TypeScript
import { describe, expect, it, vi } from "vitest";
|
|
import { PLUGIN_INSTALL_ERROR_CODE } from "../plugins/install.js";
|
|
import {
|
|
resolveBundledInstallPlanForCatalogEntry,
|
|
resolveBundledInstallPlanBeforeNpm,
|
|
resolveBundledInstallPlanForNpmFailure,
|
|
} from "./plugin-install-plan.js";
|
|
|
|
describe("plugin install plan helpers", () => {
|
|
it("prefers bundled plugin for bare plugin-id specs", () => {
|
|
const findBundledSource = vi.fn().mockReturnValue({
|
|
pluginId: "voice-call",
|
|
localPath: "/tmp/extensions/voice-call",
|
|
npmSpec: "@openclaw/voice-call",
|
|
});
|
|
|
|
const result = resolveBundledInstallPlanBeforeNpm({
|
|
rawSpec: "voice-call",
|
|
findBundledSource,
|
|
});
|
|
|
|
expect(findBundledSource).toHaveBeenCalledWith({ kind: "pluginId", value: "voice-call" });
|
|
expect(result?.bundledSource.pluginId).toBe("voice-call");
|
|
expect(result?.warning).toContain('bare install spec "voice-call"');
|
|
});
|
|
|
|
it("skips bundled pre-plan for scoped npm specs", () => {
|
|
const findBundledSource = vi.fn();
|
|
const result = resolveBundledInstallPlanBeforeNpm({
|
|
rawSpec: "@openclaw/voice-call",
|
|
findBundledSource,
|
|
});
|
|
|
|
expect(findBundledSource).not.toHaveBeenCalled();
|
|
expect(result).toBeNull();
|
|
});
|
|
|
|
it("prefers bundled catalog plugin by id before npm spec", () => {
|
|
const findBundledSource = vi
|
|
.fn()
|
|
.mockImplementation(({ kind, value }: { kind: "pluginId" | "npmSpec"; value: string }) => {
|
|
if (kind === "pluginId" && value === "voice-call") {
|
|
return {
|
|
pluginId: "voice-call",
|
|
localPath: "/tmp/extensions/voice-call",
|
|
npmSpec: "@openclaw/voice-call",
|
|
};
|
|
}
|
|
return undefined;
|
|
});
|
|
|
|
const result = resolveBundledInstallPlanForCatalogEntry({
|
|
pluginId: "voice-call",
|
|
npmSpec: "@openclaw/voice-call",
|
|
findBundledSource,
|
|
});
|
|
|
|
expect(findBundledSource).toHaveBeenCalledWith({ kind: "pluginId", value: "voice-call" });
|
|
expect(result?.bundledSource.localPath).toBe("/tmp/extensions/voice-call");
|
|
});
|
|
|
|
it("rejects npm-spec matches that resolve to a different plugin id", () => {
|
|
const findBundledSource = vi
|
|
.fn()
|
|
.mockImplementation(({ kind }: { kind: "pluginId" | "npmSpec"; value: string }) => {
|
|
if (kind === "npmSpec") {
|
|
return {
|
|
pluginId: "not-voice-call",
|
|
localPath: "/tmp/extensions/not-voice-call",
|
|
npmSpec: "@openclaw/voice-call",
|
|
};
|
|
}
|
|
return undefined;
|
|
});
|
|
|
|
const result = resolveBundledInstallPlanForCatalogEntry({
|
|
pluginId: "voice-call",
|
|
npmSpec: "@openclaw/voice-call",
|
|
findBundledSource,
|
|
});
|
|
|
|
expect(result).toBeNull();
|
|
});
|
|
|
|
it("uses npm-spec bundled fallback only for package-not-found", () => {
|
|
const findBundledSource = vi.fn().mockReturnValue({
|
|
pluginId: "voice-call",
|
|
localPath: "/tmp/extensions/voice-call",
|
|
npmSpec: "@openclaw/voice-call",
|
|
});
|
|
const result = resolveBundledInstallPlanForNpmFailure({
|
|
rawSpec: "@openclaw/voice-call",
|
|
code: PLUGIN_INSTALL_ERROR_CODE.NPM_PACKAGE_NOT_FOUND,
|
|
findBundledSource,
|
|
});
|
|
|
|
expect(findBundledSource).toHaveBeenCalledWith({
|
|
kind: "npmSpec",
|
|
value: "@openclaw/voice-call",
|
|
});
|
|
expect(result?.warning).toContain("npm package unavailable");
|
|
});
|
|
|
|
it("skips fallback for non-not-found npm failures", () => {
|
|
const findBundledSource = vi.fn();
|
|
const result = resolveBundledInstallPlanForNpmFailure({
|
|
rawSpec: "@openclaw/voice-call",
|
|
code: "INSTALL_FAILED",
|
|
findBundledSource,
|
|
});
|
|
|
|
expect(findBundledSource).not.toHaveBeenCalled();
|
|
expect(result).toBeNull();
|
|
});
|
|
});
|