Files
paperclip/ui/src/plugins/bridge.test.ts
T
Dotta 3c73ed26b5 Expand plugin host surface (#5205)
## Thinking Path

> - Paperclip orchestrates AI agents for zero-human companies
> - The plugin system is the extension boundary for optional product
capabilities
> - Rich plugins need more than a worker entrypoint: they need scoped
database storage, local project folders, managed agents/routines, host
navigation, and reusable UI components
> - The LLM Wiki work exposed those missing host surfaces while keeping
plugin code outside the core control plane
> - This pull request expands the core plugin host, SDK, server APIs,
and UI bridge so plugins can declare and use those surfaces
> - The benefit is that future plugins can integrate with Paperclip
through documented, validated contracts instead of bespoke server or UI
imports

## What Changed

- Added plugin-managed database namespaces and migration tracking,
including Drizzle schema/migration files and SQL validation for
namespace isolation.
- Added server support for plugin local folders, managed agents, managed
routines, scoped plugin APIs, and plugin operation visibility.
- Expanded shared plugin manifest/types/validators and SDK
host/testing/UI exports for richer plugin surfaces.
- Added reusable UI pieces for file trees, managed routines, resizable
sidebars, route sidebars, and plugin bridge initialization.
- Updated plugin docs and example plugins to use the expanded host and
SDK surface.

## Verification

- `pnpm install --frozen-lockfile`
- `pnpm run preflight:workspace-links && pnpm exec vitest run
packages/shared/src/validators/plugin.test.ts
server/src/__tests__/plugin-database.test.ts
server/src/__tests__/plugin-local-folders.test.ts
server/src/__tests__/plugin-managed-agents.test.ts
server/src/__tests__/plugin-managed-routines.test.ts
server/src/__tests__/plugin-orchestration-apis.test.ts
ui/src/api/plugins.test.ts ui/src/components/FileTree.test.tsx
ui/src/components/ResizableSidebarPane.test.tsx
ui/src/pages/PluginPage.test.tsx ui/src/plugins/bridge.test.ts` passed:
11 files, 67 tests.
- Confirmed this PR changes 89 files and does not include
`pnpm-lock.yaml` or `.github/workflows/*`.

## Risks

- Medium: this expands plugin host contracts across db/shared/server/ui
and includes a new core migration (`0076_useful_elektra.sql`).
- The plugin database namespace validator is intentionally restrictive;
plugin authors may need follow-up affordances for SQL patterns that
remain blocked.
- Merge this before the LLM Wiki plugin PR so the plugin can resolve the
new SDK and host APIs.

> For core feature work, check [`ROADMAP.md`](ROADMAP.md) first and
discuss it in `#dev` before opening the PR. Feature PRs that overlap
with planned core work may need to be redirected — check the roadmap
first. See `CONTRIBUTING.md`.

## Model Used

- OpenAI Codex, GPT-5 coding agent, tool-enabled shell/git/GitHub
workflow. Context window size was not exposed by the runtime.

## Checklist

- [x] I have included a thinking path that traces from project context
to this change
- [x] I have specified the model used (with version and capability
details)
- [x] I have checked ROADMAP.md and confirmed this PR does not duplicate
planned core work
- [x] I have run tests locally and they pass
- [x] I have added or updated tests where applicable
- [x] If this change affects the UI, I have included before/after
screenshots
- [x] I have updated relevant documentation to reflect my changes
- [x] I have considered and documented any risks above
- [x] I will address all Greptile and reviewer comments before
requesting merge

---------

Co-authored-by: Paperclip <noreply@paperclip.ing>
2026-05-05 07:42:57 -05:00

307 lines
9.5 KiB
TypeScript

// @vitest-environment jsdom
import * as React from "react";
import * as ReactDOM from "react-dom";
import { act } from "react";
import { createRoot } from "react-dom/client";
import type { MouseEvent as ReactMouseEvent } from "react";
import { renderToStaticMarkup } from "react-dom/server";
import { MemoryRouter } from "react-router-dom";
import { afterEach, describe, expect, it } from "vitest";
import {
FileTree as SdkFileTree,
ManagedRoutinesList as SdkManagedRoutinesList,
MarkdownBlock as SdkMarkdownBlock,
MarkdownEditor as SdkMarkdownEditor,
type FileTreeNode as SdkFileTreeNode,
} from "../../../packages/plugins/sdk/src/ui/components";
import { SidebarProvider, useSidebar } from "@/context/SidebarContext";
import {
PluginBridgeContext,
resolveHostNavigationHref,
shouldHandleHostNavigationClick,
useHostNavigation,
type PluginBridgeContextValue,
} from "./bridge";
import { initPluginBridge } from "./bridge-init";
function clickEvent(
overrides: Partial<ReactMouseEvent<HTMLAnchorElement>> = {},
): ReactMouseEvent<HTMLAnchorElement> {
return {
defaultPrevented: false,
button: 0,
metaKey: false,
altKey: false,
ctrlKey: false,
shiftKey: false,
currentTarget: {
hasAttribute: () => false,
},
...overrides,
} as ReactMouseEvent<HTMLAnchorElement>;
}
afterEach(() => {
delete globalThis.__paperclipPluginBridge__;
});
describe("plugin host navigation", () => {
it("resolves plugin page routes into the active company prefix", () => {
expect(resolveHostNavigationHref("/wiki", "PAP")).toBe("/PAP/wiki");
expect(resolveHostNavigationHref("/wiki?tab=browse#page", "pap")).toBe(
"/PAP/wiki?tab=browse#page",
);
});
it("does not double-prefix active company paths or global host paths", () => {
expect(resolveHostNavigationHref("/PAP/wiki", "PAP")).toBe("/PAP/wiki");
expect(resolveHostNavigationHref("/pap/wiki", "PAP")).toBe("/pap/wiki");
expect(resolveHostNavigationHref("/instance/settings/plugins", "PAP")).toBe(
"/instance/settings/plugins",
);
});
it("intercepts only same-origin plain left-click navigation", () => {
expect(shouldHandleHostNavigationClick(clickEvent(), "/PAP/wiki")).toBe(true);
expect(
shouldHandleHostNavigationClick(clickEvent({ ctrlKey: true }), "/PAP/wiki"),
).toBe(false);
expect(
shouldHandleHostNavigationClick(clickEvent(), "/PAP/wiki", "_blank"),
).toBe(false);
expect(
shouldHandleHostNavigationClick(clickEvent(), "https://example.com/wiki"),
).toBe(false);
});
});
describe("useHostNavigation mobile drawer behavior", () => {
// React 19's `act` requires the env flag and React DOM client.
(globalThis as unknown as { IS_REACT_ACT_ENVIRONMENT?: boolean }).IS_REACT_ACT_ENVIRONMENT = true;
function makeBridgeValue(): PluginBridgeContextValue {
return {
pluginId: "test-plugin",
hostContext: {
companyId: "co",
companyPrefix: "PAP",
projectId: null,
entityId: null,
entityType: null,
userId: null,
},
};
}
function setViewport(width: number) {
Object.defineProperty(window, "innerWidth", {
configurable: true,
writable: true,
value: width,
});
if (typeof window.matchMedia !== "function") {
Object.defineProperty(window, "matchMedia", {
configurable: true,
writable: true,
value: (query: string) => ({
matches: /max-width:\s*767px/.test(query) ? width < 768 : false,
media: query,
onchange: null,
addEventListener: () => undefined,
removeEventListener: () => undefined,
addListener: () => undefined,
removeListener: () => undefined,
dispatchEvent: () => false,
}),
});
}
}
it("closes the sidebar drawer on mobile after a same-origin navigate()", () => {
setViewport(390);
let nav: ReturnType<typeof useHostNavigation> | null = null;
let sidebar: ReturnType<typeof useSidebar> | null = null;
function Probe() {
nav = useHostNavigation();
sidebar = useSidebar();
return null;
}
const container = document.createElement("div");
document.body.appendChild(container);
const root = createRoot(container);
act(() => {
root.render(
React.createElement(
MemoryRouter,
{ initialEntries: ["/PAP/wiki"] },
React.createElement(
SidebarProvider,
null,
React.createElement(
PluginBridgeContext.Provider,
{ value: makeBridgeValue() },
React.createElement(Probe),
),
),
),
);
});
expect(sidebar!.isMobile).toBe(true);
act(() => sidebar!.setSidebarOpen(true));
expect(sidebar!.sidebarOpen).toBe(true);
act(() => nav!.navigate("/wiki?section=ingest"));
expect(sidebar!.sidebarOpen).toBe(false);
act(() => root.unmount());
container.remove();
});
it("leaves the sidebar open on desktop after navigate()", () => {
setViewport(1280);
let nav: ReturnType<typeof useHostNavigation> | null = null;
let sidebar: ReturnType<typeof useSidebar> | null = null;
function Probe() {
nav = useHostNavigation();
sidebar = useSidebar();
return null;
}
const container = document.createElement("div");
document.body.appendChild(container);
const root = createRoot(container);
act(() => {
root.render(
React.createElement(
MemoryRouter,
{ initialEntries: ["/PAP/wiki"] },
React.createElement(
SidebarProvider,
null,
React.createElement(
PluginBridgeContext.Provider,
{ value: makeBridgeValue() },
React.createElement(Probe),
),
),
),
);
});
expect(sidebar!.isMobile).toBe(false);
expect(sidebar!.sidebarOpen).toBe(true);
act(() => nav!.navigate("/wiki?section=ingest"));
expect(sidebar!.sidebarOpen).toBe(true);
act(() => root.unmount());
container.remove();
});
});
describe("plugin SDK FileTree bridge", () => {
const nodes: SdkFileTreeNode[] = [
{
name: "wiki",
path: "wiki",
kind: "dir",
children: [
{
name: "index.md",
path: "wiki/index.md",
kind: "file",
children: [],
},
],
},
];
it("injects the host FileTree implementation through the bridge runtime", () => {
initPluginBridge(React, ReactDOM);
const html = renderToStaticMarkup(
React.createElement(SdkFileTree, {
nodes,
expandedPaths: ["wiki"],
selectedFile: "wiki/index.md",
onToggleDir: () => undefined,
onSelectFile: () => undefined,
}),
);
expect(html).toContain('role="tree"');
expect(html).toContain("wiki");
expect(html).toContain("index.md");
});
it("throws a clear error when the host FileTree implementation is missing", () => {
globalThis.__paperclipPluginBridge__ = {
react: React,
reactDom: ReactDOM,
sdkUi: {},
};
expect(() =>
renderToStaticMarkup(
React.createElement(SdkFileTree, {
nodes,
expandedPaths: ["wiki"],
onToggleDir: () => undefined,
onSelectFile: () => undefined,
}),
),
).toThrow('Paperclip plugin UI runtime is not initialized for "FileTree"');
});
});
describe("plugin SDK markdown component bridge", () => {
it("injects markdown display and editor components through the bridge runtime", () => {
initPluginBridge(React, ReactDOM);
const registry = globalThis.__paperclipPluginBridge__?.sdkUi ?? {};
expect(registry.MarkdownBlock).toBeTypeOf("function");
expect(registry.MarkdownEditor).toBeTypeOf("function");
expect(registry.IssuesList).toBeTypeOf("function");
expect(registry.AssigneePicker).toBeTypeOf("function");
expect(registry.ProjectPicker).toBeTypeOf("function");
expect(registry.ManagedRoutinesList).toBeTypeOf("function");
});
it("renders plugin-provided markdown components when registered by the host", () => {
globalThis.__paperclipPluginBridge__ = {
react: React,
reactDom: ReactDOM,
sdkUi: {
MarkdownBlock: ({ content, enableWikiLinks, wikiLinkRoot }: { content: string; enableWikiLinks?: boolean; wikiLinkRoot?: string }) =>
React.createElement("article", {
"data-wiki-links": enableWikiLinks ? "true" : "false",
"data-wiki-root": wikiLinkRoot,
}, content),
MarkdownEditor: ({ value }: { value: string }) =>
React.createElement("textarea", { value, readOnly: true }),
ManagedRoutinesList: ({ routines }: { routines: Array<{ title: string }> }) =>
React.createElement("section", null, routines.map((routine) => routine.title).join(", ")),
},
};
const markdownHtml = renderToStaticMarkup(React.createElement(SdkMarkdownBlock, {
content: "# Wiki",
enableWikiLinks: true,
wikiLinkRoot: "/wiki/page",
}));
expect(markdownHtml).toContain("# Wiki");
expect(markdownHtml).toContain('data-wiki-links="true"');
expect(markdownHtml).toContain('data-wiki-root="/wiki/page"');
expect(renderToStaticMarkup(React.createElement(SdkMarkdownEditor, { value: "# Wiki", onChange: () => undefined }))).toContain("# Wiki");
expect(renderToStaticMarkup(React.createElement(SdkManagedRoutinesList, {
routines: [{ key: "lint", title: "Run lint", status: "active" }],
}))).toContain("Run lint");
});
});