feat: enhance email handling with improved error reporting and statistics

- Updated the `reenviarEmail` mutation to return detailed error messages for better user feedback.
- Added a new query to obtain email queue statistics, providing insights into email statuses.
- Enhanced the `processarFilaEmails` mutation to track processing failures and successes more effectively.
- Implemented a manual email processing mutation for immediate testing and control over email sending.
- Improved email validation and error handling in the email sending action, ensuring robust delivery processes.
This commit is contained in:
2025-11-04 02:14:07 -03:00
parent e6105ae8ea
commit 5d2df8077b
3 changed files with 209 additions and 17 deletions

View File

@@ -220,11 +220,21 @@ export const reenviarEmail = mutation({
args: {
emailId: v.id("notificacoesEmail"),
},
returns: v.object({ sucesso: v.boolean() }),
handler: async (ctx, args): Promise<{ sucesso: boolean }> => {
returns: v.object({ sucesso: v.boolean(), erro: v.optional(v.string()) }),
handler: async (ctx, args): Promise<{ sucesso: boolean; erro?: string }> => {
const email = await ctx.db.get(args.emailId);
if (!email) {
return { sucesso: false };
return { sucesso: false, erro: "Email não encontrado" };
}
// Verificar se o email não foi enviado com sucesso ainda
if (email.status === "enviado") {
return { sucesso: false, erro: "Este email já foi enviado com sucesso" };
}
// Verificar se ainda não excedeu o limite de tentativas (max 3)
if ((email.tentativas || 0) >= 3 && email.status !== "falha") {
return { sucesso: false, erro: "Número máximo de tentativas excedido. Crie um novo email." };
}
// Resetar status para pendente
@@ -235,6 +245,11 @@ export const reenviarEmail = mutation({
erroDetalhes: undefined,
});
// Agendar envio imediato
await ctx.scheduler.runAfter(0, api.actions.email.enviar, {
emailId: args.emailId,
});
return { sucesso: true };
},
});
@@ -317,6 +332,58 @@ export const buscarEmailsPorIds = query({
},
});
/**
* Obter estatísticas da fila de emails
*/
export const obterEstatisticasFilaEmails = query({
args: {},
returns: v.object({
total: v.number(),
pendentes: v.number(),
enviando: v.number(),
enviados: v.number(),
falhas: v.number(),
comErro: v.number(),
ultimaExecucaoCron: v.optional(v.number()),
}),
handler: async (ctx) => {
const todosEmails = await ctx.db
.query("notificacoesEmail")
.collect();
const estatisticas = {
total: todosEmails.length,
pendentes: 0,
enviando: 0,
enviados: 0,
falhas: 0,
comErro: 0,
};
for (const email of todosEmails) {
switch (email.status) {
case "pendente":
estatisticas.pendentes++;
break;
case "enviando":
estatisticas.enviando++;
break;
case "enviado":
estatisticas.enviados++;
break;
case "falha":
estatisticas.falhas++;
if (email.erroDetalhes) {
estatisticas.comErro++;
}
break;
}
}
return estatisticas;
},
});
/**
* Listar agendamentos de email do usuário atual
*/
@@ -452,15 +519,23 @@ export const markEmailFalha = internalMutation({
*/
export const processarFilaEmails = internalMutation({
args: {},
returns: v.object({ processados: v.number() }),
returns: v.object({ processados: v.number(), falhas: v.number() }),
handler: async (ctx) => {
// Buscar emails pendentes (max 10 por execução)
// Buscar emails pendentes que não estão agendados para o futuro (max 10 por execução)
const agora = Date.now();
const emailsPendentes = await ctx.db
.query("notificacoesEmail")
.withIndex("by_status", (q) => q.eq("status", "pendente"))
.filter((q) =>
q.or(
q.eq(q.field("agendadaPara"), undefined),
q.lte(q.field("agendadaPara"), agora)
)
)
.take(10);
let processados = 0;
let falhas = 0;
for (const email of emailsPendentes) {
// Verificar se não excedeu tentativas (max 3)
@@ -469,21 +544,115 @@ export const processarFilaEmails = internalMutation({
status: "falha",
erroDetalhes: "Número máximo de tentativas excedido",
});
falhas++;
continue;
}
// Agendar envio via action
await ctx.scheduler.runAfter(0, api.actions.email.enviar, {
emailId: email._id,
});
processados++;
try {
await ctx.scheduler.runAfter(0, api.actions.email.enviar, {
emailId: email._id,
});
processados++;
} catch (error) {
const errorMessage = error instanceof Error ? error.message : String(error);
console.error(`Erro ao agendar email ${email._id}:`, errorMessage);
await ctx.db.patch(email._id, {
status: "falha",
erroDetalhes: `Erro ao agendar envio: ${errorMessage}`,
tentativas: (email.tentativas || 0) + 1,
});
falhas++;
}
}
console.log(
`📧 Fila de emails processada: ${processados} emails agendados para envio`
);
if (processados > 0 || falhas > 0) {
console.log(
`📧 Fila de emails processada: ${processados} emails agendados, ${falhas} falhas`
);
}
return { processados };
return { processados, falhas };
},
});
/**
* Processar fila de emails manualmente (para testes e envio imediato)
*/
export const processarFilaEmailsManual = mutation({
args: {
limite: v.optional(v.number()),
},
returns: v.object({
sucesso: v.boolean(),
processados: v.number(),
falhas: v.number(),
erro: v.optional(v.string())
}),
handler: async (ctx, args): Promise<{
sucesso: boolean;
processados: number;
falhas: number;
erro?: string
}> => {
const usuarioAtual = await getUsuarioAutenticado(ctx);
if (!usuarioAtual) {
return { sucesso: false, processados: 0, falhas: 0, erro: "Usuário não autenticado" };
}
// Verificar se usuário tem permissão (TI_MASTER ou admin)
const role = await ctx.db.get(usuarioAtual.roleId);
if (!role || (role.nivel !== 0 && role.nivel !== 1)) {
return { sucesso: false, processados: 0, falhas: 0, erro: "Permissão negada" };
}
const limite = args.limite || 10;
const agora = Date.now();
// Buscar emails pendentes que não estão agendados para o futuro
const emailsPendentes = await ctx.db
.query("notificacoesEmail")
.withIndex("by_status", (q) => q.eq("status", "pendente"))
.filter((q) =>
q.or(
q.eq(q.field("agendadaPara"), undefined),
q.lte(q.field("agendadaPara"), agora)
)
)
.take(limite);
let processados = 0;
let falhas = 0;
for (const email of emailsPendentes) {
// Verificar se não excedeu tentativas (max 3)
if ((email.tentativas || 0) >= 3) {
await ctx.db.patch(email._id, {
status: "falha",
erroDetalhes: "Número máximo de tentativas excedido",
});
falhas++;
continue;
}
// Agendar envio via action
try {
await ctx.scheduler.runAfter(0, api.actions.email.enviar, {
emailId: email._id,
});
processados++;
} catch (error) {
const errorMessage = error instanceof Error ? error.message : String(error);
console.error(`Erro ao agendar email ${email._id}:`, errorMessage);
await ctx.db.patch(email._id, {
status: "falha",
erroDetalhes: `Erro ao agendar envio: ${errorMessage}`,
tentativas: (email.tentativas || 0) + 1,
});
falhas++;
}
}
return { sucesso: true, processados, falhas };
},
});