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
| Hook | Gatilho | Pode Modificar | Exclusivo |
|---|---|---|---|
content:beforeSave | Antes do conteúdo ser guardado | Dados do conteúdo | Não |
content:afterSave | Após o conteúdo ser guardado | Nada | Não |
content:beforeDelete | Antes do conteúdo ser eliminado | Pode cancelar | Não |
content:afterDelete | Após o conteúdo ser eliminado | Nada | Não |
media:beforeUpload | Antes do ficheiro ser enviado | Metadados do ficheiro | Não |
media:afterUpload | Após o ficheiro ser enviado | Nada | Não |
cron | Tarefa agendada dispara | Nada | Não |
email:beforeSend | Antes do envio de e-mail | Mensagem, pode cancelar | Não |
email:deliver | Entregar e-mail via transporte | Nada | Sim |
email:afterSend | Após entrega de e-mail bem-sucedida | Nada | Não |
comment:beforeCreate | Antes do comentário ser armazenado | Comentário, pode cancelar | Não |
comment:moderate | Decidir status de aprovação | Status | Sim |
comment:afterCreate | Após o comentário ser armazenado | Nada | Não |
comment:afterModerate | Após admin alterar status do comentário | Nada | Não |
page:metadata | Renderização do head da página pública | Contribuir tags | Não |
page:fragments | Renderização do body da página pública | Injetar scripts | Não |
plugin:install | Quando o plugin é instalado pela primeira vez | Nada | Não |
plugin:activate | Quando o plugin é ativado | Nada | Não |
plugin:deactivate | Quando o plugin é desativado | Nada | Não |
plugin:uninstall | Quando o plugin é removido | Nada | Nã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
voidpara 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
falsepara cancelar a eliminação - Retorne
trueouvoidpara 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
voidpara 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:beforeSend → email:deliver → email: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
falsepara cancelar a entrega - Retorne
voidpara 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:beforeCreate → comment:moderate → comment: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
falsepara rejeitar - Retorne
voidpara 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ção | Tipo | Padrão | Descrição |
|---|---|---|---|
priority | number | 100 | Ordem de execução (menor = mais cedo) |
timeout | number | 5000 | Tempo máximo de execução em milissegundos |
dependencies | string[] | [] | IDs de plugins que devem executar primeiro |
errorPolicy | string | "abort" | "continue" para ignorar erros |
exclusive | boolean | false | Apenas 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:
- Ordenados por
priority(ascendente) - Plugins com
dependenciesexecutam após as suas dependências - 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: ... }