mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-19 22:10:51 +00:00
* Browser: replace extension path with Chrome MCP * Browser: clarify relay stub and doctor checks * Docs: mark browser MCP migration as breaking * Browser: reject unsupported profile drivers * Browser: accept clawd alias on profile create * Doctor: narrow legacy browser driver migration
117 lines
3.4 KiB
TypeScript
117 lines
3.4 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
const { stopOpenClawChromeMock } = vi.hoisted(() => ({
|
|
stopOpenClawChromeMock: vi.fn(async () => {}),
|
|
}));
|
|
|
|
const { createBrowserRouteContextMock, listKnownProfileNamesMock } = vi.hoisted(() => ({
|
|
createBrowserRouteContextMock: vi.fn(),
|
|
listKnownProfileNamesMock: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("./chrome.js", () => ({
|
|
stopOpenClawChrome: stopOpenClawChromeMock,
|
|
}));
|
|
|
|
vi.mock("./server-context.js", () => ({
|
|
createBrowserRouteContext: createBrowserRouteContextMock,
|
|
listKnownProfileNames: listKnownProfileNamesMock,
|
|
}));
|
|
|
|
import { ensureExtensionRelayForProfiles, stopKnownBrowserProfiles } from "./server-lifecycle.js";
|
|
|
|
describe("ensureExtensionRelayForProfiles", () => {
|
|
it("is a no-op after removing the Chrome extension relay path", async () => {
|
|
await expect(
|
|
ensureExtensionRelayForProfiles({
|
|
resolved: { profiles: {} } as never,
|
|
onWarn: vi.fn(),
|
|
}),
|
|
).resolves.toBeUndefined();
|
|
});
|
|
});
|
|
|
|
describe("stopKnownBrowserProfiles", () => {
|
|
beforeEach(() => {
|
|
createBrowserRouteContextMock.mockClear();
|
|
listKnownProfileNamesMock.mockClear();
|
|
stopOpenClawChromeMock.mockClear();
|
|
});
|
|
|
|
it("stops all known profiles and ignores per-profile failures", async () => {
|
|
listKnownProfileNamesMock.mockReturnValue(["openclaw", "user"]);
|
|
const stopMap: Record<string, ReturnType<typeof vi.fn>> = {
|
|
openclaw: vi.fn(async () => {}),
|
|
user: vi.fn(async () => {
|
|
throw new Error("profile stop failed");
|
|
}),
|
|
};
|
|
createBrowserRouteContextMock.mockReturnValue({
|
|
forProfile: (name: string) => ({
|
|
stopRunningBrowser: stopMap[name],
|
|
}),
|
|
});
|
|
const onWarn = vi.fn();
|
|
const state = { resolved: { profiles: {} }, profiles: new Map() };
|
|
|
|
await stopKnownBrowserProfiles({
|
|
getState: () => state as never,
|
|
onWarn,
|
|
});
|
|
|
|
expect(stopMap.openclaw).toHaveBeenCalledTimes(1);
|
|
expect(stopMap.user).toHaveBeenCalledTimes(1);
|
|
expect(onWarn).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("stops tracked runtime browsers even when the profile no longer resolves", async () => {
|
|
listKnownProfileNamesMock.mockReturnValue(["deleted-local"]);
|
|
createBrowserRouteContextMock.mockReturnValue({
|
|
forProfile: vi.fn(() => {
|
|
throw new Error("profile not found");
|
|
}),
|
|
});
|
|
const localRuntime = {
|
|
profile: {
|
|
name: "deleted-local",
|
|
driver: "openclaw",
|
|
},
|
|
running: {
|
|
pid: 42,
|
|
cdpPort: 18888,
|
|
},
|
|
};
|
|
const launchedBrowser = localRuntime.running;
|
|
const profiles = new Map<string, unknown>([["deleted-local", localRuntime]]);
|
|
const state = {
|
|
resolved: { profiles: {} },
|
|
profiles,
|
|
};
|
|
|
|
await stopKnownBrowserProfiles({
|
|
getState: () => state as never,
|
|
onWarn: vi.fn(),
|
|
});
|
|
|
|
expect(stopOpenClawChromeMock).toHaveBeenCalledWith(launchedBrowser);
|
|
expect(localRuntime.running).toBeNull();
|
|
});
|
|
|
|
it("warns when profile enumeration fails", async () => {
|
|
listKnownProfileNamesMock.mockImplementation(() => {
|
|
throw new Error("oops");
|
|
});
|
|
createBrowserRouteContextMock.mockReturnValue({
|
|
forProfile: vi.fn(),
|
|
});
|
|
const onWarn = vi.fn();
|
|
|
|
await stopKnownBrowserProfiles({
|
|
getState: () => ({ resolved: { profiles: {} }, profiles: new Map() }) as never,
|
|
onWarn,
|
|
});
|
|
|
|
expect(onWarn).toHaveBeenCalledWith("openclaw browser stop failed: Error: oops");
|
|
});
|
|
});
|