feat(yachts): updateYacht + archiveYacht

This commit is contained in:
Matt Ciaccio
2026-04-23 23:52:24 +02:00
parent aaf4847fc2
commit d0ab4b8102
3 changed files with 187 additions and 4 deletions

View File

@@ -1,8 +1,8 @@
import { describe, it, expect } from 'vitest';
import { createYacht } from '@/lib/services/yachts.service';
import { makeClient, makePort, makeAuditMeta } from '../../helpers/factories';
import { createYacht, updateYacht, archiveYacht } from '@/lib/services/yachts.service';
import { makeClient, makePort, makeYacht, makeAuditMeta } from '../../helpers/factories';
import { db } from '@/lib/db';
import { yachtOwnershipHistory } from '@/lib/db/schema';
import { yachts, yachtOwnershipHistory } from '@/lib/db/schema';
import { eq } from 'drizzle-orm';
describe('yachts.service — createYacht', () => {
@@ -65,3 +65,98 @@ describe('yachts.service — createYacht', () => {
).rejects.toThrow(/owner not found/i);
});
});
describe('yachts.service — updateYacht', () => {
it('updates name and notes', async () => {
const port = await makePort();
const client = await makeClient({ portId: port.id });
const yacht = await makeYacht({
portId: port.id,
ownerType: 'client',
ownerId: client.id,
overrides: { name: 'Original Name' },
});
const updated = await updateYacht(
yacht.id,
port.id,
{ name: 'New Name', notes: 'Updated notes' },
makeAuditMeta(),
);
expect(updated.name).toBe('New Name');
expect(updated.notes).toBe('Updated notes');
const [row] = await db.select().from(yachts).where(eq(yachts.id, yacht.id));
expect(row!.name).toBe('New Name');
expect(row!.notes).toBe('Updated notes');
});
it('rejects when id does not exist or is cross-tenant', async () => {
const portA = await makePort();
const portB = await makePort();
const clientInB = await makeClient({ portId: portB.id });
const yachtInB = await makeYacht({
portId: portB.id,
ownerType: 'client',
ownerId: clientInB.id,
});
await expect(
updateYacht(yachtInB.id, portA.id, { name: 'Hijack' }, makeAuditMeta()),
).rejects.toThrow(/yacht/i);
await expect(
updateYacht('nonexistent-id', portA.id, { name: 'Phantom' }, makeAuditMeta()),
).rejects.toThrow(/yacht/i);
});
it('rejects attempt to change currentOwnerId via update', async () => {
const port = await makePort();
const client = await makeClient({ portId: port.id });
const yacht = await makeYacht({
portId: port.id,
ownerType: 'client',
ownerId: client.id,
});
await expect(
updateYacht(
yacht.id,
port.id,
{ currentOwnerId: 'some-other-id' } as unknown as { name: string },
makeAuditMeta(),
),
).rejects.toThrow(/transfer to change ownership/i);
});
});
describe('yachts.service — archiveYacht', () => {
it('sets archivedAt to a non-null timestamp', async () => {
const port = await makePort();
const client = await makeClient({ portId: port.id });
const yacht = await makeYacht({
portId: port.id,
ownerType: 'client',
ownerId: client.id,
});
await archiveYacht(yacht.id, port.id, makeAuditMeta());
const [row] = await db.select().from(yachts).where(eq(yachts.id, yacht.id));
expect(row!.archivedAt).not.toBeNull();
});
it('throws NotFound for cross-tenant or missing yacht', async () => {
const portA = await makePort();
const portB = await makePort();
const clientInB = await makeClient({ portId: portB.id });
const yachtInB = await makeYacht({
portId: portB.id,
ownerType: 'client',
ownerId: clientInB.id,
});
await expect(archiveYacht(yachtInB.id, portA.id, makeAuditMeta())).rejects.toThrow(/yacht/i);
});
});