This commit is contained in:
149
src/app/api/support/tickets/[id]/comments/route.ts
Normal file
149
src/app/api/support/tickets/[id]/comments/route.ts
Normal file
@@ -0,0 +1,149 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import {
|
||||
getSupportTicketById,
|
||||
createSupportTicketComment,
|
||||
updateSupportTicket,
|
||||
} from "@/lib/db";
|
||||
import {
|
||||
sendSupportTicketReplyEmail,
|
||||
sendSupportAdminNotificationEmail,
|
||||
} from "@/lib/email";
|
||||
import { safeError } from "@/lib/errors";
|
||||
import type { SupportTicketStatus } from "@/types";
|
||||
|
||||
/**
|
||||
* Comments on a support ticket (Feature 5). Threaded chronologically;
|
||||
* no nested replies.
|
||||
*
|
||||
* Auto status transitions on comment:
|
||||
* - Customer reply on a `waiting_for_customer` → `in_progress`
|
||||
* (the ball is back in admin's court).
|
||||
* - Customer reply on a `resolved` ticket → `reopened`
|
||||
* (customer disagreed with the resolution).
|
||||
* - Admin reply on `open` or `reopened` → `in_progress`
|
||||
* (signals admin has engaged).
|
||||
* - Admin reply on `in_progress` → `waiting_for_customer`
|
||||
* (admin's response, ball is in customer's court).
|
||||
* - Otherwise no change.
|
||||
*
|
||||
* The auto-bump is opportunistic — caller may pass an explicit
|
||||
* status override via the PATCH endpoint instead. We only auto-bump
|
||||
* here when no comment-side override is provided (the comment POST
|
||||
* doesn't accept a status field).
|
||||
*
|
||||
* Email rules:
|
||||
* - Customer replies → admin queue gets an "admin notification" email.
|
||||
* - Admin replies → customer gets a reply email (with the body
|
||||
* inline so they can read on mobile without clicking).
|
||||
* - No "you just commented" confirmation back to the author.
|
||||
*
|
||||
* The customer reply path skips the separate status-change email
|
||||
* even when the status auto-bumps, on the principle that one email
|
||||
* per action is enough; the admin will see the reply notification
|
||||
* and the new status in the queue.
|
||||
*/
|
||||
|
||||
const createSchema = z.object({
|
||||
body: z.string().trim().min(1, "required").max(10_000),
|
||||
});
|
||||
|
||||
/**
|
||||
* Compute the auto-bumped status (if any) for a comment from a given
|
||||
* author kind. Returns the new status if it should change, or null
|
||||
* if it should stay the same.
|
||||
*/
|
||||
function autoBumpStatus(
|
||||
current: SupportTicketStatus,
|
||||
authorKind: "customer" | "admin"
|
||||
): SupportTicketStatus | null {
|
||||
if (authorKind === "customer") {
|
||||
if (current === "waiting_for_customer") return "in_progress";
|
||||
if (current === "resolved") return "reopened";
|
||||
return null;
|
||||
}
|
||||
// admin
|
||||
if (current === "open" || current === "reopened") return "in_progress";
|
||||
if (current === "in_progress") return "waiting_for_customer";
|
||||
return null;
|
||||
}
|
||||
|
||||
export async function POST(
|
||||
req: NextRequest,
|
||||
{ params }: { params: Promise<{ id: string }> }
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
const { id } = await params;
|
||||
const ticket = await getSupportTicketById(id);
|
||||
if (!ticket) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
// Same authorization as the GET on the parent resource.
|
||||
if (!user.isPlatform && ticket.zitadelUserId !== user.id) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = createSchema.safeParse(body);
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid input", details: parsed.error.flatten() },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
const authorKind: "customer" | "admin" = user.isPlatform
|
||||
? "admin"
|
||||
: "customer";
|
||||
|
||||
try {
|
||||
const comment = await createSupportTicketComment({
|
||||
ticketId: id,
|
||||
authorUserId: user.id,
|
||||
authorName: user.name,
|
||||
authorKind,
|
||||
body: parsed.data.body,
|
||||
});
|
||||
|
||||
// Auto-bump status if the comment changes the ball's court.
|
||||
const nextStatus = autoBumpStatus(ticket.status, authorKind);
|
||||
if (nextStatus) {
|
||||
await updateSupportTicket(id, { status: nextStatus });
|
||||
}
|
||||
|
||||
// Email the other side. Customer's reply → admin queue;
|
||||
// admin's reply → customer.
|
||||
if (authorKind === "customer") {
|
||||
sendSupportAdminNotificationEmail({
|
||||
reason: "replied",
|
||||
ticketId: ticket.id,
|
||||
title: ticket.title,
|
||||
contactName: ticket.contactName,
|
||||
contactEmail: ticket.contactEmail,
|
||||
body: parsed.data.body,
|
||||
category: ticket.category,
|
||||
}).catch((e) => console.error("admin notification:", e));
|
||||
} else {
|
||||
sendSupportTicketReplyEmail({
|
||||
to: ticket.contactEmail,
|
||||
contactName: ticket.contactName,
|
||||
ticketId: ticket.id,
|
||||
title: ticket.title,
|
||||
authorName: user.name,
|
||||
body: parsed.data.body,
|
||||
}).catch((e) => console.error("reply email:", e));
|
||||
}
|
||||
|
||||
return NextResponse.json({ comment }, { status: 201 });
|
||||
} catch (e: any) {
|
||||
console.error("Failed to create support ticket comment:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to add comment") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
132
src/app/api/support/tickets/[id]/route.ts
Normal file
132
src/app/api/support/tickets/[id]/route.ts
Normal file
@@ -0,0 +1,132 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import {
|
||||
getSupportTicketById,
|
||||
listCommentsForTicket,
|
||||
updateSupportTicket,
|
||||
} from "@/lib/db";
|
||||
import { sendSupportTicketStatusEmail } from "@/lib/email";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Single support ticket detail (Feature 5).
|
||||
*
|
||||
* GET — returns the ticket plus all comments in chronological order.
|
||||
* Authorization: customer sees their own; platform admin sees any.
|
||||
*
|
||||
* PATCH — change status and/or category. Admin only. Sends a status
|
||||
* change email to the customer if status changed, UNLESS the same
|
||||
* call also creates a comment (in that case the comment endpoint
|
||||
* handles the email so the customer doesn't get two messages).
|
||||
*
|
||||
* No DELETE — tickets are durable history. Resolved tickets stay in
|
||||
* the DB for the audit trail.
|
||||
*/
|
||||
|
||||
const patchSchema = z.object({
|
||||
status: z
|
||||
.enum(["open", "in_progress", "waiting_for_customer", "resolved", "reopened"])
|
||||
.optional(),
|
||||
category: z
|
||||
.enum(["bug", "feature_request", "question", "billing", "other"])
|
||||
.optional(),
|
||||
});
|
||||
|
||||
export async function GET(
|
||||
_req: NextRequest,
|
||||
{ params }: { params: Promise<{ id: string }> }
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
const { id } = await params;
|
||||
const ticket = await getSupportTicketById(id);
|
||||
if (!ticket) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
// Authorization: customer can see their own; platform admin can
|
||||
// see any. Owners cannot see their org's tickets — confirmed by
|
||||
// Feature 5 visibility design (per-user, not per-org).
|
||||
if (!user.isPlatform && ticket.zitadelUserId !== user.id) {
|
||||
// Don't leak existence — same 404 as if the ticket didn't exist.
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
const comments = await listCommentsForTicket(id);
|
||||
return NextResponse.json({ ticket, comments });
|
||||
}
|
||||
|
||||
export async function PATCH(
|
||||
req: NextRequest,
|
||||
{ params }: { params: Promise<{ id: string }> }
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
const { id } = await params;
|
||||
const ticket = await getSupportTicketById(id);
|
||||
if (!ticket) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = patchSchema.safeParse(body);
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid input", details: parsed.error.flatten() },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
// Authorization: status/category changes are admin-only EXCEPT
|
||||
// the customer can close their own ticket via status='resolved'
|
||||
// (Feature 5 design — gives them an "I figured it out, never mind"
|
||||
// escape hatch). Customer cannot reopen via this endpoint — that
|
||||
// happens automatically when they comment on a resolved ticket
|
||||
// (handled in the comments POST).
|
||||
const isCustomerSelfClose =
|
||||
!user.isPlatform &&
|
||||
ticket.zitadelUserId === user.id &&
|
||||
parsed.data.status === "resolved" &&
|
||||
parsed.data.category === undefined;
|
||||
|
||||
if (!user.isPlatform && !isCustomerSelfClose) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
|
||||
try {
|
||||
const previousStatus = ticket.status;
|
||||
const updated = await updateSupportTicket(id, parsed.data);
|
||||
if (!updated) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
|
||||
// Email customer when admin (not the customer themselves)
|
||||
// changes status. Skip on customer-self-close — they know what
|
||||
// they did. Skip when status didn't actually change (admin
|
||||
// edited only category).
|
||||
if (
|
||||
user.isPlatform &&
|
||||
parsed.data.status !== undefined &&
|
||||
parsed.data.status !== previousStatus
|
||||
) {
|
||||
sendSupportTicketStatusEmail({
|
||||
to: ticket.contactEmail,
|
||||
contactName: ticket.contactName,
|
||||
ticketId: ticket.id,
|
||||
title: ticket.title,
|
||||
newStatus: parsed.data.status,
|
||||
}).catch((e) => console.error("status email:", e));
|
||||
}
|
||||
|
||||
return NextResponse.json({ ticket: updated });
|
||||
} catch (e: any) {
|
||||
console.error("Failed to update support ticket:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to update ticket") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
103
src/app/api/support/tickets/route.ts
Normal file
103
src/app/api/support/tickets/route.ts
Normal file
@@ -0,0 +1,103 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import {
|
||||
createSupportTicket,
|
||||
listSupportTicketsForUser,
|
||||
listAllSupportTickets,
|
||||
} from "@/lib/db";
|
||||
import {
|
||||
sendSupportTicketCreatedEmail,
|
||||
sendSupportAdminNotificationEmail,
|
||||
} from "@/lib/email";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Support tickets API (Feature 5).
|
||||
*
|
||||
* Visibility: tickets are scoped strictly per-user (zitadel_user_id).
|
||||
* Coworkers in the same org cannot see each other's tickets — this
|
||||
* is the team's design choice for privacy. Platform admins see
|
||||
* everything (the admin queue lives at the same UI but pulls from
|
||||
* a different list).
|
||||
*
|
||||
* GET — for platform users, returns all tickets across all users.
|
||||
* For everyone else, returns only the caller's own tickets. The
|
||||
* client decides the rendering based on user role; we just return
|
||||
* the right list.
|
||||
*
|
||||
* POST — creates a ticket, sends a confirmation email to the
|
||||
* customer and a notification email to the admin distribution list.
|
||||
*/
|
||||
|
||||
const createSchema = z.object({
|
||||
title: z.string().trim().min(3, "required").max(200),
|
||||
description: z.string().trim().min(10, "required").max(10_000),
|
||||
category: z.enum(["bug", "feature_request", "question", "billing", "other"]),
|
||||
});
|
||||
|
||||
export async function GET() {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
// Platform admins get the global queue; everyone else sees their
|
||||
// own tickets only. Visibility-by-default-deny: even an org owner
|
||||
// doesn't see their coworkers' tickets, by Feature 5 design.
|
||||
const tickets = user.isPlatform
|
||||
? await listAllSupportTickets()
|
||||
: await listSupportTicketsForUser(user.id);
|
||||
return NextResponse.json({ tickets });
|
||||
}
|
||||
|
||||
export async function POST(req: NextRequest) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = createSchema.safeParse(body);
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid input", details: parsed.error.flatten() },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
try {
|
||||
const ticket = await createSupportTicket({
|
||||
zitadelOrgId: user.orgId,
|
||||
zitadelUserId: user.id,
|
||||
title: parsed.data.title,
|
||||
description: parsed.data.description,
|
||||
category: parsed.data.category,
|
||||
contactName: user.name,
|
||||
contactEmail: user.email,
|
||||
});
|
||||
|
||||
// Fire-and-log email notifications. Both are best-effort;
|
||||
// failure to send doesn't roll back the ticket creation.
|
||||
sendSupportTicketCreatedEmail({
|
||||
to: user.email,
|
||||
contactName: user.name,
|
||||
ticketId: ticket.id,
|
||||
title: ticket.title,
|
||||
}).catch((e) => console.error("ticket created email:", e));
|
||||
sendSupportAdminNotificationEmail({
|
||||
reason: "created",
|
||||
ticketId: ticket.id,
|
||||
title: ticket.title,
|
||||
contactName: user.name,
|
||||
contactEmail: user.email,
|
||||
body: ticket.description,
|
||||
category: ticket.category,
|
||||
}).catch((e) => console.error("admin notification:", e));
|
||||
|
||||
return NextResponse.json({ ticket }, { status: 201 });
|
||||
} catch (e: any) {
|
||||
console.error("Failed to create support ticket:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to create ticket") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user