fix email sender all flow
This commit is contained in:
@@ -0,0 +1,57 @@
|
||||
"use server";
|
||||
|
||||
import { getServerSession } from "next-auth";
|
||||
import { revalidatePath } from "next/cache";
|
||||
import { authOptions } from "@/lib/auth";
|
||||
import { isAdminEmail } from "@/lib/admin";
|
||||
import { emailService } from "@/lib/email/send";
|
||||
import { auditLog } from "@/server/services/audit-log.service";
|
||||
|
||||
async function requireAdmin() {
|
||||
const session = await getServerSession(authOptions);
|
||||
if (!session?.user || !isAdminEmail(session.user.email)) {
|
||||
return null;
|
||||
}
|
||||
return session.user;
|
||||
}
|
||||
|
||||
/** E5.2 — admin retry satu EmailJob yang gagal/antri, kirim ulang langsung. */
|
||||
export async function retryEmailJobAction(jobId: string) {
|
||||
const admin = await requireAdmin();
|
||||
if (!admin) return { error: "Tidak memiliki akses admin" };
|
||||
if (!jobId) return { error: "jobId tidak valid" };
|
||||
|
||||
const result = await emailService.retryJob(jobId);
|
||||
if (!result.ok) {
|
||||
return { error: result.error ?? "Gagal mengirim ulang email" };
|
||||
}
|
||||
await auditLog.record({
|
||||
admin: { id: admin.id, email: admin.email },
|
||||
action: "EMAIL_JOB_RETRY",
|
||||
entityType: "EmailJob",
|
||||
entityId: jobId,
|
||||
});
|
||||
revalidatePath("/admin/emails");
|
||||
revalidatePath("/admin/system");
|
||||
return { success: true as const };
|
||||
}
|
||||
|
||||
/** E5.3 — admin resend email yang sudah pernah terkirim (mis. user lapor tidak terima). */
|
||||
export async function resendEmailAction(emailSentId: string) {
|
||||
const admin = await requireAdmin();
|
||||
if (!admin) return { error: "Tidak memiliki akses admin" };
|
||||
if (!emailSentId) return { error: "emailSentId tidak valid" };
|
||||
|
||||
const result = await emailService.resendEmail(emailSentId);
|
||||
if (!result.ok) {
|
||||
return { error: result.error ?? "Gagal mengirim ulang email" };
|
||||
}
|
||||
await auditLog.record({
|
||||
admin: { id: admin.id, email: admin.email },
|
||||
action: "EMAIL_RESEND",
|
||||
entityType: "EmailSent",
|
||||
entityId: emailSentId,
|
||||
});
|
||||
revalidatePath("/admin/emails");
|
||||
return { success: true as const };
|
||||
}
|
||||
@@ -0,0 +1,97 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
import { useRouter } from "next/navigation";
|
||||
import { retryEmailJobAction, resendEmailAction } from "@/features/email/actions";
|
||||
|
||||
const BTN_CLS =
|
||||
"rounded-lg border border-primary-200 bg-primary-50 px-2.5 py-1 text-[11px] font-semibold text-primary-700 transition-colors hover:bg-primary-100 disabled:cursor-not-allowed disabled:opacity-50";
|
||||
|
||||
/** E5.2 — tombol kirim ulang untuk satu EmailJob (antri / gagal). */
|
||||
export function RetryEmailButton({ jobId }: { jobId: string }) {
|
||||
const router = useRouter();
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [error, setError] = useState("");
|
||||
|
||||
async function handleRetry() {
|
||||
setLoading(true);
|
||||
setError("");
|
||||
const res = await retryEmailJobAction(jobId);
|
||||
setLoading(false);
|
||||
if ("error" in res) {
|
||||
setError(res.error ?? "Gagal");
|
||||
return;
|
||||
}
|
||||
router.refresh();
|
||||
}
|
||||
|
||||
return (
|
||||
<div>
|
||||
<button
|
||||
type="button"
|
||||
onClick={handleRetry}
|
||||
disabled={loading}
|
||||
className={BTN_CLS}
|
||||
>
|
||||
{loading ? "Mengirim…" : "Kirim ulang"}
|
||||
</button>
|
||||
{error && (
|
||||
<p className="mt-1 max-w-[200px] text-[10px] text-red-600">{error}</p>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
/** E5.3 — tombol resend untuk email yang sudah terkirim. */
|
||||
export function ResendEmailButton({
|
||||
emailSentId,
|
||||
disabled,
|
||||
}: {
|
||||
emailSentId: string;
|
||||
disabled?: boolean;
|
||||
}) {
|
||||
const router = useRouter();
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [error, setError] = useState("");
|
||||
const [done, setDone] = useState(false);
|
||||
|
||||
async function handleResend() {
|
||||
setLoading(true);
|
||||
setError("");
|
||||
const res = await resendEmailAction(emailSentId);
|
||||
setLoading(false);
|
||||
if ("error" in res) {
|
||||
setError(res.error ?? "Gagal");
|
||||
return;
|
||||
}
|
||||
setDone(true);
|
||||
router.refresh();
|
||||
}
|
||||
|
||||
if (disabled) {
|
||||
return (
|
||||
<span
|
||||
className="text-[10px] text-neutral-400"
|
||||
title="Body email lama tidak tersimpan"
|
||||
>
|
||||
—
|
||||
</span>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div>
|
||||
<button
|
||||
type="button"
|
||||
onClick={handleResend}
|
||||
disabled={loading || done}
|
||||
className={BTN_CLS}
|
||||
>
|
||||
{loading ? "Mengirim…" : done ? "✓ Terkirim" : "Resend"}
|
||||
</button>
|
||||
{error && (
|
||||
<p className="mt-1 max-w-[200px] text-[10px] text-red-600">{error}</p>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user