Hooks ermöglichen es Plugins, das Verhalten von EmDash an bestimmten Punkten im Lebenszyklus von Inhalten, Medien, E-Mails, Kommentaren und Seiten abzufangen und zu modifizieren.
Hook-Übersicht
| Hook | Auslöser | Kann ändern | Exklusiv |
|---|---|---|---|
content:beforeSave | Bevor Inhalt gespeichert wird | Inhaltsdaten | Nein |
content:afterSave | Nachdem Inhalt gespeichert wurde | Nichts | Nein |
content:beforeDelete | Bevor Inhalt gelöscht wird | Kann abbrechen | Nein |
content:afterDelete | Nachdem Inhalt gelöscht wurde | Nichts | Nein |
media:beforeUpload | Bevor eine Datei hochgeladen wird | Datei-Metadaten | Nein |
media:afterUpload | Nachdem eine Datei hochgeladen wurde | Nichts | Nein |
cron | Geplante Aufgabe wird ausgelöst | Nichts | Nein |
email:beforeSend | Vor dem E-Mail-Versand | Nachricht, kann abbrechen | Nein |
email:deliver | E-Mail über Transport zustellen | Nichts | Ja |
email:afterSend | Nach erfolgreichem E-Mail-Versand | Nichts | Nein |
comment:beforeCreate | Bevor ein Kommentar gespeichert wird | Kommentar, kann abbrechen | Nein |
comment:moderate | Genehmigungsstatus des Kommentars bestimmen | Status | Ja |
comment:afterCreate | Nachdem ein Kommentar gespeichert wurde | Nichts | Nein |
comment:afterModerate | Nachdem ein Admin den Kommentarstatus ändert | Nichts | Nein |
page:metadata | Beim Rendern des öffentlichen Seiten-Heads | Tags beitragen | Nein |
page:fragments | Beim Rendern des öffentlichen Seiten-Bodys | Skripte einfügen | Nein |
plugin:install | Wenn ein Plugin erstmals installiert wird | Nichts | Nein |
plugin:activate | Wenn ein Plugin aktiviert wird | Nichts | Nein |
plugin:deactivate | Wenn ein Plugin deaktiviert wird | Nichts | Nein |
plugin:uninstall | Wenn ein Plugin entfernt wird | Nichts | Nein |
Inhalts-Hooks
content:beforeSave
Wird ausgeführt, bevor Inhalt in der Datenbank gespeichert wird. Verwenden Sie diesen Hook zum Validieren, Transformieren oder Anreichern von Inhalten.
import { definePlugin } from "emdash";
export default definePlugin({
id: "my-plugin",
version: "1.0.0",
hooks: {
"content:beforeSave": async (event, ctx) => {
const { content, collection, isNew } = event;
// Add timestamps
if (isNew) {
content.createdBy = "system";
}
content.modifiedAt = new Date().toISOString();
// Return modified content
return content;
},
},
});
Event
interface ContentHookEvent {
content: Record<string, unknown>; // Content data
collection: string; // Collection slug
isNew: boolean; // True for creates, false for updates
}
Rückgabewert
- Geben Sie ein modifiziertes Inhaltsobjekt zurück, um Änderungen anzuwenden
- Geben Sie
voidzurück, um den Inhalt unverändert weiterzuleiten
content:afterSave
Wird nach dem Speichern von Inhalt ausgeführt. Verwenden Sie diesen Hook für Nebeneffekte wie Benachrichtigungen, Cache-Invalidierung oder externe Synchronisierung.
hooks: {
"content:afterSave": async (event, ctx) => {
const { content, collection, isNew } = event;
if (collection === "posts" && content.status === "published") {
// Notify external service
await ctx.http?.fetch("https://api.example.com/notify", {
method: "POST",
body: JSON.stringify({ postId: content.id }),
});
}
},
}
Rückgabewert
Kein Rückgabewert erwartet.
content:beforeDelete
Wird ausgeführt, bevor Inhalt gelöscht wird. Verwenden Sie diesen Hook zum Validieren oder Verhindern der Löschung.
hooks: {
"content:beforeDelete": async (event, ctx) => {
const { id, collection } = event;
// Prevent deletion of protected content
const item = await ctx.content?.get(collection, id);
if (item?.data.protected) {
return false; // Cancel deletion
}
// Allow deletion
return true;
},
}
Event
interface ContentDeleteEvent {
id: string; // Entry ID
collection: string; // Collection slug
}
Rückgabewert
- Geben Sie
falsezurück, um die Löschung abzubrechen - Geben Sie
trueodervoidzurück, um sie zu erlauben
content:afterDelete
Wird nach dem Löschen von Inhalt ausgeführt. Verwenden Sie diesen Hook für Aufräumarbeiten.
hooks: {
"content:afterDelete": async (event, ctx) => {
const { id, collection } = event;
// Clean up related data
await ctx.storage.relatedItems.delete(`${collection}:${id}`);
},
}
Medien-Hooks
media:beforeUpload
Wird ausgeführt, bevor eine Datei hochgeladen wird. Verwenden Sie diesen Hook zum Validieren, Umbenennen oder Ablehnen von Dateien.
hooks: {
"media:beforeUpload": async (event, ctx) => {
const { file } = event;
// Reject files over 10MB
if (file.size > 10 * 1024 * 1024) {
throw new Error("File too large");
}
// Rename file
return {
name: `${Date.now()}-${file.name}`,
type: file.type,
size: file.size,
};
},
}
Event
interface MediaUploadEvent {
file: {
name: string; // Original filename
type: string; // MIME type
size: number; // Size in bytes
};
}
Rückgabewert
- Geben Sie modifizierte Datei-Metadaten zurück, um Änderungen anzuwenden
- Geben Sie
voidzurück, um die Datei unverändert weiterzuleiten - Werfen Sie eine Exception, um den Upload abzulehnen
media:afterUpload
Wird nach dem Hochladen einer Datei ausgeführt. Verwenden Sie diesen Hook für Verarbeitung, Thumbnails oder Metadaten-Extraktion.
hooks: {
"media:afterUpload": async (event, ctx) => {
const { media } = event;
if (media.mimeType.startsWith("image/")) {
// Store image metadata
await ctx.kv.set(`media:${media.id}:analyzed`, {
processedAt: new Date().toISOString(),
});
}
},
}
Event
interface MediaAfterUploadEvent {
media: {
id: string;
filename: string;
mimeType: string;
size: number | null;
url: string;
createdAt: string;
};
}
Lebenszyklus-Hooks
plugin:install
Wird ausgeführt, wenn ein Plugin erstmals installiert wird. Verwenden Sie diesen Hook für die Ersteinrichtung, das Erstellen von Speicher-Collections oder das Befüllen mit Anfangsdaten.
hooks: {
"plugin:install": async (event, ctx) => {
// Initialize default settings
await ctx.kv.set("settings:enabled", true);
await ctx.kv.set("settings:threshold", 100);
ctx.log.info("Plugin installed successfully");
},
}
plugin:activate
Wird ausgeführt, wenn ein Plugin aktiviert wird (nach Installation oder erneuter Aktivierung).
hooks: {
"plugin:activate": async (event, ctx) => {
ctx.log.info("Plugin activated");
},
}
plugin:deactivate
Wird ausgeführt, wenn ein Plugin deaktiviert wird.
hooks: {
"plugin:deactivate": async (event, ctx) => {
ctx.log.info("Plugin deactivated");
},
}
plugin:uninstall
Wird ausgeführt, wenn ein Plugin entfernt wird. Verwenden Sie diesen Hook für Aufräumarbeiten.
hooks: {
"plugin:uninstall": async (event, ctx) => {
const { deleteData } = event;
if (deleteData) {
// Clean up all plugin data
const items = await ctx.kv.list("settings:");
for (const { key } of items) {
await ctx.kv.delete(key);
}
}
ctx.log.info("Plugin uninstalled");
},
}
Event
interface UninstallEvent {
deleteData: boolean; // User chose to delete data
}
Cron-Hook
cron
Wird ausgelöst, wenn eine geplante Aufgabe ausgeführt wird. Planen Sie Aufgaben mit ctx.cron.schedule().
hooks: {
"cron": async (event, ctx) => {
if (event.name === "daily-sync") {
const data = await ctx.http?.fetch("https://api.example.com/data");
ctx.log.info("Sync complete");
}
},
}
Event
interface CronEvent {
name: string;
data?: Record<string, unknown>;
scheduledAt: string;
}
E-Mail-Hooks
E-Mail-Hooks bilden eine Pipeline: email:beforeSend → email:deliver → email:afterSend.
email:beforeSend
Fähigkeit: email:intercept
Middleware-Hook, der vor der Zustellung ausgeführt wird. Transformieren Sie Nachrichten oder brechen Sie die Zustellung ab.
hooks: {
"email:beforeSend": async (event, ctx) => {
// Add footer to all emails
return {
...event.message,
text: event.message.text + "\n\n—Sent from My Site",
};
// Or return false to cancel delivery
},
}
Event
interface EmailBeforeSendEvent {
message: { to: string; subject: string; text: string; html?: string };
source: string;
}
Rückgabewert
- Geben Sie eine modifizierte Nachricht zurück, um sie zu transformieren
- Geben Sie
falsezurück, um die Zustellung abzubrechen - Geben Sie
voidzurück, um die Nachricht unverändert weiterzuleiten
email:deliver
Fähigkeit: email:provide | Exklusiv: Ja
Der Transport-Provider. Nur ein Plugin kann E-Mails zustellen. Verantwortlich für das tatsächliche Senden der Nachricht über einen E-Mail-Dienst.
hooks: {
"email:deliver": {
exclusive: true,
handler: async (event, ctx) => {
await sendViaSES(event.message);
},
},
}
email:afterSend
Fähigkeit: email:intercept
Fire-and-Forget-Hook nach erfolgreicher Zustellung. Fehler werden protokolliert, aber nicht weitergegeben.
hooks: {
"email:afterSend": async (event, ctx) => {
await ctx.kv.set(`email:log:${Date.now()}`, {
to: event.message.to,
subject: event.message.subject,
});
},
}
Kommentar-Hooks
Kommentar-Hooks bilden eine Pipeline: comment:beforeCreate → comment:moderate → comment:afterCreate. Der comment:afterModerate-Hook wird separat ausgelöst, wenn ein Administrator den Status eines Kommentars ändert.
comment:beforeCreate
Fähigkeit: read:users
Middleware-Hook vor dem Speichern eines Kommentars. Anreichern, Validieren oder Ablehnen von Kommentaren.
hooks: {
"comment:beforeCreate": async (event, ctx) => {
// Reject comments with links
if (event.comment.body.includes("http")) {
return false;
}
},
}
Event
interface CommentBeforeCreateEvent {
comment: {
collection: string;
contentId: string;
parentId: string | null;
authorName: string;
authorEmail: string;
authorUserId: string | null;
body: string;
ipHash: string | null;
userAgent: string | null;
};
metadata: Record<string, unknown>;
}
Rückgabewert
- Geben Sie ein modifiziertes Event zurück, um es zu transformieren
- Geben Sie
falsezurück, um den Kommentar abzulehnen - Geben Sie
voidzurück, um ihn unverändert weiterzuleiten
comment:moderate
Fähigkeit: read:users | Exklusiv: Ja
Entscheidet, ob ein Kommentar genehmigt, ausstehend oder Spam ist. Nur ein Moderations-Provider ist aktiv.
hooks: {
"comment:moderate": {
exclusive: true,
handler: async (event, ctx) => {
const score = await checkSpam(event.comment);
return {
status: score > 0.8 ? "spam" : score > 0.5 ? "pending" : "approved",
reason: `Spam score: ${score}`,
};
},
},
}
Event
interface CommentModerateEvent {
comment: { /* same as beforeCreate */ };
metadata: Record<string, unknown>;
collectionSettings: {
commentsEnabled: boolean;
commentsModeration: "all" | "first_time" | "none";
commentsClosedAfterDays: number;
commentsAutoApproveUsers: boolean;
};
priorApprovedCount: number;
}
Rückgabewert
{ status: "approved" | "pending" | "spam"; reason?: string }
comment:afterCreate
Fähigkeit: read:users
Fire-and-Forget-Hook nach dem Speichern eines Kommentars. Verwenden Sie ihn für Benachrichtigungen.
hooks: {
"comment:afterCreate": async (event, ctx) => {
if (event.comment.status === "approved") {
await ctx.email?.send({
to: event.contentAuthor?.email,
subject: `New comment on "${event.content.title}"`,
text: `${event.comment.authorName} commented: ${event.comment.body}`,
});
}
},
}
comment:afterModerate
Fähigkeit: read:users
Fire-and-Forget-Hook, wenn ein Administrator den Status eines Kommentars manuell ändert.
Event
interface CommentAfterModerateEvent {
comment: { id: string; /* ... */ };
previousStatus: string;
newStatus: string;
moderator: { id: string; name: string | null };
}
Seiten-Hooks
Seiten-Hooks werden beim Rendern öffentlicher Seiten ausgeführt. Sie ermöglichen es Plugins, Metadaten und Skripte einzufügen.
page:metadata
Fähigkeit: page:inject
Tragen Sie Meta-Tags, Open-Graph-Eigenschaften, JSON-LD-strukturierte Daten oder Link-Tags zum Seiten-Head bei.
hooks: {
"page:metadata": async (event, ctx) => {
return [
{ kind: "meta", name: "generator", content: "EmDash" },
{ kind: "property", property: "og:site_name", content: event.page.siteName },
{ kind: "jsonld", graph: { "@type": "WebSite", name: event.page.siteName } },
];
},
}
Beitragstypen
type PageMetadataContribution =
| { kind: "meta"; name: string; content: string; key?: string }
| { kind: "property"; property: string; content: string; key?: string }
| { kind: "link"; rel: string; href: string; hreflang?: string; key?: string }
| { kind: "jsonld"; id?: string; graph: Record<string, unknown> };
Das key-Feld dient der Deduplizierung von Beiträgen — nur der letzte Beitrag mit einem bestimmten Schlüssel wird verwendet.
page:fragments
Fähigkeit: page:inject
Fügen Sie Skripte oder HTML in Seiten ein. Nur für vertrauenswürdige (native) Plugins verfügbar.
hooks: {
"page:fragments": async (event, ctx) => {
return [
{
kind: "external-script",
placement: "body:end",
src: "https://analytics.example.com/script.js",
async: true,
},
{
kind: "inline-script",
placement: "head",
code: `window.siteId = "abc123";`,
},
];
},
}
Beitragstypen
type PageFragmentContribution =
| {
kind: "external-script";
placement: "head" | "body:start" | "body:end";
src: string;
async?: boolean;
defer?: boolean;
attributes?: Record<string, string>;
key?: string;
}
| {
kind: "inline-script";
placement: "head" | "body:start" | "body:end";
code: string;
attributes?: Record<string, string>;
key?: string;
}
| {
kind: "html";
placement: "head" | "body:start" | "body:end";
html: string;
key?: string;
};
Hook-Konfiguration
Hooks akzeptieren entweder eine Handler-Funktion oder ein Konfigurationsobjekt:
hooks: {
// Simple handler
"content:afterSave": async (event, ctx) => { ... },
// With configuration
"content:beforeSave": {
priority: 50, // Lower runs first (default: 100)
timeout: 10000, // Max execution time in ms (default: 5000)
dependencies: [], // Run after these plugins
errorPolicy: "abort", // "continue" or "abort" (default)
handler: async (event, ctx) => { ... },
},
}
Konfigurationsoptionen
| Option | Typ | Standard | Beschreibung |
|---|---|---|---|
priority | number | 100 | Ausführungsreihenfolge (niedriger = früher) |
timeout | number | 5000 | Maximale Ausführungszeit in Millisekunden |
dependencies | string[] | [] | Plugin-IDs, die zuerst ausgeführt werden müssen |
errorPolicy | string | "abort" | "continue", um Fehler zu ignorieren |
exclusive | boolean | false | Nur ein Plugin kann der aktive Provider sein (für Provider-Pattern-Hooks wie email:deliver, comment:moderate) |
Plugin-Kontext
Alle Hooks erhalten ein Kontextobjekt mit Zugriff auf Plugin-APIs:
interface PluginContext {
plugin: { id: string; version: string };
storage: PluginStorage;
kv: KVAccess;
content?: ContentAccess;
media?: MediaAccess;
http?: HttpAccess;
log: LogAccess;
site: { name: string; url: string; locale: string };
url(path: string): string;
users?: UserAccess;
cron?: CronAccess;
email?: EmailAccess;
}
Siehe Plugin-Übersicht — Plugin-Kontext für Fähigkeitsanforderungen und Methodendetails.
Fehlerbehandlung
Fehler in Hooks werden protokolliert und basierend auf errorPolicy behandelt:
"abort"(Standard) — Ausführung stoppen, Transaktion zurückrollen, falls zutreffend"continue"— Fehler protokollieren und mit dem nächsten Hook fortfahren
hooks: {
"content:beforeSave": {
errorPolicy: "continue", // Don't block save if this fails
handler: async (event, ctx) => {
try {
await ctx.http?.fetch("https://api.example.com/validate");
} catch (error) {
ctx.log.warn("Validation service unavailable", error);
}
},
},
}
Ausführungsreihenfolge
Hooks werden in dieser Reihenfolge ausgeführt:
- Sortiert nach
priority(aufsteigend) - Plugins mit
dependencieswerden nach ihren Abhängigkeiten ausgeführt - Bei gleicher Priorität ist die Reihenfolge deterministisch, aber nicht spezifiziert
// This runs first (priority 10)
{ priority: 10, handler: ... }
// This runs second (priority 50)
{ priority: 50, handler: ... }
// This runs last (default priority 100)
{ handler: ... }