feat(assignments): close and reopen assignment on person swap

This commit is contained in:
2026-06-25 17:20:24 +02:00
parent b4b63e107a
commit 18e274ef37
5 changed files with 472 additions and 78 deletions
+11
View File
@@ -0,0 +1,11 @@
# Changelog
## Unreleased
### Breaking Changes
- **W-3 (fix-assignment-asset-movement-audit)**: `Assignment.id` is no longer stable across person swaps. When `updateAssignmentUseCase` or `updateAssetUseCase` changes a person on an active assignment, the old assignment is now closed (with `closedAt`/`closedById` set; `AssignmentStockReturn` row created for QUANTITY; `AssignmentAssetLine.returnedAt` set for SERIALIZED) and a NEW `Assignment` is created with a new `id`. Any code that holds an `Assignment.id` expecting stability across person swaps must be updated to look up the new id (e.g., via the active `Assignment` for the person/asset). The change replaces a phantom `RETURN` + `ASSIGNMENT` pattern that wrote both movements to the same `Assignment.id` without creating an `AssignmentStockReturn` row.
## Released
_Unreleased changes appear above. Released versions will be added below as they are tagged._
+9 -39
View File
@@ -6,6 +6,7 @@ import { AssignmentService } from "@/services/assignment.service"
import { ItemService } from "@/services/item.service"
import { MovementService } from "@/services/movement.service"
import type { Assignment } from "@/types"
import { reassignAssignment } from "@/use-cases/assignment.use-cases"
type FieldErrors = Record<string, string[]>
@@ -472,46 +473,15 @@ export async function updateAssetUseCase(
if (activeAssignment) {
if (transition.personChanged) {
await MovementService.create(
{
type: "RETURN",
quantity: activeAssignment.quantity || 1,
itemId: activeAssignment.itemId || undefined,
assetId: activeAssignment.assetId || undefined,
personId: activeAssignment.personId || undefined,
assignmentId: activeAssignment.id,
userId: actorId,
previousStatus: transition.previousStatus,
},
await reassignAssignment({
oldAssignment: activeAssignment,
newPersonId: transition.nextPersonId,
newItemId: itemId,
newAssetId: id,
newQuantity: 1,
actorId,
tx,
)
await MovementService.create(
{
type: "ASSIGNMENT",
quantity: 1,
itemId,
assetId: id,
personId: transition.nextPersonId,
assignmentId: activeAssignment.id,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
await AssignmentService.update(
activeAssignment.id,
{
createdBy: actorId,
itemId,
assetId: id,
personId: transition.nextPersonId,
quantity: 1,
returnDate: null,
},
tx,
)
})
} else {
await AssignmentService.update(
activeAssignment.id,
+103 -39
View File
@@ -1,3 +1,4 @@
import { type Prisma } from "@/generated/prisma/client"
import prisma from "@/lib/prisma"
import type {
CreateAssignmentData,
@@ -8,6 +9,7 @@ import { AssignmentService } from "@/services/assignment.service"
import { ItemService } from "@/services/item.service"
import { MovementService } from "@/services/movement.service"
import type {
Assignment,
AssignmentStockLineInput,
AssignmentStockReturnInput,
} from "@/types"
@@ -123,6 +125,98 @@ function resolveAssignmentLine(
}
}
export type ReassignAssignmentArgs = {
oldAssignment: Assignment
newPersonId: string
newItemId?: string
newAssetId?: string
newQuantity: number
newNotes?: string
actorId: string
tx: Prisma.TransactionClient
}
export async function reassignAssignment(
args: ReassignAssignmentArgs,
): Promise<{ newAssignment: Assignment }> {
const {
oldAssignment,
newPersonId,
newItemId,
newAssetId,
newQuantity,
newNotes,
actorId,
tx,
} = args
const returnedQuantity = oldAssignment.quantity ?? 1
// 1. Close old assignment (handles both QUANTITY and SERIALIZED via markReturnedIfActive)
await AssignmentService.markReturnedIfActive(oldAssignment.id, actorId, tx)
// 2. Mutate item.stock to restore the returned quantity, then write RETURN movement
// Skip for SERIALIZED items (stock is constrained to 0 and is not affected by assignments).
if (oldAssignment.itemId) {
const oldItem = await ItemService.findById(oldAssignment.itemId, tx)
if (oldItem && oldItem.trackingType === "QUANTITY") {
await ItemService.updateStock(
oldAssignment.itemId,
returnedQuantity,
tx,
)
}
}
await MovementService.create(
{
type: "RETURN",
quantity: returnedQuantity,
itemId: oldAssignment.itemId || undefined,
assetId: oldAssignment.assetId || undefined,
personId: oldAssignment.personId || undefined,
assignmentId: oldAssignment.id,
userId: actorId,
},
tx,
)
// 3. Create new assignment for the new person
const newAssignment = await AssignmentService.create(
{
personId: newPersonId,
itemId: newItemId,
assetId: newAssetId,
quantity: newQuantity,
notes: newNotes,
assignmentDate: new Date(),
createdBy: actorId,
},
tx,
)
// 4. Mutate item.stock to deduct the new assigned quantity, then write ASSIGNMENT movement
if (newItemId) {
const newItem = await ItemService.findById(newItemId, tx)
if (newItem && newItem.trackingType === "QUANTITY") {
await ItemService.updateStock(newItemId, -newQuantity, tx)
}
}
await MovementService.create(
{
type: "ASSIGNMENT",
quantity: newQuantity,
itemId: newItemId,
assetId: newAssetId,
personId: newPersonId,
assignmentId: newAssignment.id,
userId: actorId,
},
tx,
)
return { newAssignment }
}
export async function createAssignmentUseCase(
input: CreateAssignmentUseCaseInput,
): Promise<CreateAssignmentUseCaseResult> {
@@ -276,46 +370,16 @@ export async function updateAssignmentUseCase(
}
if (assignment.personId !== personId) {
await MovementService.create(
{
type: "RETURN",
quantity: assignment.quantity || 1,
itemId: assignment.itemId || undefined,
assetId: assignment.assetId || undefined,
personId: assignment.personId || undefined,
assignmentId: id,
userId: actorId,
},
await reassignAssignment({
oldAssignment: assignment,
newPersonId: personId,
newItemId: itemId,
newAssetId: assetId,
newQuantity: quantity,
newNotes: notes,
actorId,
tx,
)
await MovementService.create(
{
type: "ASSIGNMENT",
quantity,
itemId,
assetId: assetId || undefined,
personId,
assignmentId: id,
userId: actorId,
},
tx,
)
await AssignmentService.update(
id,
{
createdBy: actorId,
personId: personId,
itemId,
assetId,
quantity,
notes,
assignmentDate,
returnDate: null,
},
tx,
)
})
} else {
if (item && assignment.quantity !== quantity) {
const stockDelta = quantity - (assignment.quantity ?? 0)
@@ -832,4 +832,113 @@ describe("asset use-cases", () => {
newStatus: "ASSIGNED",
})
})
it("closes the old assignment and creates a new one on person swap via updateAssetUseCase", async () => {
const actor = await createTestUser(prisma)
const personA = await createTestPerson(prisma, { firstName: "Alice" })
const personB = await createTestPerson(prisma, { firstName: "Bob" })
const category = await createTestCategory(prisma)
const item = await prisma.item.create({
data: {
sku: "W3-ASSET-SKU",
name: "Asset Item",
trackingType: "SERIALIZED",
stock: 0,
category: { connect: { id: category.id } },
},
})
const created = await createAssetUseCase({
actorId: actor.id,
itemId: item.id,
serialNumber: "W3-ASSET-001",
status: "ASSIGNED",
personId: personA.id,
})
expect(created.success).toBe(true)
if (!created.success) throw new Error("Expected asset creation success")
const oldAssignment = await prisma.assignment.findFirstOrThrow({
where: {
status: "OPEN",
assetLines: { some: { assetId: created.assetId, returnedAt: null } },
},
})
const updated = await updateAssetUseCase({
actorId: actor.id,
id: created.assetId,
itemId: item.id,
serialNumber: "W3-ASSET-001",
status: "ASSIGNED",
personId: personB.id,
})
expect(updated.success).toBe(true)
if (!updated.success) throw new Error("Expected asset update success")
const [oldAssignmentAfter, newAssignment, movements] = await Promise.all([
prisma.assignment.findUniqueOrThrow({
where: { id: oldAssignment.id },
include: { assetLines: true },
}),
prisma.assignment.findFirstOrThrow({
where: { personId: personB.id, status: "OPEN" },
include: { assetLines: true },
}),
prisma.inventoryMovement.findMany({
include: {
stockLines: true,
assetLines: true,
assignment: { select: { personId: true } },
},
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
}),
])
expect(oldAssignmentAfter.status).toBe("RETURNED")
expect(oldAssignmentAfter.closedAt).toBeInstanceOf(Date)
expect(oldAssignmentAfter.assetLines[0].returnedAt).toBeInstanceOf(Date)
expect(newAssignment.id).not.toBe(oldAssignment.id)
expect(newAssignment.personId).toBe(personB.id)
expect(newAssignment.assetLines[0]).toMatchObject({
assetId: created.assetId,
returnedAt: null,
})
expect(movements).toHaveLength(3)
expect(movements.map((m) => m.type)).toEqual([
"ASSIGNMENT",
"RETURN",
"ASSIGNMENT",
])
for (const movement of movements) {
expect(movement.stockLines).toEqual([])
}
const returnMovement = movements[1]
const newAssignmentMovement = movements[2]
expect(returnMovement).toMatchObject({
type: "RETURN",
assignmentId: oldAssignment.id,
})
expect(returnMovement.assignment).toMatchObject({
personId: personA.id,
})
expect(returnMovement.assetLines[0]).toMatchObject({
assetId: created.assetId,
})
expect(newAssignmentMovement).toMatchObject({
type: "ASSIGNMENT",
assignmentId: newAssignment.id,
})
expect(newAssignmentMovement.assignment).toMatchObject({
personId: personB.id,
})
expect(newAssignmentMovement.assetLines[0]).toMatchObject({
assetId: created.assetId,
})
})
})
@@ -644,4 +644,244 @@ describe("assignment use-cases", () => {
expect(movements).toHaveLength(movementsBefore)
expect(movements.some((movement) => movement.type === "ADJUSTMENT")).toBe(false)
})
it("closes the old assignment and creates a new one with RETURN+ASSIGNMENT pair on person swap for QUANTITY", async () => {
const actor = await createTestUser(prisma)
const personA = await createTestPerson(prisma, { firstName: "Alice" })
const personB = await createTestPerson(prisma, { firstName: "Bob" })
const item = await createTestItem(prisma, { stock: 5 })
const created = await createAssignmentUseCase({
actorId: actor.id,
personId: personA.id,
lines: [{ itemId: item.id, quantity: 2 }],
})
expect(created.success).toBe(true)
if (!created.success) throw new Error("Expected assignment creation success")
const updated = await updateAssignmentUseCase({
actorId: actor.id,
id: created.assignmentId,
personId: personB.id,
lines: [{ itemId: item.id, quantity: 2 }],
})
expect(updated.success).toBe(true)
if (!updated.success) throw new Error("Expected assignment update success")
const [oldAssignment, newAssignment, updatedItem, movements] =
await Promise.all([
prisma.assignment.findUniqueOrThrow({
where: { id: created.assignmentId },
include: { stockLines: { include: { returns: true } } },
}),
prisma.assignment.findFirstOrThrow({
where: { personId: personB.id, status: "OPEN" },
include: { stockLines: true },
}),
prisma.item.findUniqueOrThrow({ where: { id: item.id } }),
prisma.inventoryMovement.findMany({
include: {
stockLines: true,
assignment: { select: { personId: true } },
},
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
}),
])
expect(oldAssignment.status).toBe("RETURNED")
expect(oldAssignment.closedAt).toBeInstanceOf(Date)
expect(oldAssignment.closedById).toBe(actor.id)
expect(oldAssignment.stockLines[0].returns).toHaveLength(1)
expect(oldAssignment.stockLines[0].returns[0]).toMatchObject({
quantity: 2,
receivedById: actor.id,
})
expect(newAssignment.id).not.toBe(created.assignmentId)
expect(newAssignment.personId).toBe(personB.id)
expect(newAssignment.stockLines[0]).toMatchObject({
itemId: item.id,
quantity: 2,
})
expect(updatedItem.stock).toBe(3)
expect(movements).toHaveLength(3)
expect(movements.map((m) => m.type)).toEqual([
"ASSIGNMENT",
"RETURN",
"ASSIGNMENT",
])
const returnMovement = movements[1]
const newAssignmentMovement = movements[2]
expect(returnMovement).toMatchObject({
type: "RETURN",
assignmentId: created.assignmentId,
})
expect(returnMovement.assignment).toMatchObject({
personId: personA.id,
})
expect(returnMovement.stockLines[0]).toMatchObject({
itemId: item.id,
stockDelta: 2,
previousStock: 3,
newStock: 5,
})
expect(newAssignmentMovement).toMatchObject({
type: "ASSIGNMENT",
assignmentId: newAssignment.id,
})
expect(newAssignmentMovement.assignment).toMatchObject({
personId: personB.id,
})
expect(newAssignmentMovement.stockLines[0]).toMatchObject({
itemId: item.id,
stockDelta: -2,
previousStock: 5,
newStock: 3,
})
})
it("closes the old AssignmentAssetLine and creates a new one on person swap for SERIALIZED", async () => {
const actor = await createTestUser(prisma)
const personA = await createTestPerson(prisma, { firstName: "Alice" })
const personB = await createTestPerson(prisma, { firstName: "Bob" })
const category = await createTestCategory(prisma)
const item = await prisma.item.create({
data: {
sku: "W3-SERIAL-SKU",
name: "Serial Item",
trackingType: "SERIALIZED",
stock: 0,
category: { connect: { id: category.id } },
},
})
const asset = await prisma.asset.create({
data: {
serialNumber: "W3-SERIAL-ASSET-001",
itemId: item.id,
status: "AVAILABLE",
},
})
const created = await createAssignmentUseCase({
actorId: actor.id,
personId: personA.id,
assetId: asset.id,
lines: [{ itemId: item.id, quantity: 1 }],
})
expect(created.success).toBe(true)
if (!created.success) throw new Error("Expected assignment creation success")
const updated = await updateAssignmentUseCase({
actorId: actor.id,
id: created.assignmentId,
personId: personB.id,
assetId: asset.id,
lines: [{ itemId: item.id, quantity: 1 }],
})
expect(updated.success).toBe(true)
if (!updated.success) throw new Error("Expected assignment update success")
const [oldAssignment, newAssignment, movements] = await Promise.all([
prisma.assignment.findUniqueOrThrow({
where: { id: created.assignmentId },
include: { assetLines: true },
}),
prisma.assignment.findFirstOrThrow({
where: { personId: personB.id, status: "OPEN" },
include: { assetLines: true },
}),
prisma.inventoryMovement.findMany({
include: {
stockLines: true,
assetLines: true,
assignment: { select: { personId: true } },
},
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
}),
])
expect(oldAssignment.status).toBe("RETURNED")
expect(oldAssignment.closedAt).toBeInstanceOf(Date)
expect(oldAssignment.assetLines[0].returnedAt).toBeInstanceOf(Date)
expect(newAssignment.id).not.toBe(created.assignmentId)
expect(newAssignment.personId).toBe(personB.id)
expect(newAssignment.assetLines[0]).toMatchObject({
assetId: asset.id,
returnedAt: null,
})
expect(movements).toHaveLength(3)
expect(movements.map((m) => m.type)).toEqual([
"ASSIGNMENT",
"RETURN",
"ASSIGNMENT",
])
for (const movement of movements) {
expect(movement.stockLines).toEqual([])
}
const returnMovement = movements[1]
const newAssignmentMovement = movements[2]
expect(returnMovement).toMatchObject({
type: "RETURN",
assignmentId: created.assignmentId,
})
expect(returnMovement.assignment).toMatchObject({
personId: personA.id,
})
expect(returnMovement.assetLines[0]).toMatchObject({
assetId: asset.id,
})
expect(newAssignmentMovement).toMatchObject({
type: "ASSIGNMENT",
assignmentId: newAssignment.id,
})
expect(newAssignmentMovement.assignment).toMatchObject({
personId: personB.id,
})
expect(newAssignmentMovement.assetLines[0]).toMatchObject({
assetId: asset.id,
})
})
it("writes no RETURN or ASSIGNMENT when the person is unchanged on update with quantity change", async () => {
const actor = await createTestUser(prisma)
const person = await createTestPerson(prisma)
const item = await createTestItem(prisma, { stock: 5 })
const created = await createAssignmentUseCase({
actorId: actor.id,
personId: person.id,
lines: [{ itemId: item.id, quantity: 2 }],
})
expect(created.success).toBe(true)
if (!created.success) throw new Error("Expected assignment creation success")
const updated = await updateAssignmentUseCase({
actorId: actor.id,
id: created.assignmentId,
personId: person.id,
lines: [{ itemId: item.id, quantity: 3 }],
})
expect(updated.success).toBe(true)
if (!updated.success) throw new Error("Expected assignment update success")
const movements = await prisma.inventoryMovement.findMany({
include: { stockLines: true },
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
})
// ASSIGNMENT (create) + ADJUSTMENT (W-1) = 2 movements
expect(movements).toHaveLength(2)
expect(movements.map((m) => m.type)).toEqual(["ASSIGNMENT", "ADJUSTMENT"])
expect(movements.some((m) => m.type === "RETURN")).toBe(false)
})
})