Both berth-detail surfaces were stubbed/hidden behind a comment in berth-tabs.tsx. Their backing schema already existed; this wires the UI and fills the service gaps. Maintenance Log (was ~60% built: schema/migration/add+get service/route): - new edit + delete: updateMaintenanceLog / deleteMaintenanceLog service (port-scoped tenant guard), PATCH/DELETE at maintenance/[logId], plus updateMaintenanceLogSchema. add schema now accepts null for cost / responsibleParty so the shared add+edit dialog sends one body shape. - BerthMaintenanceTab: list (newest first) + add/edit dialog + delete confirm, realtime invalidation. New berth:maintenanceUpdated/Removed socket events. Waiting List (un-hide the orphaned manager + next-in-line notify): - getWaitingList now left-joins the client so the queue renders names, not raw ids. - WaitingListManager rewritten: ClientPicker instead of free-text id, client names, manage_waiting_list gating on add/reorder/remove, and a "Next in line" marker on position 1. - notifyWaitlistNextInLine: when a berth transitions to available, surface the #1 client to staff who hold berths.manage_waiting_list (mirrors the interest-based notifyNextInLine; dedupeKey-suppressed). Hooked into updateBerthStatus on any -> available transition. Tests: maintenance add/get/update/delete + cross-port guard; waitlist notify recipient-resolution / payload / empty + no-permission no-ops. Verified end-to-end in the browser (create/render/delete for both). Also adds scripts/dev-reset-admin-pw.ts (reset a synthetic user's password via the better-auth hasher after a dev reseed). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
29 lines
959 B
TypeScript
29 lines
959 B
TypeScript
import 'dotenv/config';
|
|
import { and, eq } from 'drizzle-orm';
|
|
|
|
import { auth } from '@/lib/auth';
|
|
import { db } from '@/lib/db';
|
|
import { user, account } from '@/lib/db/schema/users';
|
|
|
|
async function main() {
|
|
const email = process.argv[2] ?? 'admin@portnimara.test';
|
|
const pw = process.argv[3] ?? 'SuperAdmin12345!';
|
|
const [u] = await db.select().from(user).where(eq(user.email, email)).limit(1);
|
|
if (!u) throw new Error(`user not found: ${email}`);
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
const ctx = await (auth as any).$context;
|
|
const hash = await ctx.password.hash(pw);
|
|
const res = await db
|
|
.update(account)
|
|
.set({ password: hash })
|
|
.where(and(eq(account.userId, u.id), eq(account.providerId, 'credential')))
|
|
.returning({ id: account.id });
|
|
console.log(`updated ${res.length} credential row(s) for ${email}`);
|
|
process.exit(0);
|
|
}
|
|
|
|
main().catch((e) => {
|
|
console.error(e);
|
|
process.exit(1);
|
|
});
|