Hook-Referenz

Auf dieser Seite

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

HookAuslöserKann ändernExklusiv
content:beforeSaveBevor Inhalt gespeichert wirdInhaltsdatenNein
content:afterSaveNachdem Inhalt gespeichert wurdeNichtsNein
content:beforeDeleteBevor Inhalt gelöscht wirdKann abbrechenNein
content:afterDeleteNachdem Inhalt gelöscht wurdeNichtsNein
media:beforeUploadBevor eine Datei hochgeladen wirdDatei-MetadatenNein
media:afterUploadNachdem eine Datei hochgeladen wurdeNichtsNein
cronGeplante Aufgabe wird ausgelöstNichtsNein
email:beforeSendVor dem E-Mail-VersandNachricht, kann abbrechenNein
email:deliverE-Mail über Transport zustellenNichtsJa
email:afterSendNach erfolgreichem E-Mail-VersandNichtsNein
comment:beforeCreateBevor ein Kommentar gespeichert wirdKommentar, kann abbrechenNein
comment:moderateGenehmigungsstatus des Kommentars bestimmenStatusJa
comment:afterCreateNachdem ein Kommentar gespeichert wurdeNichtsNein
comment:afterModerateNachdem ein Admin den Kommentarstatus ändertNichtsNein
page:metadataBeim Rendern des öffentlichen Seiten-HeadsTags beitragenNein
page:fragmentsBeim Rendern des öffentlichen Seiten-BodysSkripte einfügenNein
plugin:installWenn ein Plugin erstmals installiert wirdNichtsNein
plugin:activateWenn ein Plugin aktiviert wirdNichtsNein
plugin:deactivateWenn ein Plugin deaktiviert wirdNichtsNein
plugin:uninstallWenn ein Plugin entfernt wirdNichtsNein

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 void zurü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 false zurück, um die Löschung abzubrechen
  • Geben Sie true oder void zurü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 void zurü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:beforeSendemail:deliveremail: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 false zurück, um die Zustellung abzubrechen
  • Geben Sie void zurü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:beforeCreatecomment:moderatecomment: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 false zurück, um den Kommentar abzulehnen
  • Geben Sie void zurü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

OptionTypStandardBeschreibung
prioritynumber100Ausführungsreihenfolge (niedriger = früher)
timeoutnumber5000Maximale Ausführungszeit in Millisekunden
dependenciesstring[][]Plugin-IDs, die zuerst ausgeführt werden müssen
errorPolicystring"abort""continue", um Fehler zu ignorieren
exclusivebooleanfalseNur 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:

  1. Sortiert nach priority (aufsteigend)
  2. Plugins mit dependencies werden nach ihren Abhängigkeiten ausgeführt
  3. 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: ... }