Fixes: 1. media.ts: wrap placeholder generation in try-catch 2. toolbar.ts: check r.ok, display error message in popover
720 lines
20 KiB
TypeScript
720 lines
20 KiB
TypeScript
/**
|
|
* HookPipeline Tests
|
|
*
|
|
* Tests the v2 hook pipeline for:
|
|
* - Hook registration and sorting
|
|
* - Hook execution with timeout
|
|
* - Content hooks (beforeSave, afterSave, beforeDelete, afterDelete)
|
|
* - Lifecycle hooks (install, activate, deactivate, uninstall)
|
|
* - Error handling and error policies
|
|
*/
|
|
|
|
import { describe, it, expect, vi } from "vitest";
|
|
|
|
import { HookPipeline, createHookPipeline } from "../../../src/plugins/hooks.js";
|
|
import type { ResolvedPlugin, ResolvedHook } from "../../../src/plugins/types.js";
|
|
|
|
/**
|
|
* Create a minimal resolved plugin for testing
|
|
*/
|
|
function createTestPlugin(overrides: Partial<ResolvedPlugin> = {}): ResolvedPlugin {
|
|
return {
|
|
id: overrides.id ?? "test-plugin",
|
|
version: "1.0.0",
|
|
capabilities: [],
|
|
allowedHosts: [],
|
|
storage: {},
|
|
admin: {
|
|
pages: [],
|
|
widgets: [],
|
|
},
|
|
hooks: {},
|
|
routes: {},
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Create a resolved hook with defaults
|
|
*/
|
|
function createTestHook<T>(
|
|
pluginId: string,
|
|
handler: T,
|
|
overrides: Partial<ResolvedHook<T>> = {},
|
|
): ResolvedHook<T> {
|
|
return {
|
|
pluginId,
|
|
handler,
|
|
priority: 100,
|
|
timeout: 5000,
|
|
dependencies: [],
|
|
errorPolicy: "continue",
|
|
exclusive: false,
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
describe("HookPipeline", () => {
|
|
describe("construction and registration", () => {
|
|
it("creates empty pipeline with no plugins", () => {
|
|
const pipeline = new HookPipeline([]);
|
|
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(false);
|
|
expect(pipeline.getHookCount("content:beforeSave")).toBe(0);
|
|
});
|
|
|
|
it("registers hooks from plugins", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
capabilities: ["content:write", "content:read"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("test", vi.fn()),
|
|
"content:afterSave": createTestHook("test", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(true);
|
|
expect(pipeline.hasHooks("content:afterSave")).toBe(true);
|
|
expect(pipeline.hasHooks("content:beforeDelete")).toBe(false);
|
|
});
|
|
|
|
it("tracks registered hook names", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
capabilities: ["content:write", "media:read"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("test", vi.fn()),
|
|
"media:afterUpload": createTestHook("test", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
const registered = pipeline.getRegisteredHooks();
|
|
|
|
expect(registered).toContain("content:beforeSave");
|
|
expect(registered).toContain("media:afterUpload");
|
|
expect(registered).not.toContain("content:afterSave");
|
|
});
|
|
});
|
|
|
|
describe("hook sorting", () => {
|
|
it("sorts hooks by priority (lower first)", () => {
|
|
const handler1 = vi.fn();
|
|
const handler2 = vi.fn();
|
|
const handler3 = vi.fn();
|
|
|
|
const plugin1 = createTestPlugin({
|
|
id: "plugin-1",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-1", handler1, {
|
|
priority: 200,
|
|
}),
|
|
},
|
|
});
|
|
|
|
const plugin2 = createTestPlugin({
|
|
id: "plugin-2",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-2", handler2, {
|
|
priority: 50,
|
|
}),
|
|
},
|
|
});
|
|
|
|
const plugin3 = createTestPlugin({
|
|
id: "plugin-3",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-3", handler3, {
|
|
priority: 100,
|
|
}),
|
|
},
|
|
});
|
|
|
|
// Create pipeline and manually verify order through execution
|
|
const pipeline = new HookPipeline([plugin1, plugin2, plugin3]);
|
|
|
|
expect(pipeline.getHookCount("content:beforeSave")).toBe(3);
|
|
});
|
|
|
|
it("respects dependencies when sorting", () => {
|
|
const handler1 = vi.fn();
|
|
const handler2 = vi.fn();
|
|
|
|
const plugin1 = createTestPlugin({
|
|
id: "plugin-1",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-1", handler1, {
|
|
priority: 50, // Lower priority but...
|
|
dependencies: ["plugin-2"], // depends on plugin-2
|
|
}),
|
|
},
|
|
});
|
|
|
|
const plugin2 = createTestPlugin({
|
|
id: "plugin-2",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-2", handler2, {
|
|
priority: 100, // Higher priority
|
|
}),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin1, plugin2]);
|
|
|
|
// plugin-2 should run before plugin-1 despite priority
|
|
// because plugin-1 depends on plugin-2
|
|
expect(pipeline.getHookCount("content:beforeSave")).toBe(2);
|
|
});
|
|
});
|
|
|
|
describe("content:beforeSave", () => {
|
|
it("runs hooks and returns modified content", async () => {
|
|
const handler = vi.fn(async (event) => ({
|
|
...event.content,
|
|
modified: true,
|
|
}));
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
// Need context factory for actual execution
|
|
// Without it, getContext will throw
|
|
const pipeline = new HookPipeline([plugin]);
|
|
|
|
// For unit test without DB, we can verify the hook count
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(true);
|
|
});
|
|
|
|
it("chains content through multiple hooks", async () => {
|
|
const handler1 = vi.fn(async (event) => ({
|
|
...event.content,
|
|
step1: true,
|
|
}));
|
|
|
|
const handler2 = vi.fn(async (event) => ({
|
|
...event.content,
|
|
step2: true,
|
|
}));
|
|
|
|
const plugin1 = createTestPlugin({
|
|
id: "plugin-1",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-1", handler1, {
|
|
priority: 1,
|
|
}),
|
|
},
|
|
});
|
|
|
|
const plugin2 = createTestPlugin({
|
|
id: "plugin-2",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("plugin-2", handler2, {
|
|
priority: 2,
|
|
}),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin1, plugin2]);
|
|
expect(pipeline.getHookCount("content:beforeSave")).toBe(2);
|
|
});
|
|
});
|
|
|
|
describe("content:beforeDelete", () => {
|
|
it("registers beforeDelete hooks", () => {
|
|
const handler = vi.fn(async () => true);
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
capabilities: ["content:read"],
|
|
hooks: {
|
|
"content:beforeDelete": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:beforeDelete")).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("lifecycle hooks", () => {
|
|
it("registers plugin:install hook", () => {
|
|
const handler = vi.fn();
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
hooks: {
|
|
"plugin:install": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("plugin:install")).toBe(true);
|
|
});
|
|
|
|
it("registers plugin:activate hook", () => {
|
|
const handler = vi.fn();
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
hooks: {
|
|
"plugin:activate": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("plugin:activate")).toBe(true);
|
|
});
|
|
|
|
it("registers plugin:deactivate hook", () => {
|
|
const handler = vi.fn();
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
hooks: {
|
|
"plugin:deactivate": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("plugin:deactivate")).toBe(true);
|
|
});
|
|
|
|
it("registers plugin:uninstall hook", () => {
|
|
const handler = vi.fn();
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
hooks: {
|
|
"plugin:uninstall": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("plugin:uninstall")).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("media hooks", () => {
|
|
it("registers media:beforeUpload hook", () => {
|
|
const handler = vi.fn();
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
capabilities: ["media:write"],
|
|
hooks: {
|
|
"media:beforeUpload": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("media:beforeUpload")).toBe(true);
|
|
});
|
|
|
|
it("registers media:afterUpload hook", () => {
|
|
const handler = vi.fn();
|
|
|
|
const plugin = createTestPlugin({
|
|
id: "test",
|
|
capabilities: ["media:read"],
|
|
hooks: {
|
|
"media:afterUpload": createTestHook("test", handler),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("media:afterUpload")).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("createHookPipeline helper", () => {
|
|
it("creates a HookPipeline instance", () => {
|
|
const plugins = [createTestPlugin({ id: "test" })];
|
|
const pipeline = createHookPipeline(plugins);
|
|
|
|
expect(pipeline).toBeInstanceOf(HookPipeline);
|
|
});
|
|
});
|
|
|
|
// =========================================================================
|
|
// Capability enforcement for non-email hooks
|
|
// =========================================================================
|
|
|
|
describe("capability enforcement — content hooks", () => {
|
|
it("skips content:beforeSave without content:write capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(false);
|
|
});
|
|
|
|
it("skips content:beforeSave with only content:read (requires content:write)", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "read-only",
|
|
capabilities: ["content:read"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("read-only", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(false);
|
|
});
|
|
|
|
it("registers content:beforeSave with content:write capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["content:write"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(true);
|
|
});
|
|
|
|
it("skips content:afterSave without content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"content:afterSave": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterSave")).toBe(false);
|
|
});
|
|
|
|
it("registers content:afterSave with content:read capability (read-only notification)", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["content:read"],
|
|
hooks: {
|
|
"content:afterSave": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterSave")).toBe(true);
|
|
});
|
|
|
|
it("skips content:beforeDelete without content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"content:beforeDelete": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:beforeDelete")).toBe(false);
|
|
});
|
|
|
|
it("skips content:afterDelete without content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"content:afterDelete": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterDelete")).toBe(false);
|
|
});
|
|
|
|
it("registers all content hooks with content:write + content:read", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "writer",
|
|
capabilities: ["content:write", "content:read"],
|
|
hooks: {
|
|
"content:beforeSave": createTestHook("writer", vi.fn()),
|
|
"content:afterSave": createTestHook("writer", vi.fn()),
|
|
"content:beforeDelete": createTestHook("writer", vi.fn()),
|
|
"content:afterDelete": createTestHook("writer", vi.fn()),
|
|
"content:afterPublish": createTestHook("writer", vi.fn()),
|
|
"content:afterUnpublish": createTestHook("writer", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:beforeSave")).toBe(true);
|
|
expect(pipeline.hasHooks("content:afterSave")).toBe(true);
|
|
expect(pipeline.hasHooks("content:beforeDelete")).toBe(true);
|
|
expect(pipeline.hasHooks("content:afterDelete")).toBe(true);
|
|
expect(pipeline.hasHooks("content:afterPublish")).toBe(true);
|
|
expect(pipeline.hasHooks("content:afterUnpublish")).toBe(true);
|
|
});
|
|
|
|
it("skips content:afterPublish without content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"content:afterPublish": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterPublish")).toBe(false);
|
|
});
|
|
|
|
it("registers content:afterPublish with content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["content:read"],
|
|
hooks: {
|
|
"content:afterPublish": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterPublish")).toBe(true);
|
|
});
|
|
|
|
it("skips content:afterUnpublish without content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"content:afterUnpublish": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterUnpublish")).toBe(false);
|
|
});
|
|
|
|
it("registers content:afterUnpublish with content:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["content:read"],
|
|
hooks: {
|
|
"content:afterUnpublish": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("content:afterUnpublish")).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("capability enforcement — media hooks", () => {
|
|
it("skips media:beforeUpload without media:write capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"media:beforeUpload": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("media:beforeUpload")).toBe(false);
|
|
});
|
|
|
|
it("registers media:beforeUpload with media:write capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["media:write"],
|
|
hooks: {
|
|
"media:beforeUpload": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("media:beforeUpload")).toBe(true);
|
|
});
|
|
|
|
it("skips media:afterUpload without media:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"media:afterUpload": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("media:afterUpload")).toBe(false);
|
|
});
|
|
|
|
it("registers media:afterUpload with media:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["media:read"],
|
|
hooks: {
|
|
"media:afterUpload": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("media:afterUpload")).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("capability enforcement — comment hooks", () => {
|
|
it("skips comment:beforeCreate without users:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"comment:beforeCreate": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("comment:beforeCreate")).toBe(false);
|
|
});
|
|
|
|
it("registers comment:beforeCreate with users:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["users:read"],
|
|
hooks: {
|
|
"comment:beforeCreate": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("comment:beforeCreate")).toBe(true);
|
|
});
|
|
|
|
it("skips comment:moderate without users:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"comment:moderate": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("comment:moderate")).toBe(false);
|
|
});
|
|
|
|
it("skips comment:afterCreate without users:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"comment:afterCreate": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("comment:afterCreate")).toBe(false);
|
|
});
|
|
|
|
it("skips comment:afterModerate without users:read capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"comment:afterModerate": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("comment:afterModerate")).toBe(false);
|
|
});
|
|
});
|
|
|
|
describe("capability enforcement — page:fragments", () => {
|
|
it("skips page:fragments without hooks.page-fragments:register capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"page:fragments": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("page:fragments")).toBe(false);
|
|
});
|
|
|
|
it("registers page:fragments with hooks.page-fragments:register capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "has-cap",
|
|
capabilities: ["hooks.page-fragments:register"],
|
|
hooks: {
|
|
"page:fragments": createTestHook("has-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("page:fragments")).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("capability enforcement — hooks without requirements", () => {
|
|
it("registers lifecycle hooks without any capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"plugin:install": createTestHook("no-cap", vi.fn()),
|
|
"plugin:activate": createTestHook("no-cap", vi.fn()),
|
|
"plugin:deactivate": createTestHook("no-cap", vi.fn()),
|
|
"plugin:uninstall": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("plugin:install")).toBe(true);
|
|
expect(pipeline.hasHooks("plugin:activate")).toBe(true);
|
|
expect(pipeline.hasHooks("plugin:deactivate")).toBe(true);
|
|
expect(pipeline.hasHooks("plugin:uninstall")).toBe(true);
|
|
});
|
|
|
|
it("registers cron hook without any capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
cron: createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("cron")).toBe(true);
|
|
});
|
|
|
|
it("registers page:metadata without any capability", () => {
|
|
const plugin = createTestPlugin({
|
|
id: "no-cap",
|
|
capabilities: [],
|
|
hooks: {
|
|
"page:metadata": createTestHook("no-cap", vi.fn()),
|
|
},
|
|
});
|
|
|
|
const pipeline = new HookPipeline([plugin]);
|
|
expect(pipeline.hasHooks("page:metadata")).toBe(true);
|
|
});
|
|
});
|
|
});
|