refactor(api): extract management level support
This commit is contained in:
@@ -0,0 +1,71 @@
|
|||||||
|
import { TRPCError } from "@trpc/server";
|
||||||
|
import { describe, expect, it, vi } from "vitest";
|
||||||
|
import {
|
||||||
|
assertManagementLevelDeletable,
|
||||||
|
assertManagementLevelGroupNameAvailable,
|
||||||
|
assertManagementLevelNameAvailable,
|
||||||
|
buildManagementLevelCreateData,
|
||||||
|
buildManagementLevelGroupCreateData,
|
||||||
|
buildManagementLevelGroupUpdateData,
|
||||||
|
buildManagementLevelUpdateData,
|
||||||
|
} from "../router/management-level-support.js";
|
||||||
|
|
||||||
|
describe("management level support", () => {
|
||||||
|
it("rejects duplicate group names outside the ignored id", async () => {
|
||||||
|
const db = {
|
||||||
|
managementLevelGroup: {
|
||||||
|
findUnique: vi.fn().mockResolvedValue({ id: "group_existing", name: "Team Leads" }),
|
||||||
|
},
|
||||||
|
} as never;
|
||||||
|
|
||||||
|
await expect(assertManagementLevelGroupNameAvailable(db, "Team Leads")).rejects.toBeInstanceOf(TRPCError);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("rejects duplicate level names outside the ignored id", async () => {
|
||||||
|
const db = {
|
||||||
|
managementLevel: {
|
||||||
|
findUnique: vi.fn().mockResolvedValue({ id: "level_existing", name: "Senior Team Lead" }),
|
||||||
|
},
|
||||||
|
} as never;
|
||||||
|
|
||||||
|
await expect(assertManagementLevelNameAvailable(db, "Senior Team Lead")).rejects.toBeInstanceOf(TRPCError);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("builds create and sparse update payloads", () => {
|
||||||
|
expect(buildManagementLevelGroupCreateData({
|
||||||
|
name: "Team Leads",
|
||||||
|
targetPercentage: 0.72,
|
||||||
|
sortOrder: 10,
|
||||||
|
})).toEqual({
|
||||||
|
name: "Team Leads",
|
||||||
|
targetPercentage: 0.72,
|
||||||
|
sortOrder: 10,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(buildManagementLevelGroupUpdateData({
|
||||||
|
targetPercentage: 0.8,
|
||||||
|
})).toEqual({
|
||||||
|
targetPercentage: 0.8,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(buildManagementLevelCreateData({
|
||||||
|
name: "Senior Team Lead",
|
||||||
|
groupId: "group_1",
|
||||||
|
})).toEqual({
|
||||||
|
name: "Senior Team Lead",
|
||||||
|
groupId: "group_1",
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(buildManagementLevelUpdateData({
|
||||||
|
groupId: "group_2",
|
||||||
|
})).toEqual({
|
||||||
|
groupId: "group_2",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("rejects deletion when a level is still assigned", () => {
|
||||||
|
expect(() => assertManagementLevelDeletable({
|
||||||
|
_count: { resources: 2 },
|
||||||
|
})).toThrow(TRPCError);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,100 @@
|
|||||||
|
import type { Prisma, PrismaClient } from "@capakraken/db";
|
||||||
|
import {
|
||||||
|
CreateManagementLevelGroupSchema,
|
||||||
|
CreateManagementLevelSchema,
|
||||||
|
UpdateManagementLevelGroupSchema,
|
||||||
|
UpdateManagementLevelSchema,
|
||||||
|
} from "@capakraken/shared";
|
||||||
|
import { TRPCError } from "@trpc/server";
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
type ManagementLevelGroupDb = Pick<PrismaClient, "managementLevelGroup">;
|
||||||
|
type ManagementLevelDb = Pick<PrismaClient, "managementLevel">;
|
||||||
|
|
||||||
|
type ManagementLevelDeleteRecord = {
|
||||||
|
_count: {
|
||||||
|
resources: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type CreateManagementLevelGroupInput = z.infer<typeof CreateManagementLevelGroupSchema>;
|
||||||
|
type UpdateManagementLevelGroupInput = z.infer<typeof UpdateManagementLevelGroupSchema>;
|
||||||
|
type CreateManagementLevelInput = z.infer<typeof CreateManagementLevelSchema>;
|
||||||
|
type UpdateManagementLevelInput = z.infer<typeof UpdateManagementLevelSchema>;
|
||||||
|
|
||||||
|
export async function assertManagementLevelGroupNameAvailable(
|
||||||
|
db: ManagementLevelGroupDb,
|
||||||
|
name: string,
|
||||||
|
ignoreId?: string,
|
||||||
|
): Promise<void> {
|
||||||
|
const existing = await db.managementLevelGroup.findUnique({ where: { name } });
|
||||||
|
if (existing && existing.id !== ignoreId) {
|
||||||
|
throw new TRPCError({
|
||||||
|
code: "CONFLICT",
|
||||||
|
message: `Group "${name}" already exists`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function assertManagementLevelNameAvailable(
|
||||||
|
db: ManagementLevelDb,
|
||||||
|
name: string,
|
||||||
|
ignoreId?: string,
|
||||||
|
): Promise<void> {
|
||||||
|
const existing = await db.managementLevel.findUnique({ where: { name } });
|
||||||
|
if (existing && existing.id !== ignoreId) {
|
||||||
|
throw new TRPCError({
|
||||||
|
code: "CONFLICT",
|
||||||
|
message: `Level "${name}" already exists`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildManagementLevelGroupCreateData(
|
||||||
|
input: CreateManagementLevelGroupInput,
|
||||||
|
): Prisma.ManagementLevelGroupUncheckedCreateInput {
|
||||||
|
return {
|
||||||
|
name: input.name,
|
||||||
|
targetPercentage: input.targetPercentage,
|
||||||
|
sortOrder: input.sortOrder,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildManagementLevelGroupUpdateData(
|
||||||
|
input: UpdateManagementLevelGroupInput,
|
||||||
|
): Prisma.ManagementLevelGroupUncheckedUpdateInput {
|
||||||
|
return {
|
||||||
|
...(input.name !== undefined ? { name: input.name } : {}),
|
||||||
|
...(input.targetPercentage !== undefined ? { targetPercentage: input.targetPercentage } : {}),
|
||||||
|
...(input.sortOrder !== undefined ? { sortOrder: input.sortOrder } : {}),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildManagementLevelCreateData(
|
||||||
|
input: CreateManagementLevelInput,
|
||||||
|
): Prisma.ManagementLevelUncheckedCreateInput {
|
||||||
|
return {
|
||||||
|
name: input.name,
|
||||||
|
groupId: input.groupId,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildManagementLevelUpdateData(
|
||||||
|
input: UpdateManagementLevelInput,
|
||||||
|
): Prisma.ManagementLevelUncheckedUpdateInput {
|
||||||
|
return {
|
||||||
|
...(input.name !== undefined ? { name: input.name } : {}),
|
||||||
|
...(input.groupId !== undefined ? { groupId: input.groupId } : {}),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function assertManagementLevelDeletable(
|
||||||
|
level: ManagementLevelDeleteRecord,
|
||||||
|
): void {
|
||||||
|
if (level._count.resources > 0) {
|
||||||
|
throw new TRPCError({
|
||||||
|
code: "PRECONDITION_FAILED",
|
||||||
|
message: `Cannot delete level assigned to ${level._count.resources} resource(s)`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,11 +4,19 @@ import {
|
|||||||
UpdateManagementLevelGroupSchema,
|
UpdateManagementLevelGroupSchema,
|
||||||
UpdateManagementLevelSchema,
|
UpdateManagementLevelSchema,
|
||||||
} from "@capakraken/shared";
|
} from "@capakraken/shared";
|
||||||
import { TRPCError } from "@trpc/server";
|
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { findUniqueOrThrow } from "../db/helpers.js";
|
import { findUniqueOrThrow } from "../db/helpers.js";
|
||||||
import { createAuditEntry } from "../lib/audit.js";
|
import { createAuditEntry } from "../lib/audit.js";
|
||||||
import { adminProcedure, createTRPCRouter, planningReadProcedure } from "../trpc.js";
|
import { adminProcedure, createTRPCRouter, planningReadProcedure } from "../trpc.js";
|
||||||
|
import {
|
||||||
|
assertManagementLevelDeletable,
|
||||||
|
assertManagementLevelGroupNameAvailable,
|
||||||
|
assertManagementLevelNameAvailable,
|
||||||
|
buildManagementLevelCreateData,
|
||||||
|
buildManagementLevelGroupCreateData,
|
||||||
|
buildManagementLevelGroupUpdateData,
|
||||||
|
buildManagementLevelUpdateData,
|
||||||
|
} from "./management-level-support.js";
|
||||||
|
|
||||||
export const managementLevelRouter = createTRPCRouter({
|
export const managementLevelRouter = createTRPCRouter({
|
||||||
// ─── Groups ─────────────────────────────────────────────
|
// ─── Groups ─────────────────────────────────────────────
|
||||||
@@ -39,16 +47,9 @@ export const managementLevelRouter = createTRPCRouter({
|
|||||||
createGroup: adminProcedure
|
createGroup: adminProcedure
|
||||||
.input(CreateManagementLevelGroupSchema)
|
.input(CreateManagementLevelGroupSchema)
|
||||||
.mutation(async ({ ctx, input }) => {
|
.mutation(async ({ ctx, input }) => {
|
||||||
const existing = await ctx.db.managementLevelGroup.findUnique({ where: { name: input.name } });
|
await assertManagementLevelGroupNameAvailable(ctx.db, input.name);
|
||||||
if (existing) {
|
|
||||||
throw new TRPCError({ code: "CONFLICT", message: `Group "${input.name}" already exists` });
|
|
||||||
}
|
|
||||||
const created = await ctx.db.managementLevelGroup.create({
|
const created = await ctx.db.managementLevelGroup.create({
|
||||||
data: {
|
data: buildManagementLevelGroupCreateData(input),
|
||||||
name: input.name,
|
|
||||||
targetPercentage: input.targetPercentage,
|
|
||||||
sortOrder: input.sortOrder,
|
|
||||||
},
|
|
||||||
include: { levels: true },
|
include: { levels: true },
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -75,21 +76,14 @@ export const managementLevelRouter = createTRPCRouter({
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (input.data.name && input.data.name !== existing.name) {
|
if (input.data.name && input.data.name !== existing.name) {
|
||||||
const conflict = await ctx.db.managementLevelGroup.findUnique({ where: { name: input.data.name } });
|
await assertManagementLevelGroupNameAvailable(ctx.db, input.data.name, existing.id);
|
||||||
if (conflict) {
|
|
||||||
throw new TRPCError({ code: "CONFLICT", message: `Group "${input.data.name}" already exists` });
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const before = existing as unknown as Record<string, unknown>;
|
const before = existing as unknown as Record<string, unknown>;
|
||||||
|
|
||||||
const updated = await ctx.db.managementLevelGroup.update({
|
const updated = await ctx.db.managementLevelGroup.update({
|
||||||
where: { id: input.id },
|
where: { id: input.id },
|
||||||
data: {
|
data: buildManagementLevelGroupUpdateData(input.data),
|
||||||
...(input.data.name !== undefined ? { name: input.data.name } : {}),
|
|
||||||
...(input.data.targetPercentage !== undefined ? { targetPercentage: input.data.targetPercentage } : {}),
|
|
||||||
...(input.data.sortOrder !== undefined ? { sortOrder: input.data.sortOrder } : {}),
|
|
||||||
},
|
|
||||||
include: { levels: true },
|
include: { levels: true },
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -118,13 +112,10 @@ export const managementLevelRouter = createTRPCRouter({
|
|||||||
"Group",
|
"Group",
|
||||||
);
|
);
|
||||||
|
|
||||||
const existing = await ctx.db.managementLevel.findUnique({ where: { name: input.name } });
|
await assertManagementLevelNameAvailable(ctx.db, input.name);
|
||||||
if (existing) {
|
|
||||||
throw new TRPCError({ code: "CONFLICT", message: `Level "${input.name}" already exists` });
|
|
||||||
}
|
|
||||||
|
|
||||||
const created = await ctx.db.managementLevel.create({
|
const created = await ctx.db.managementLevel.create({
|
||||||
data: { name: input.name, groupId: input.groupId },
|
data: buildManagementLevelCreateData(input),
|
||||||
});
|
});
|
||||||
|
|
||||||
void createAuditEntry({
|
void createAuditEntry({
|
||||||
@@ -150,20 +141,14 @@ export const managementLevelRouter = createTRPCRouter({
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (input.data.name && input.data.name !== existing.name) {
|
if (input.data.name && input.data.name !== existing.name) {
|
||||||
const conflict = await ctx.db.managementLevel.findUnique({ where: { name: input.data.name } });
|
await assertManagementLevelNameAvailable(ctx.db, input.data.name, existing.id);
|
||||||
if (conflict) {
|
|
||||||
throw new TRPCError({ code: "CONFLICT", message: `Level "${input.data.name}" already exists` });
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const before = existing as unknown as Record<string, unknown>;
|
const before = existing as unknown as Record<string, unknown>;
|
||||||
|
|
||||||
const updated = await ctx.db.managementLevel.update({
|
const updated = await ctx.db.managementLevel.update({
|
||||||
where: { id: input.id },
|
where: { id: input.id },
|
||||||
data: {
|
data: buildManagementLevelUpdateData(input.data),
|
||||||
...(input.data.name !== undefined ? { name: input.data.name } : {}),
|
|
||||||
...(input.data.groupId !== undefined ? { groupId: input.data.groupId } : {}),
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
void createAuditEntry({
|
void createAuditEntry({
|
||||||
@@ -191,12 +176,7 @@ export const managementLevelRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
"Level",
|
"Level",
|
||||||
);
|
);
|
||||||
if (level._count.resources > 0) {
|
assertManagementLevelDeletable(level);
|
||||||
throw new TRPCError({
|
|
||||||
code: "PRECONDITION_FAILED",
|
|
||||||
message: `Cannot delete level assigned to ${level._count.resources} resource(s)`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
await ctx.db.managementLevel.delete({ where: { id: input.id } });
|
await ctx.db.managementLevel.delete({ where: { id: input.id } });
|
||||||
|
|
||||||
void createAuditEntry({
|
void createAuditEntry({
|
||||||
|
|||||||
Reference in New Issue
Block a user