Referência de Hooks

Nesta página

Os hooks permitem que plugins interceptem e modifiquem o comportamento do EmDash em pontos específicos do ciclo de vida de conteúdo, media, e-mail, comentários e páginas.

Visão Geral dos Hooks

HookGatilhoPode ModificarExclusivo
content:beforeSaveAntes do conteúdo ser guardadoDados do conteúdoNão
content:afterSaveApós o conteúdo ser guardadoNadaNão
content:beforeDeleteAntes do conteúdo ser eliminadoPode cancelarNão
content:afterDeleteApós o conteúdo ser eliminadoNadaNão
media:beforeUploadAntes do ficheiro ser enviadoMetadados do ficheiroNão
media:afterUploadApós o ficheiro ser enviadoNadaNão
cronTarefa agendada disparaNadaNão
email:beforeSendAntes do envio de e-mailMensagem, pode cancelarNão
email:deliverEntregar e-mail via transporteNadaSim
email:afterSendApós entrega de e-mail bem-sucedidaNadaNão
comment:beforeCreateAntes do comentário ser armazenadoComentário, pode cancelarNão
comment:moderateDecidir status de aprovaçãoStatusSim
comment:afterCreateApós o comentário ser armazenadoNadaNão
comment:afterModerateApós admin alterar status do comentárioNadaNão
page:metadataRenderização do head da página públicaContribuir tagsNão
page:fragmentsRenderização do body da página públicaInjetar scriptsNão
plugin:installQuando o plugin é instalado pela primeira vezNadaNão
plugin:activateQuando o plugin é ativadoNadaNão
plugin:deactivateQuando o plugin é desativadoNadaNão
plugin:uninstallQuando o plugin é removidoNadaNão

Hooks de Conteúdo

content:beforeSave

Executa antes do conteúdo ser guardado na base de dados. Use para validar, transformar ou enriquecer conteúdo.

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;

			// Adicionar timestamps
			if (isNew) {
				content.createdBy = "system";
			}
			content.modifiedAt = new Date().toISOString();

			// Retornar conteúdo modificado
			return content;
		},
	},
});

Evento

interface ContentHookEvent {
	content: Record<string, unknown>; // Dados do conteúdo
	collection: string; // Slug da coleção
	isNew: boolean; // True para criações, false para atualizações
}

Valor de Retorno

  • Retorne o objeto de conteúdo modificado para aplicar alterações
  • Retorne void para passar sem alterações

content:afterSave

Executa após o conteúdo ser guardado. Use para efeitos colaterais como notificações, invalidação de cache ou sincronização externa.

hooks: {
  "content:afterSave": async (event, ctx) => {
    const { content, collection, isNew } = event;

    if (collection === "posts" && content.status === "published") {
      // Notificar serviço externo
      await ctx.http?.fetch("https://api.example.com/notify", {
        method: "POST",
        body: JSON.stringify({ postId: content.id }),
      });
    }
  },
}

Valor de Retorno

Nenhum valor de retorno esperado.

content:beforeDelete

Executa antes do conteúdo ser eliminado. Use para validar a eliminação ou impedi-la.

hooks: {
  "content:beforeDelete": async (event, ctx) => {
    const { id, collection } = event;

    // Impedir eliminação de conteúdo protegido
    const item = await ctx.content?.get(collection, id);
    if (item?.data.protected) {
      return false; // Cancelar eliminação
    }

    // Permitir eliminação
    return true;
  },
}

Evento

interface ContentDeleteEvent {
	id: string; // ID da entrada
	collection: string; // Slug da coleção
}

Valor de Retorno

  • Retorne false para cancelar a eliminação
  • Retorne true ou void para permitir

content:afterDelete

Executa após o conteúdo ser eliminado. Use para tarefas de limpeza.

hooks: {
  "content:afterDelete": async (event, ctx) => {
    const { id, collection } = event;

    // Limpar dados relacionados
    await ctx.storage.relatedItems.delete(`${collection}:${id}`);
  },
}

Hooks de Media

media:beforeUpload

Executa antes de um ficheiro ser enviado. Use para validar, renomear ou rejeitar ficheiros.

hooks: {
  "media:beforeUpload": async (event, ctx) => {
    const { file } = event;

    // Rejeitar ficheiros acima de 10MB
    if (file.size > 10 * 1024 * 1024) {
      throw new Error("File too large");
    }

    // Renomear ficheiro
    return {
      name: `${Date.now()}-${file.name}`,
      type: file.type,
      size: file.size,
    };
  },
}

Evento

interface MediaUploadEvent {
	file: {
		name: string; // Nome original do ficheiro
		type: string; // Tipo MIME
		size: number; // Tamanho em bytes
	};
}

Valor de Retorno

  • Retorne metadados do ficheiro modificados para aplicar alterações
  • Retorne void para passar sem alterações
  • Lance uma exceção para rejeitar o envio

media:afterUpload

Executa após um ficheiro ser enviado. Use para processamento, miniaturas ou extração de metadados.

hooks: {
  "media:afterUpload": async (event, ctx) => {
    const { media } = event;

    if (media.mimeType.startsWith("image/")) {
      // Armazenar metadados da imagem
      await ctx.kv.set(`media:${media.id}:analyzed`, {
        processedAt: new Date().toISOString(),
      });
    }
  },
}

Evento

interface MediaAfterUploadEvent {
	media: {
		id: string;
		filename: string;
		mimeType: string;
		size: number | null;
		url: string;
		createdAt: string;
	};
}

Hooks de Ciclo de Vida

plugin:install

Executa quando um plugin é instalado pela primeira vez. Use para configuração inicial, criação de coleções de armazenamento ou preenchimento de dados.

hooks: {
  "plugin:install": async (event, ctx) => {
    // Inicializar configurações padrão
    await ctx.kv.set("settings:enabled", true);
    await ctx.kv.set("settings:threshold", 100);

    ctx.log.info("Plugin installed successfully");
  },
}

plugin:activate

Executa quando um plugin é ativado (após instalação ou reativação).

hooks: {
  "plugin:activate": async (event, ctx) => {
    ctx.log.info("Plugin activated");
  },
}

plugin:deactivate

Executa quando um plugin é desativado.

hooks: {
  "plugin:deactivate": async (event, ctx) => {
    ctx.log.info("Plugin deactivated");
  },
}

plugin:uninstall

Executa quando um plugin é removido. Use para limpeza.

hooks: {
  "plugin:uninstall": async (event, ctx) => {
    const { deleteData } = event;

    if (deleteData) {
      // Limpar todos os dados do plugin
      const items = await ctx.kv.list("settings:");
      for (const { key } of items) {
        await ctx.kv.delete(key);
      }
    }

    ctx.log.info("Plugin uninstalled");
  },
}

Evento

interface UninstallEvent {
	deleteData: boolean; // O utilizador escolheu eliminar os dados
}

Hook Cron

cron

Disparado quando uma tarefa agendada é executada. Agende tarefas com 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");
    }
  },
}

Evento

interface CronEvent {
	name: string;
	data?: Record<string, unknown>;
	scheduledAt: string;
}

Hooks de E-mail

Os hooks de e-mail formam um pipeline: email:beforeSendemail:deliveremail:afterSend.

email:beforeSend

Capacidade: email:intercept

Hook middleware que executa antes da entrega. Transforme mensagens ou cancele a entrega.

hooks: {
  "email:beforeSend": async (event, ctx) => {
    // Adicionar rodapé a todos os e-mails
    return {
      ...event.message,
      text: event.message.text + "\n\n—Sent from My Site",
    };

    // Ou retornar false para cancelar a entrega
  },
}

Evento

interface EmailBeforeSendEvent {
	message: { to: string; subject: string; text: string; html?: string };
	source: string;
}

Valor de Retorno

  • Retorne a mensagem modificada para transformar
  • Retorne false para cancelar a entrega
  • Retorne void para passar sem alterações

email:deliver

Capacidade: email:provide | Exclusivo: Sim

O fornecedor de transporte. Apenas um plugin pode entregar e-mails. Responsável por efetivamente enviar a mensagem através de um serviço de e-mail.

hooks: {
  "email:deliver": {
    exclusive: true,
    handler: async (event, ctx) => {
      await sendViaSES(event.message);
    },
  },
}

email:afterSend

Capacidade: email:intercept

Hook “dispara e esquece” após entrega bem-sucedida. Os erros são registados mas não propagados.

hooks: {
  "email:afterSend": async (event, ctx) => {
    await ctx.kv.set(`email:log:${Date.now()}`, {
      to: event.message.to,
      subject: event.message.subject,
    });
  },
}

Hooks de Comentários

Os hooks de comentários formam um pipeline: comment:beforeCreatecomment:moderatecomment:afterCreate. O hook comment:afterModerate dispara separadamente quando um admin altera o status de um comentário.

comment:beforeCreate

Capacidade: read:users

Hook middleware antes de um comentário ser armazenado. Enriqueça, valide ou rejeite comentários.

hooks: {
  "comment:beforeCreate": async (event, ctx) => {
    // Rejeitar comentários com links
    if (event.comment.body.includes("http")) {
      return false;
    }
  },
}

Evento

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>;
}

Valor de Retorno

  • Retorne o evento modificado para transformar
  • Retorne false para rejeitar
  • Retorne void para passar sem alterações

comment:moderate

Capacidade: read:users | Exclusivo: Sim

Decide se um comentário é aprovado, pendente ou spam. Apenas um fornecedor de moderação está ativo.

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}`,
      };
    },
  },
}

Evento

interface CommentModerateEvent {
	comment: { /* igual ao beforeCreate */ };
	metadata: Record<string, unknown>;
	collectionSettings: {
		commentsEnabled: boolean;
		commentsModeration: "all" | "first_time" | "none";
		commentsClosedAfterDays: number;
		commentsAutoApproveUsers: boolean;
	};
	priorApprovedCount: number;
}

Valor de Retorno

{ status: "approved" | "pending" | "spam"; reason?: string }

comment:afterCreate

Capacidade: read:users

Hook “dispara e esquece” após um comentário ser armazenado. Use para notificações.

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

Capacidade: read:users

Hook “dispara e esquece” quando um admin altera manualmente o status de um comentário.

Evento

interface CommentAfterModerateEvent {
	comment: { id: string; /* ... */ };
	previousStatus: string;
	newStatus: string;
	moderator: { id: string; name: string | null };
}

Hooks de Página

Os hooks de página executam ao renderizar páginas públicas. Permitem que plugins injetem metadados e scripts.

page:metadata

Capacidade: page:inject

Contribua com meta tags, propriedades Open Graph, dados estruturados JSON-LD ou link tags para o head da página.

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 } },
    ];
  },
}

Tipos de Contribuição

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> };

O campo key deduplica contribuições — apenas a última contribuição com uma determinada chave é utilizada.

page:fragments

Capacidade: page:inject

Injete scripts ou HTML nas páginas. Disponível apenas para plugins confiáveis (nativos).

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";`,
      },
    ];
  },
}

Tipos de Contribuição

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;
		};

Configuração de Hooks

Os hooks aceitam uma função handler ou um objeto de configuração:

hooks: {
  // Handler simples
  "content:afterSave": async (event, ctx) => { ... },

  // Com configuração
  "content:beforeSave": {
    priority: 50,        // Menor executa primeiro (padrão: 100)
    timeout: 10000,      // Tempo máximo de execução em ms (padrão: 5000)
    dependencies: [],    // Executar após estes plugins
    errorPolicy: "abort", // "continue" ou "abort" (padrão)
    handler: async (event, ctx) => { ... },
  },
}

Opções de Configuração

OpçãoTipoPadrãoDescrição
prioritynumber100Ordem de execução (menor = mais cedo)
timeoutnumber5000Tempo máximo de execução em milissegundos
dependenciesstring[][]IDs de plugins que devem executar primeiro
errorPolicystring"abort""continue" para ignorar erros
exclusivebooleanfalseApenas um plugin pode ser o fornecedor ativo (para hooks de padrão fornecedor como email:deliver, comment:moderate)

Contexto do Plugin

Todos os hooks recebem um objeto de contexto com acesso às APIs do plugin:

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;
}

Consulte Visão Geral de Plugins — Contexto do Plugin para requisitos de capacidades e detalhes dos métodos.

Tratamento de Erros

Os erros nos hooks são registados e tratados com base no errorPolicy:

  • "abort" (padrão) — Para a execução, reverte a transação se aplicável
  • "continue" — Regista o erro e continua para o próximo hook
hooks: {
  "content:beforeSave": {
    errorPolicy: "continue", // Não bloquear a gravação se isto falhar
    handler: async (event, ctx) => {
      try {
        await ctx.http?.fetch("https://api.example.com/validate");
      } catch (error) {
        ctx.log.warn("Validation service unavailable", error);
      }
    },
  },
}

Ordem de Execução

Os hooks executam nesta ordem:

  1. Ordenados por priority (ascendente)
  2. Plugins com dependencies executam após as suas dependências
  3. Dentro da mesma prioridade, a ordem é determinística mas não especificada
// Este executa primeiro (prioridade 10)
{ priority: 10, handler: ... }

// Este executa segundo (prioridade 50)
{ priority: 50, handler: ... }

// Este executa por último (prioridade padrão 100)
{ handler: ... }