feat(assets): thread previousStatus through movement writes

This commit is contained in:
2026-06-25 03:25:47 +02:00
parent a0a1e1bdc8
commit 1142855578
3 changed files with 132 additions and 1 deletions
+4 -1
View File
@@ -1,4 +1,4 @@
import type { InventoryMovementType, Prisma } from "@/generated/prisma/client"
import type { AssetStatus, InventoryMovementType, Prisma } from "@/generated/prisma/client"
import { paginate } from "@/lib/paginate"
import prisma from "@/lib/prisma"
import type { CreateMovementFormType } from "@/schemas/movement.schema"
@@ -115,6 +115,7 @@ function getMovementPerson(movement: {
type CreateMovementServiceInput = CreateMovementFormType & {
userId: string
stockDeltaSign?: 1 | -1
previousStatus?: AssetStatus | null
}
export const MovementService = {
@@ -190,6 +191,7 @@ export const MovementService = {
type,
userId,
stockDeltaSign,
previousStatus,
...rest
} = data
const sign = stockDeltaSign ?? stockDeltaSignMap[type]
@@ -228,6 +230,7 @@ export const MovementService = {
? {
create: {
assetId,
previousStatus: previousStatus ?? null,
newStatus: asset.status,
},
}
+9
View File
@@ -186,6 +186,7 @@ export async function createAssetUseCase(
personId: createdAssignment?.personId || undefined,
assignmentId: createdAssignment?.id,
userId: actorId,
previousStatus: null,
},
tx,
)
@@ -352,6 +353,7 @@ export async function updateAssetUseCase(
personId: activeAssignment.personId || undefined,
assignmentId: activeAssignment.id,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -384,6 +386,7 @@ export async function updateAssetUseCase(
: "ADJUSTMENT",
details: `Status changed from ${transition.previousStatus} to ${transition.nextStatus}`,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -408,6 +411,7 @@ export async function updateAssetUseCase(
type: "OUT",
details: `Asset moved from item ${transition.previousItemId} to ${transition.nextItemId}`,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -420,6 +424,7 @@ export async function updateAssetUseCase(
type: "IN",
details: `Asset moved from item ${transition.previousItemId} to ${transition.nextItemId}`,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -444,6 +449,7 @@ export async function updateAssetUseCase(
type: "OUT",
details: `Asset assigned from item ${transition.previousItemId} to item ${transition.nextItemId}`,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -463,6 +469,7 @@ export async function updateAssetUseCase(
personId: activeAssignment.personId || undefined,
assignmentId: activeAssignment.id,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -476,6 +483,7 @@ export async function updateAssetUseCase(
personId: transition.nextPersonId,
assignmentId: activeAssignment.id,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -527,6 +535,7 @@ export async function updateAssetUseCase(
personId: transition.nextPersonId,
assignmentId: createdAssignment.id,
userId: actorId,
previousStatus: transition.previousStatus,
},
tx,
)
@@ -446,6 +446,11 @@ describe("asset use-cases", () => {
previousStock: 0,
newStock: 1,
})
expect(movements[0].assetLines[0]).toMatchObject({
assetId: created.assetId,
previousStatus: null,
newStatus: "AVAILABLE",
})
expect(movements[1]).toMatchObject({
assignmentId: activeAssignment.id,
performedById: actor.id,
@@ -458,6 +463,8 @@ describe("asset use-cases", () => {
})
expect(movements[1].assetLines[0]).toMatchObject({
assetId: created.assetId,
previousStatus: "AVAILABLE",
newStatus: "ASSIGNED",
})
expect(movements[2]).toMatchObject({
assignmentId: activeAssignment.id,
@@ -471,9 +478,121 @@ describe("asset use-cases", () => {
})
expect(movements[2].assetLines[0]).toMatchObject({
assetId: created.assetId,
previousStatus: "ASSIGNED",
newStatus: "AVAILABLE",
})
})
it("sets previousStatus to null on first asset status (create AVAILABLE)", async () => {
const actor = await createTestUser(prisma)
const item = await createTestItem(prisma, { stock: 0 })
const result = await createAssetUseCase({
actorId: actor.id,
itemId: item.id,
serialNumber: "ASSET-CREATE-PREV-001",
status: "AVAILABLE",
})
expect(result.success).toBe(true)
if (!result.success) throw new Error("Expected asset creation success")
const movements = await prisma.inventoryMovement.findMany({
include: { assetLines: true },
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
})
expect(movements).toHaveLength(1)
expect(movements[0].assetLines[0]).toMatchObject({
assetId: result.assetId,
previousStatus: null,
newStatus: "AVAILABLE",
})
})
it("records previousStatus AVAILABLE on AVAILABLE to BROKEN transition", async () => {
const actor = await createTestUser(prisma)
const item = await createTestItem(prisma, { stock: 0 })
const created = await createAssetUseCase({
actorId: actor.id,
itemId: item.id,
serialNumber: "ASSET-BROKEN-PREV-001",
status: "AVAILABLE",
})
expect(created.success).toBe(true)
if (!created.success) throw new Error("Expected asset creation success")
const updated = await updateAssetUseCase({
actorId: actor.id,
id: created.assetId,
itemId: item.id,
serialNumber: "ASSET-BROKEN-PREV-001",
status: "BROKEN",
})
expect(updated.success).toBe(true)
const movements = await prisma.inventoryMovement.findMany({
include: { assetLines: true },
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
})
expect(movements).toHaveLength(2)
expect(movements[1].assetLines[0]).toMatchObject({
assetId: created.assetId,
previousStatus: "AVAILABLE",
newStatus: "BROKEN",
})
})
it("writes two AssetMovementLines with correct previousStatus on itemChanged AVAILABLE to AVAILABLE", async () => {
const actor = await createTestUser(prisma)
const sourceItem = await createTestItem(prisma, { stock: 1 })
const targetItem = await createTestItem(prisma, { stock: 0 })
const created = await createAssetUseCase({
actorId: actor.id,
itemId: sourceItem.id,
serialNumber: "ASSET-ITEMCHG-001",
status: "AVAILABLE",
})
expect(created.success).toBe(true)
if (!created.success) throw new Error("Expected asset creation success")
const updated = await updateAssetUseCase({
actorId: actor.id,
id: created.assetId,
itemId: targetItem.id,
serialNumber: "ASSET-ITEMCHG-001",
status: "AVAILABLE",
})
expect(updated.success).toBe(true)
const movements = await prisma.inventoryMovement.findMany({
include: { assetLines: true, stockLines: true },
orderBy: [{ createdAt: "asc" }, { id: "asc" }],
})
const itemChangedMovements = movements.slice(1)
expect(itemChangedMovements).toHaveLength(2)
const allAssetLines = itemChangedMovements.flatMap(
(movement) => movement.assetLines,
)
expect(allAssetLines).toHaveLength(2)
for (const line of allAssetLines) {
expect(line).toMatchObject({
assetId: created.assetId,
previousStatus: "AVAILABLE",
newStatus: "AVAILABLE",
})
}
})
it("rejects updating an asset to assigned without a person", async () => {
const actor = await createTestUser(prisma)
const item = await createTestItem(prisma, { stock: 0 })