feat(factories): add makeMembership, makeReservation, makeOwnershipTransfer

This commit is contained in:
Matt Ciaccio
2026-04-24 13:19:54 +02:00
parent 94f8b76a03
commit 7abbdd4913
2 changed files with 197 additions and 1 deletions

View File

@@ -10,12 +10,19 @@
* — return in-memory objects suitable for unit tests with mocked `db`. * — return in-memory objects suitable for unit tests with mocked `db`.
*/ */
import { and, eq, sql } from 'drizzle-orm';
import { db } from '@/lib/db'; import { db } from '@/lib/db';
import { ports, type NewPort, type Port } from '@/lib/db/schema/ports'; import { ports, type NewPort, type Port } from '@/lib/db/schema/ports';
import { clients, type NewClient, type Client } from '@/lib/db/schema/clients'; import { clients, type NewClient, type Client } from '@/lib/db/schema/clients';
import { berths, type NewBerth, type Berth } from '@/lib/db/schema/berths'; import { berths, type NewBerth, type Berth } from '@/lib/db/schema/berths';
import { yachts, yachtOwnershipHistory, type NewYacht, type Yacht } from '@/lib/db/schema/yachts'; import { yachts, yachtOwnershipHistory, type NewYacht, type Yacht } from '@/lib/db/schema/yachts';
import { companies, type NewCompany, type Company } from '@/lib/db/schema/companies'; import {
companies,
companyMemberships,
type NewCompany,
type Company,
} from '@/lib/db/schema/companies';
import { berthReservations, type BerthReservation } from '@/lib/db/schema/reservations';
// ─── Port ──────────────────────────────────────────────────────────────────── // ─── Port ────────────────────────────────────────────────────────────────────
@@ -119,6 +126,125 @@ export async function makeCompany(args: {
return company!; return company!;
} }
// ─── Company Membership ──────────────────────────────────────────────────────
export async function makeMembership(args: {
companyId: string;
clientId: string;
role?: string;
roleDetail?: string;
startDate?: Date;
endDate?: Date | null;
isPrimary?: boolean;
notes?: string;
}) {
const [row] = await db
.insert(companyMemberships)
.values({
companyId: args.companyId,
clientId: args.clientId,
role: args.role ?? 'director',
roleDetail: args.roleDetail ?? null,
startDate: args.startDate ?? new Date(),
endDate: args.endDate ?? null,
isPrimary: args.isPrimary ?? false,
notes: args.notes ?? null,
})
.returning();
if (!row) throw new Error('Failed to create membership');
return row;
}
// ─── Berth Reservation ───────────────────────────────────────────────────────
export async function makeReservation(args: {
berthId: string;
portId: string;
clientId: string;
yachtId: string;
status: 'pending' | 'active' | 'ended' | 'cancelled';
startDate?: Date;
endDate?: Date | null;
tenureType?: 'permanent' | 'fixed_term' | 'seasonal';
interestId?: string;
createdBy?: string;
notes?: string;
}): Promise<BerthReservation> {
const [row] = await db
.insert(berthReservations)
.values({
berthId: args.berthId,
portId: args.portId,
clientId: args.clientId,
yachtId: args.yachtId,
interestId: args.interestId ?? null,
status: args.status,
startDate: args.startDate ?? new Date(),
endDate: args.endDate ?? null,
tenureType: args.tenureType ?? 'permanent',
contractFileId: null,
notes: args.notes ?? null,
createdBy: args.createdBy ?? 'seed-user',
})
.returning();
if (!row) throw new Error('Failed to create reservation');
return row;
}
// ─── Yacht Ownership Transfer ────────────────────────────────────────────────
export async function makeOwnershipTransfer(args: {
yachtId: string;
newOwner: { type: 'client' | 'company'; id: string };
effectiveDate?: Date;
transferReason?: string;
transferNotes?: string;
createdBy?: string;
}) {
const effective = args.effectiveDate ?? new Date();
const createdBy = args.createdBy ?? 'seed-user';
return await db.transaction(async (tx) => {
// Close current open row
await tx
.update(yachtOwnershipHistory)
.set({ endDate: effective })
.where(
and(
eq(yachtOwnershipHistory.yachtId, args.yachtId),
sql`${yachtOwnershipHistory.endDate} IS NULL`,
),
);
// Insert new open row
const [newHistory] = await tx
.insert(yachtOwnershipHistory)
.values({
yachtId: args.yachtId,
ownerType: args.newOwner.type,
ownerId: args.newOwner.id,
startDate: effective,
endDate: null,
transferReason: args.transferReason ?? null,
transferNotes: args.transferNotes ?? null,
createdBy,
})
.returning();
// Update yacht's denormalized current owner
const [updated] = await tx
.update(yachts)
.set({
currentOwnerType: args.newOwner.type,
currentOwnerId: args.newOwner.id,
updatedAt: new Date(),
})
.where(eq(yachts.id, args.yachtId))
.returning();
return { history: newHistory!, yacht: updated! };
});
}
// ─── Webhook ────────────────────────────────────────────────────────────────── // ─── Webhook ──────────────────────────────────────────────────────────────────
export interface WebhookData { export interface WebhookData {

View File

@@ -0,0 +1,70 @@
import { describe, it, expect } from 'vitest';
import {
makePort,
makeClient,
makeCompany,
makeBerth,
makeYacht,
makeMembership,
makeReservation,
makeOwnershipTransfer,
} from '../helpers/factories';
import { db } from '@/lib/db';
import { yachtOwnershipHistory } from '@/lib/db/schema';
import { eq, isNull, and } from 'drizzle-orm';
describe('factory helpers smoke', () => {
it('makeMembership inserts a row', async () => {
const port = await makePort();
const client = await makeClient({ portId: port.id });
const company = await makeCompany({ portId: port.id });
const m = await makeMembership({ companyId: company.id, clientId: client.id });
expect(m.role).toBe('director');
expect(m.endDate).toBeNull();
});
it('makeReservation inserts a row in any status', async () => {
const port = await makePort();
const berth = await makeBerth({ portId: port.id });
const client = await makeClient({ portId: port.id });
const yacht = await makeYacht({
portId: port.id,
ownerType: 'client',
ownerId: client.id,
});
const r = await makeReservation({
berthId: berth.id,
portId: port.id,
clientId: client.id,
yachtId: yacht.id,
status: 'pending',
});
expect(r.status).toBe('pending');
});
it('makeOwnershipTransfer closes prior history and opens new', async () => {
const port = await makePort();
const clientA = await makeClient({ portId: port.id });
const clientB = await makeClient({ portId: port.id });
const yacht = await makeYacht({
portId: port.id,
ownerType: 'client',
ownerId: clientA.id,
});
const { yacht: updated } = await makeOwnershipTransfer({
yachtId: yacht.id,
newOwner: { type: 'client', id: clientB.id },
});
expect(updated.currentOwnerId).toBe(clientB.id);
const open = await db
.select()
.from(yachtOwnershipHistory)
.where(
and(eq(yachtOwnershipHistory.yachtId, yacht.id), isNull(yachtOwnershipHistory.endDate)),
);
expect(open).toHaveLength(1);
expect(open[0]!.ownerId).toBe(clientB.id);
});
});