fix(api): harden notification assignee persistence
This commit is contained in:
@@ -82,6 +82,24 @@ describe("notification procedure support", () => {
|
||||
}
|
||||
});
|
||||
|
||||
it("rewrites assignee foreign-key errors to a not found TRPC error", () => {
|
||||
const error = {
|
||||
code: "P2003",
|
||||
meta: { field_name: "Notification_assigneeId_fkey" },
|
||||
};
|
||||
|
||||
try {
|
||||
rethrowNotificationReferenceError(error);
|
||||
throw new Error("expected notification reference error");
|
||||
} catch (caught) {
|
||||
expect(caught).toBeInstanceOf(TRPCError);
|
||||
expect(caught).toMatchObject<Partial<TRPCError>>({
|
||||
code: "NOT_FOUND",
|
||||
message: "Assignee user not found",
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
it("rewrites broadcast source foreign-key errors to a not found TRPC error", () => {
|
||||
const error = {
|
||||
code: "P2003",
|
||||
|
||||
@@ -336,6 +336,45 @@ describe("notification.createBroadcast", () => {
|
||||
);
|
||||
});
|
||||
|
||||
it("rejects immediate broadcasts when transactional persistence support is unavailable", async () => {
|
||||
resolveRecipientsMock.mockResolvedValue(["user_a", "user_b"]);
|
||||
|
||||
const create = vi.fn();
|
||||
const update = vi.fn();
|
||||
const createNotification = vi.fn();
|
||||
const db = {
|
||||
notificationBroadcast: {
|
||||
create,
|
||||
update,
|
||||
},
|
||||
notification: {
|
||||
create: createNotification,
|
||||
},
|
||||
};
|
||||
|
||||
const caller = createManagerCaller(db);
|
||||
|
||||
await expect(caller.createBroadcast({
|
||||
title: "Ops update",
|
||||
targetType: "all",
|
||||
})).rejects.toMatchObject({
|
||||
code: "INTERNAL_SERVER_ERROR",
|
||||
message: "Immediate broadcasts require transactional persistence support.",
|
||||
});
|
||||
|
||||
expect(resolveRecipientsMock).toHaveBeenCalledWith(
|
||||
"all",
|
||||
undefined,
|
||||
db,
|
||||
"user_mgr",
|
||||
);
|
||||
expect(create).not.toHaveBeenCalled();
|
||||
expect(update).not.toHaveBeenCalled();
|
||||
expect(createNotification).not.toHaveBeenCalled();
|
||||
expect(emitNotificationCreated).not.toHaveBeenCalled();
|
||||
expect(emitTaskAssigned).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it("rejects broadcasts with no recipients before opening a broadcast transaction", async () => {
|
||||
resolveRecipientsMock.mockResolvedValue([]);
|
||||
|
||||
@@ -1294,6 +1333,38 @@ describe("notification.assignTask", () => {
|
||||
});
|
||||
expect(emitTaskAssigned).toHaveBeenCalledWith("user_4", "task_9");
|
||||
});
|
||||
|
||||
it("returns NOT_FOUND when assigning a task to a missing assignee", async () => {
|
||||
const findUnique = vi.fn().mockResolvedValue({
|
||||
id: "task_9",
|
||||
category: "TASK",
|
||||
assigneeId: "user_2",
|
||||
});
|
||||
const update = vi.fn().mockRejectedValue({
|
||||
code: "P2003",
|
||||
message: "Foreign key constraint failed",
|
||||
meta: { field_name: "Notification_assigneeId_fkey" },
|
||||
});
|
||||
const db = {
|
||||
notification: {
|
||||
findUnique,
|
||||
update,
|
||||
},
|
||||
};
|
||||
|
||||
const caller = createManagerCaller(db);
|
||||
await expect(caller.assignTask({ id: "task_9", assigneeId: "user_missing" })).rejects.toMatchObject({
|
||||
code: "NOT_FOUND",
|
||||
message: "Assignee user not found",
|
||||
});
|
||||
|
||||
expect(findUnique).toHaveBeenCalledWith({ where: { id: "task_9" } });
|
||||
expect(update).toHaveBeenCalledWith({
|
||||
where: { id: "task_9" },
|
||||
data: { assigneeId: "user_missing" },
|
||||
});
|
||||
expect(emitTaskAssigned).not.toHaveBeenCalledWith("user_missing", "task_9");
|
||||
});
|
||||
});
|
||||
|
||||
// ─── reminders ──────────────────────────────────────────────────────────────
|
||||
|
||||
Reference in New Issue
Block a user