Emdash source with visual editor image upload fix

Fixes:
1. media.ts: wrap placeholder generation in try-catch
2. toolbar.ts: check r.ok, display error message in popover
This commit is contained in:
2026-05-03 10:44:54 +07:00
parent 78f81bebb6
commit 2d1be52177
2352 changed files with 662964 additions and 0 deletions

View File

@@ -0,0 +1,256 @@
import * as React from "react";
import { describe, it, expect, vi, beforeEach } from "vitest";
import { CapabilityConsentDialog } from "../../src/components/CapabilityConsentDialog";
import { render } from "../utils/render.tsx";
describe("CapabilityConsentDialog", () => {
let onConfirm: ReturnType<typeof vi.fn>;
let onCancel: ReturnType<typeof vi.fn>;
beforeEach(() => {
onConfirm = vi.fn();
onCancel = vi.fn();
});
it("renders dialog with plugin name and capabilities", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="SEO Helper"
capabilities={["read:content", "write:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect
.element(screen.getByText("SEO Helper requires the following permissions:"))
.toBeInTheDocument();
await expect.element(screen.getByText("Read your content")).toBeInTheDocument();
await expect
.element(screen.getByText("Create, update, and delete content"))
.toBeInTheDocument();
});
it("shows 'Plugin Permissions' title for fresh install", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("Plugin Permissions")).toBeInTheDocument();
});
it("shows 'Review New Permissions' title for update with new capabilities", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content", "write:content"]}
newCapabilities={["write:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("Review New Permissions")).toBeInTheDocument();
await expect
.element(screen.getByText("Test is requesting additional permissions:"))
.toBeInTheDocument();
});
it("marks new capabilities with NEW badge", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content", "write:content", "network:fetch"]}
newCapabilities={["network:fetch"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
// The NEW badge should appear for network:fetch (exact match to avoid matching "New" in header)
const newBadges = screen.getByText("NEW", { exact: true }).all();
expect(newBadges.length).toBeGreaterThanOrEqual(1);
});
it("shows 'Accept & Install' button for fresh install", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("Accept & Install")).toBeInTheDocument();
});
it("shows 'Accept & Update' button for update", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
newCapabilities={["read:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("Accept & Update")).toBeInTheDocument();
});
it("calls onConfirm when confirm button is clicked", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await screen.getByText("Accept & Install").click();
expect(onConfirm).toHaveBeenCalledOnce();
});
it("calls onCancel when cancel button is clicked", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await screen.getByText("Cancel").click();
expect(onCancel).toHaveBeenCalledOnce();
});
it("shows warning banner for 'warn' audit verdict", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
auditVerdict="warn"
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect
.element(screen.getByText("Security audit flagged potential concerns with this plugin."))
.toBeInTheDocument();
});
it("shows danger banner for 'fail' audit verdict", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
auditVerdict="fail"
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect
.element(screen.getByText("Security audit flagged this plugin as potentially unsafe."))
.toBeInTheDocument();
});
it("shows no audit banner for 'pass' verdict", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
auditVerdict="pass"
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
const warnText = screen.getByText(
"Security audit flagged potential concerns with this plugin.",
);
await expect.element(warnText).not.toBeInTheDocument();
});
it("shows pending state during install", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
isPending={true}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("Installing...")).toBeInTheDocument();
});
it("shows pending state during update", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
newCapabilities={["read:content"]}
isPending={true}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("Updating...")).toBeInTheDocument();
});
it("appends allowed hosts for network:fetch", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["network:fetch"]}
allowedHosts={["api.example.com"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect
.element(screen.getByText("Make network requests to: api.example.com"))
.toBeInTheDocument();
});
it("renders raw capability string for unknown capabilities", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["custom:magic"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
await expect.element(screen.getByText("custom:magic")).toBeInTheDocument();
});
it("has correct dialog role and aria attributes", async () => {
const screen = await render(
<CapabilityConsentDialog
pluginName="Test"
capabilities={["read:content"]}
onConfirm={onConfirm}
onCancel={onCancel}
/>,
);
const dialog = screen.getByRole("dialog");
await expect.element(dialog).toBeInTheDocument();
});
});