fix(gro66): E2E selector fix + groomer isolation + portal confirm/cancel
* Implement confirm/cancel in customer portal (GRO-50) Backend: - Add POST /api/portal/appointments/:id/confirm endpoint - Validates impersonation session auth and ownership - Rejects past/in-progress, non-pending, or already-cancelled/completed - Sets confirmationStatus="confirmed", confirmedAt, updatedAt - Add POST /api/portal/appointments/:id/cancel endpoint - Same auth/ownership pattern - Rejects past/in-progress or already-cancelled/completed - Sets status="cancelled", confirmationStatus="cancelled", cancelledAt, updatedAt Frontend (Appointments.tsx): - Add confirmationStatus field to Appointment type and mock data - Add ConfirmationSection component: shows status badge + confirm button - Add CancelAppointmentButton: wires to cancel API with loading/error state - Wire existing Cancel button to CancelAppointmentButton - Show confirmation status badge in expanded view for upcoming appointments Co-Authored-By: Paperclip <noreply@paperclip.ing> * feat(gro-48): row-level data scoping for groomer role (RBAC Phase 2) Filter query results at the route handler level when staff role is groomer: - GET /api/appointments: WHERE staffId = groomer OR batherStaffId = groomer - GET /api/appointments/🆔 403 if not assigned to groomer (as staff or bather) - GET /api/clients: Clients with ≥1 appointment for this groomer (via exists subquery) - GET /api/clients/🆔 403 if no appointment linkage - GET /api/pets: Pets owned by groomer-linked clients (via exists subquery) - GET /api/pets/:petId: 403 if no appointment linkage Managers and receptionists: no change. Added exists to @groombook/db exports (was missing from re-export). Added groomerIsolation unit tests for role guard and filter logic. Co-Authored-By: Paperclip <noreply@paperclip.ing> * fix(gro-50): add portal confirm/cancel tests and fix ConfirmationSection state - Add test coverage for POST /portal/appointments/:id/confirm endpoint - Add test coverage for POST /portal/appointments/:id/cancel endpoint - Fix ConfirmationSection not updating local status after successful confirm - Remove unused onCancel prop from ConfirmationSection call site - Fix Appointments.test.tsx missing confirmationStatus field Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * test(gro-50): add ConfirmationSection UI component tests Add tests for the ConfirmationSection component: - Renders correct badge for each confirmationStatus state - Shows/hides Confirm button based on status - Calls confirm API with correct headers - Handles sessionId null case - Shows error messages for 401/403/422 responses - Shows loading state while confirming - Shows success message briefly after confirm - Does not call API if user cancels confirm dialog Co-Authored-By: Paperclip <noreply@paperclip.ing> * fix(gro-48): address QA review feedback — staffRow?.role and portal TS guards - appointments.ts: use staffRow?.role (consistent with clients.ts/pets.ts) to handle undefined staff context safely - portal.ts: add null guards on .returning() results for confirm and cancel endpoints (TS18048: 'updated' is possibly undefined) - All 188 tests passing; TypeScript typecheck clean Co-Authored-By: Paperclip <noreply@paperclip.ing> * fix(gro66): use specific selector for banner visibility assertion Replace ambiguous `getByText("STAFF VIEW")` that matched both the ImpersonationBanner and the CustomerPortal watermark with a precise `getByTestId("impersonation-banner")` selector to eliminate strict mode violations. Co-Authored-By: Paperclip <noreply@paperclip.ing> * fix(gro-66): add missing afterEach to vitest import Co-Authored-By: Paperclip <noreply@paperclip.ing> * fix(gro-48): add icalToken to MANAGER mock after rebase After rebasing onto origin/main (which added icalToken to the staff schema via GRO-107), the MANAGER mock in groomerIsolation.test.ts was missing the new required field. Added icalToken: null to the MANAGER constant. factories.ts is clean (no duplicate icalToken after rebase). Co-Authored-By: Paperclip <noreply@paperclip.ing> * fix(gro-47): add non-null assertions on Drizzle RETURNING results Drizzle's update().returning() types the array element as T | undefined. After the if (!appt) guard, updated is still typed as possibly undefined because RETURNING can succeed with no rows. Add ! assertions since we already guard with the existence check. Co-Authored-By: Paperclip <noreply@paperclip.ing> --------- Co-authored-by: Flea Flicker <fleaflicker@groombook.ai> Co-authored-by: Paperclip <noreply@paperclip.ing> Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com> Co-authored-by: Flea Flicker <flea-flicker@paperclip.ing>
This commit was merged in pull request #128.
This commit is contained in:
committed by
GitHub
parent
8ab6319311
commit
9eb0c3d151
@@ -0,0 +1,104 @@
|
|||||||
|
/**
|
||||||
|
* Groomer Isolation Tests
|
||||||
|
*
|
||||||
|
* Validates row-level data scoping for the groomer role.
|
||||||
|
*
|
||||||
|
* The role guard tests verify the core groomer identification logic.
|
||||||
|
* Integration tests with the real database validate the full filter behavior.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { describe, it, expect } from "vitest";
|
||||||
|
import type { StaffRow } from "../middleware/rbac.js";
|
||||||
|
|
||||||
|
// ─── Mock staff ───────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
const MANAGER: StaffRow = {
|
||||||
|
id: "staff-manager-id",
|
||||||
|
oidcSub: "oidc-manager-sub",
|
||||||
|
role: "manager",
|
||||||
|
name: "Manager McManager",
|
||||||
|
email: "manager@example.com",
|
||||||
|
active: true,
|
||||||
|
icalToken: null,
|
||||||
|
createdAt: new Date(),
|
||||||
|
updatedAt: new Date(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const GROOMER: StaffRow = {
|
||||||
|
...MANAGER,
|
||||||
|
id: "staff-groomer-id",
|
||||||
|
oidcSub: "oidc-groomer-sub",
|
||||||
|
role: "groomer",
|
||||||
|
name: "Groomer Gary",
|
||||||
|
email: "groomer@example.com",
|
||||||
|
};
|
||||||
|
|
||||||
|
const RECEPTIONIST: StaffRow = {
|
||||||
|
...MANAGER,
|
||||||
|
id: "staff-receptionist-id",
|
||||||
|
oidcSub: "oidc-receptionist-sub",
|
||||||
|
role: "receptionist",
|
||||||
|
name: "Receptionist Rita",
|
||||||
|
email: "receptionist@example.com",
|
||||||
|
};
|
||||||
|
|
||||||
|
// ─── Role guard ──────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The isGroomer guard (staffRow?.role === "groomer") is the foundation of
|
||||||
|
* all row-level filtering in appointments.ts, clients.ts, and pets.ts.
|
||||||
|
* These tests verify it handles all roles correctly.
|
||||||
|
*/
|
||||||
|
describe("Groomer role guard", () => {
|
||||||
|
const isGroomer = (s: StaffRow | undefined) => s?.role === "groomer";
|
||||||
|
|
||||||
|
it("manager is not groomer", () => expect(isGroomer(MANAGER)).toBe(false));
|
||||||
|
it("receptionist is not groomer", () => expect(isGroomer(RECEPTIONIST)).toBe(false));
|
||||||
|
it("groomer is groomer", () => expect(isGroomer(GROOMER)).toBe(true));
|
||||||
|
|
||||||
|
/** Safe fallback when staff context is not set (e.g., missing auth middleware) */
|
||||||
|
it("undefined staff is not groomer", () => expect(isGroomer(undefined)).toBe(false));
|
||||||
|
});
|
||||||
|
|
||||||
|
// ─── Groomer filter data shapes ───────────────────────────────────────────────
|
||||||
|
|
||||||
|
/**
|
||||||
|
* These constants match the shape used in route handlers to validate
|
||||||
|
* the groomer filter conditions:
|
||||||
|
* or(eq(appointments.staffId, staffRow.id), eq(appointments.batherStaffId, staffRow.id))
|
||||||
|
* This verifies the groomer can see appointments they own OR bathe.
|
||||||
|
*/
|
||||||
|
describe("Groomer appointment filter data", () => {
|
||||||
|
const GROOMER_APPT = { id: "appt-1", staffId: GROOMER.id, batherStaffId: null as string | null };
|
||||||
|
const BATHER_APPT = { id: "appt-2", staffId: MANAGER.id, batherStaffId: GROOMER.id };
|
||||||
|
const OTHER_APPT = { id: "appt-3", staffId: MANAGER.id, batherStaffId: null as string | null };
|
||||||
|
|
||||||
|
it("groomer appointment has groomer staffId", () => {
|
||||||
|
expect(GROOMER_APPT.staffId).toBe(GROOMER.id);
|
||||||
|
expect(GROOMER_APPT.batherStaffId).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("groomer can see appointment where they are the bather", () => {
|
||||||
|
expect(BATHER_APPT.batherStaffId).toBe(GROOMER.id);
|
||||||
|
expect(BATHER_APPT.staffId).toBe(MANAGER.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("other appointment is not assigned to groomer", () => {
|
||||||
|
expect(OTHER_APPT.staffId).toBe(MANAGER.id);
|
||||||
|
expect(OTHER_APPT.batherStaffId).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("filter: groomer sees only their appointments", () => {
|
||||||
|
const all = [GROOMER_APPT, BATHER_APPT, OTHER_APPT];
|
||||||
|
const groomerView = all.filter(
|
||||||
|
(a) => a.staffId === GROOMER.id || a.batherStaffId === GROOMER.id
|
||||||
|
);
|
||||||
|
expect(groomerView).toHaveLength(2);
|
||||||
|
expect(groomerView.map((a) => a.id)).toEqual(["appt-1", "appt-2"]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("filter: manager sees all appointments", () => {
|
||||||
|
const all = [GROOMER_APPT, BATHER_APPT, OTHER_APPT];
|
||||||
|
expect(all).toHaveLength(3);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -31,6 +31,10 @@ const APPOINTMENT = {
|
|||||||
endTime: futureDate(),
|
endTime: futureDate(),
|
||||||
customerNotes: null,
|
customerNotes: null,
|
||||||
confirmationToken: "secret-token-leak-test",
|
confirmationToken: "secret-token-leak-test",
|
||||||
|
status: "scheduled" as const,
|
||||||
|
confirmationStatus: "pending" as const,
|
||||||
|
confirmedAt: null,
|
||||||
|
cancelledAt: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
let selectSessionRow: Record<string, unknown> | null = null;
|
let selectSessionRow: Record<string, unknown> | null = null;
|
||||||
@@ -246,4 +250,174 @@ describe("PATCH /portal/appointments/:id/notes", () => {
|
|||||||
);
|
);
|
||||||
expect(res.status).toBe(400);
|
expect(res.status).toBe(400);
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// ─── POST /portal/appointments/:id/confirm ────────────────────────────────────
|
||||||
|
|
||||||
|
function jsonPost(path: string, headers?: Record<string, string>) {
|
||||||
|
return app.request(path, {
|
||||||
|
method: "POST",
|
||||||
|
headers,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
describe("POST /portal/appointments/:id/confirm", () => {
|
||||||
|
it("confirms a pending appointment and returns updated status", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, confirmationStatus: "pending" };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(200);
|
||||||
|
const body = await res.json();
|
||||||
|
expect(body.confirmationStatus).toBe("confirmed");
|
||||||
|
expect(body).toHaveProperty("confirmedAt");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 401 without X-Impersonation-Session-Id header", async () => {
|
||||||
|
const res = await jsonPost(`/portal/appointments/${APPOINTMENT_ID}/confirm`);
|
||||||
|
expect(res.status).toBe(401);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 401 with expired session", async () => {
|
||||||
|
selectSessionRow = EXPIRED_SESSION;
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(401);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 403 when appointment belongs to a different client", async () => {
|
||||||
|
selectSessionRow = { ...ACTIVE_SESSION, clientId: "different-client-id" };
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(403);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 422 when appointment is in the past", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, startTime: pastDate() };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(422);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 422 when appointment is not pending confirmation", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, confirmationStatus: "confirmed" };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(422);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 422 when cancelling an already-cancelled appointment", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, status: "cancelled", confirmationStatus: "cancelled" };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(422);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 404 when appointment not found", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = null;
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/nonexistent-id/confirm`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(404);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// ─── POST /portal/appointments/:id/cancel ─────────────────────────────────────
|
||||||
|
|
||||||
|
describe("POST /portal/appointments/:id/cancel", () => {
|
||||||
|
it("cancels a pending appointment and returns updated status", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, confirmationStatus: "pending" };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(200);
|
||||||
|
const body = await res.json();
|
||||||
|
expect(body.status).toBe("cancelled");
|
||||||
|
expect(body.confirmationStatus).toBe("cancelled");
|
||||||
|
expect(body).toHaveProperty("cancelledAt");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 401 without X-Impersonation-Session-Id header", async () => {
|
||||||
|
const res = await jsonPost(`/portal/appointments/${APPOINTMENT_ID}/cancel`);
|
||||||
|
expect(res.status).toBe(401);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 401 with expired session", async () => {
|
||||||
|
selectSessionRow = EXPIRED_SESSION;
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(401);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 403 when appointment belongs to a different client", async () => {
|
||||||
|
selectSessionRow = { ...ACTIVE_SESSION, clientId: "different-client-id" };
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(403);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 422 when appointment is in the past", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, startTime: pastDate() };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(422);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 422 when appointment is already cancelled", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, status: "cancelled", confirmationStatus: "cancelled" };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(422);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 422 when appointment is already completed", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = { ...APPOINTMENT, status: "completed" };
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/${APPOINTMENT_ID}/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(422);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns 404 when appointment not found", async () => {
|
||||||
|
selectSessionRow = ACTIVE_SESSION;
|
||||||
|
selectAppointmentRow = null;
|
||||||
|
const res = await jsonPost(
|
||||||
|
`/portal/appointments/nonexistent-id/cancel`,
|
||||||
|
{ "X-Impersonation-Session-Id": SESSION_ID }
|
||||||
|
);
|
||||||
|
expect(res.status).toBe(404);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
@@ -10,6 +10,7 @@ import {
|
|||||||
lt,
|
lt,
|
||||||
lte,
|
lte,
|
||||||
ne,
|
ne,
|
||||||
|
or,
|
||||||
appointments,
|
appointments,
|
||||||
clients,
|
clients,
|
||||||
pets,
|
pets,
|
||||||
@@ -20,8 +21,9 @@ import {
|
|||||||
} from "@groombook/db";
|
} from "@groombook/db";
|
||||||
import { buildConfirmationEmail, sendEmail } from "../services/email.js";
|
import { buildConfirmationEmail, sendEmail } from "../services/email.js";
|
||||||
import { notifyWaitlistForAppointment } from "../services/waitlistNotify.js";
|
import { notifyWaitlistForAppointment } from "../services/waitlistNotify.js";
|
||||||
|
import type { AppEnv } from "../middleware/rbac.js";
|
||||||
|
|
||||||
export const appointmentsRouter = new Hono();
|
export const appointmentsRouter = new Hono<AppEnv>();
|
||||||
|
|
||||||
const createAppointmentSchema = z.object({
|
const createAppointmentSchema = z.object({
|
||||||
clientId: z.string().uuid(),
|
clientId: z.string().uuid(),
|
||||||
@@ -63,18 +65,31 @@ const updateAppointmentSchema = z.object({
|
|||||||
cascadeMode: z.enum(["this_only", "this_and_future", "all"]).optional(),
|
cascadeMode: z.enum(["this_only", "this_and_future", "all"]).optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
// List appointments, optionally filtered by date range or staffId
|
// List appointments, optionally filtered by date range or staffId.
|
||||||
|
// Groomers see only their own appointments (staffId or batherStaffId).
|
||||||
appointmentsRouter.get("/", async (c) => {
|
appointmentsRouter.get("/", async (c) => {
|
||||||
const db = getDb();
|
const db = getDb();
|
||||||
const from = c.req.query("from");
|
const from = c.req.query("from");
|
||||||
const to = c.req.query("to");
|
const to = c.req.query("to");
|
||||||
const staffId = c.req.query("staffId");
|
const staffId = c.req.query("staffId");
|
||||||
|
const staffRow = c.get("staff");
|
||||||
|
const isGroomer = staffRow?.role === "groomer";
|
||||||
|
|
||||||
const conditions = [];
|
const conditions = [];
|
||||||
if (from) conditions.push(gte(appointments.startTime, new Date(from)));
|
if (from) conditions.push(gte(appointments.startTime, new Date(from)));
|
||||||
if (to) conditions.push(lte(appointments.startTime, new Date(to)));
|
if (to) conditions.push(lte(appointments.startTime, new Date(to)));
|
||||||
if (staffId) conditions.push(eq(appointments.staffId, staffId));
|
if (staffId) conditions.push(eq(appointments.staffId, staffId));
|
||||||
|
|
||||||
|
// Groomer: restrict to their own appointments (as groomer or bather)
|
||||||
|
if (isGroomer) {
|
||||||
|
conditions.push(
|
||||||
|
or(
|
||||||
|
eq(appointments.staffId, staffRow.id),
|
||||||
|
eq(appointments.batherStaffId, staffRow.id)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const rows =
|
const rows =
|
||||||
conditions.length > 0
|
conditions.length > 0
|
||||||
? await db
|
? await db
|
||||||
@@ -92,11 +107,17 @@ appointmentsRouter.get("/", async (c) => {
|
|||||||
|
|
||||||
appointmentsRouter.get("/:id", async (c) => {
|
appointmentsRouter.get("/:id", async (c) => {
|
||||||
const db = getDb();
|
const db = getDb();
|
||||||
|
const staffRow = c.get("staff");
|
||||||
|
const isGroomer = staffRow?.role === "groomer";
|
||||||
const [row] = await db
|
const [row] = await db
|
||||||
.select()
|
.select()
|
||||||
.from(appointments)
|
.from(appointments)
|
||||||
.where(eq(appointments.id, c.req.param("id")));
|
.where(eq(appointments.id, c.req.param("id")));
|
||||||
if (!row) return c.json({ error: "Not found" }, 404);
|
if (!row) return c.json({ error: "Not found" }, 404);
|
||||||
|
// Groomer: 403 if not assigned as groomer or bather
|
||||||
|
if (isGroomer && row.staffId !== staffRow.id && row.batherStaffId !== staffRow.id) {
|
||||||
|
return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
return c.json(row);
|
return c.json(row);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import { Hono } from "hono";
|
import { Hono } from "hono";
|
||||||
import { zValidator } from "@hono/zod-validator";
|
import { zValidator } from "@hono/zod-validator";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { eq, getDb, clients } from "@groombook/db";
|
import { and, eq, exists, getDb, or, clients, appointments } from "@groombook/db";
|
||||||
|
import type { AppEnv } from "../middleware/rbac.js";
|
||||||
|
|
||||||
export const clientsRouter = new Hono();
|
export const clientsRouter = new Hono<AppEnv>();
|
||||||
|
|
||||||
const createClientSchema = z.object({
|
const createClientSchema = z.object({
|
||||||
name: z.string().min(1).max(200),
|
name: z.string().min(1).max(200),
|
||||||
@@ -14,25 +15,72 @@ const createClientSchema = z.object({
|
|||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
// List clients — defaults to active only, ?includeDisabled=true shows all
|
// List clients — defaults to active only, ?includeDisabled=true shows all.
|
||||||
|
// Groomers see only clients with ≥1 appointment assigned to them.
|
||||||
clientsRouter.get("/", async (c) => {
|
clientsRouter.get("/", async (c) => {
|
||||||
const db = getDb();
|
const db = getDb();
|
||||||
const includeDisabled = c.req.query("includeDisabled") === "true";
|
const includeDisabled = c.req.query("includeDisabled") === "true";
|
||||||
const query = includeDisabled
|
const staffRow = c.get("staff");
|
||||||
? db.select().from(clients).orderBy(clients.name)
|
const isGroomer = staffRow?.role === "groomer";
|
||||||
: db.select().from(clients).where(eq(clients.status, "active")).orderBy(clients.name);
|
|
||||||
const rows = await query;
|
// Groomer: subquery for clients with an appointment for this groomer
|
||||||
|
const groomerApptFilter = isGroomer
|
||||||
|
? exists(
|
||||||
|
db
|
||||||
|
.select({ id: appointments.id })
|
||||||
|
.from(appointments)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(appointments.clientId, clients.id),
|
||||||
|
or(
|
||||||
|
eq(appointments.staffId, staffRow.id),
|
||||||
|
eq(appointments.batherStaffId, staffRow.id)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
const conditions = [];
|
||||||
|
if (!includeDisabled) conditions.push(eq(clients.status, "active"));
|
||||||
|
if (groomerApptFilter) conditions.push(groomerApptFilter);
|
||||||
|
|
||||||
|
const rows = await db
|
||||||
|
.select()
|
||||||
|
.from(clients)
|
||||||
|
.where(conditions.length > 0 ? and(...conditions) : undefined)
|
||||||
|
.orderBy(clients.name);
|
||||||
return c.json(rows);
|
return c.json(rows);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Get a single client
|
// Get a single client
|
||||||
clientsRouter.get("/:id", async (c) => {
|
clientsRouter.get("/:id", async (c) => {
|
||||||
const db = getDb();
|
const db = getDb();
|
||||||
|
const clientId = c.req.param("id");
|
||||||
|
const staffRow = c.get("staff");
|
||||||
|
const isGroomer = staffRow?.role === "groomer";
|
||||||
const [row] = await db
|
const [row] = await db
|
||||||
.select()
|
.select()
|
||||||
.from(clients)
|
.from(clients)
|
||||||
.where(eq(clients.id, c.req.param("id")));
|
.where(eq(clients.id, clientId));
|
||||||
if (!row) return c.json({ error: "Not found" }, 404);
|
if (!row) return c.json({ error: "Not found" }, 404);
|
||||||
|
// Groomer: 403 if no appointment linkage to this client
|
||||||
|
if (isGroomer) {
|
||||||
|
const [linkage] = await db
|
||||||
|
.select({ id: appointments.id })
|
||||||
|
.from(appointments)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(appointments.clientId, clientId),
|
||||||
|
or(
|
||||||
|
eq(appointments.staffId, staffRow.id),
|
||||||
|
eq(appointments.batherStaffId, staffRow.id)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.limit(1);
|
||||||
|
if (!linkage) return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
return c.json(row);
|
return c.json(row);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { Hono } from "hono";
|
import { Hono } from "hono";
|
||||||
import { zValidator } from "@hono/zod-validator";
|
import { zValidator } from "@hono/zod-validator";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { eq, getDb, pets } from "@groombook/db";
|
import { and, eq, exists, getDb, or, pets, appointments } from "@groombook/db";
|
||||||
import type { AppEnv } from "../middleware/rbac.js";
|
import type { AppEnv } from "../middleware/rbac.js";
|
||||||
import {
|
import {
|
||||||
getPresignedUploadUrl,
|
getPresignedUploadUrl,
|
||||||
@@ -28,25 +28,70 @@ const createPetSchema = z.object({
|
|||||||
|
|
||||||
const updatePetSchema = createPetSchema.partial().omit({ clientId: true });
|
const updatePetSchema = createPetSchema.partial().omit({ clientId: true });
|
||||||
|
|
||||||
|
// List pets, optionally filtered by clientId.
|
||||||
|
// Groomers see only pets owned by clients with ≥1 appointment for this groomer.
|
||||||
petsRouter.get("/", async (c) => {
|
petsRouter.get("/", async (c) => {
|
||||||
const db = getDb();
|
const db = getDb();
|
||||||
const clientId = c.req.query("clientId");
|
const clientId = c.req.query("clientId");
|
||||||
const query = db.select().from(pets);
|
const staffRow = c.get("staff");
|
||||||
if (clientId) {
|
const isGroomer = staffRow?.role === "groomer";
|
||||||
const rows = await query.where(eq(pets.clientId, clientId));
|
|
||||||
return c.json(rows);
|
// Groomer: filter to pets whose client has an appointment for this groomer
|
||||||
}
|
const groomerClientFilter = isGroomer
|
||||||
const rows = await query;
|
? exists(
|
||||||
|
db
|
||||||
|
.select({ id: appointments.id })
|
||||||
|
.from(appointments)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(appointments.clientId, pets.clientId),
|
||||||
|
or(
|
||||||
|
eq(appointments.staffId, staffRow.id),
|
||||||
|
eq(appointments.batherStaffId, staffRow.id)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
const conditions = [];
|
||||||
|
if (clientId) conditions.push(eq(pets.clientId, clientId));
|
||||||
|
if (groomerClientFilter) conditions.push(groomerClientFilter);
|
||||||
|
|
||||||
|
const rows = await db
|
||||||
|
.select()
|
||||||
|
.from(pets)
|
||||||
|
.where(conditions.length > 0 ? and(...conditions) : undefined);
|
||||||
return c.json(rows);
|
return c.json(rows);
|
||||||
});
|
});
|
||||||
|
|
||||||
petsRouter.get("/:id", async (c) => {
|
petsRouter.get("/:id", async (c) => {
|
||||||
const db = getDb();
|
const db = getDb();
|
||||||
|
const petId = c.req.param("id");
|
||||||
|
const staffRow = c.get("staff");
|
||||||
|
const isGroomer = staffRow?.role === "groomer";
|
||||||
const [row] = await db
|
const [row] = await db
|
||||||
.select()
|
.select()
|
||||||
.from(pets)
|
.from(pets)
|
||||||
.where(eq(pets.id, c.req.param("id")));
|
.where(eq(pets.id, petId));
|
||||||
if (!row) return c.json({ error: "Not found" }, 404);
|
if (!row) return c.json({ error: "Not found" }, 404);
|
||||||
|
// Groomer: 403 if no appointment linkage to this pet's client
|
||||||
|
if (isGroomer) {
|
||||||
|
const [linkage] = await db
|
||||||
|
.select({ id: appointments.id })
|
||||||
|
.from(appointments)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(appointments.clientId, row.clientId),
|
||||||
|
or(
|
||||||
|
eq(appointments.staffId, staffRow.id),
|
||||||
|
eq(appointments.batherStaffId, staffRow.id)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.limit(1);
|
||||||
|
if (!linkage) return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
return c.json(row);
|
return c.json(row);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -77,6 +77,141 @@ portalRouter.patch(
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// ─── Appointment confirm/cancel ──────────────────────────────────────────────
|
||||||
|
|
||||||
|
portalRouter.post("/appointments/:id/confirm", async (c) => {
|
||||||
|
const db = getDb();
|
||||||
|
const id = c.req.param("id");
|
||||||
|
|
||||||
|
const sessionId = c.req.header("X-Impersonation-Session-Id");
|
||||||
|
if (!sessionId) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [session] = await db
|
||||||
|
.select()
|
||||||
|
.from(impersonationSessions)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(impersonationSessions.id, sessionId),
|
||||||
|
eq(impersonationSessions.status, "active")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!session || session.expiresAt <= new Date()) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [appt] = await db
|
||||||
|
.select()
|
||||||
|
.from(appointments)
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!appt) {
|
||||||
|
return c.json({ error: "Not found" }, 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.clientId !== session.clientId) {
|
||||||
|
return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.startTime <= new Date()) {
|
||||||
|
return c.json({ error: "Cannot confirm a past or in-progress appointment" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.confirmationStatus !== "pending") {
|
||||||
|
return c.json({ error: "Appointment is not pending confirmation" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.status === "cancelled" || appt.status === "completed") {
|
||||||
|
return c.json({ error: "Cannot confirm a cancelled or completed appointment" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [updated] = await db
|
||||||
|
.update(appointments)
|
||||||
|
.set({ confirmationStatus: "confirmed", confirmedAt: new Date(), updatedAt: new Date() })
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.returning();
|
||||||
|
|
||||||
|
if (!updated) {
|
||||||
|
return c.json({ error: "Not found" }, 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.json({
|
||||||
|
id: updated!.id,
|
||||||
|
confirmationStatus: updated!.confirmationStatus,
|
||||||
|
confirmedAt: updated!.confirmedAt,
|
||||||
|
updatedAt: updated!.updatedAt,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
portalRouter.post("/appointments/:id/cancel", async (c) => {
|
||||||
|
const db = getDb();
|
||||||
|
const id = c.req.param("id");
|
||||||
|
|
||||||
|
const sessionId = c.req.header("X-Impersonation-Session-Id");
|
||||||
|
if (!sessionId) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [session] = await db
|
||||||
|
.select()
|
||||||
|
.from(impersonationSessions)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(impersonationSessions.id, sessionId),
|
||||||
|
eq(impersonationSessions.status, "active")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!session || session.expiresAt <= new Date()) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [appt] = await db
|
||||||
|
.select()
|
||||||
|
.from(appointments)
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!appt) {
|
||||||
|
return c.json({ error: "Not found" }, 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.clientId !== session.clientId) {
|
||||||
|
return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.startTime <= new Date()) {
|
||||||
|
return c.json({ error: "Cannot cancel a past or in-progress appointment" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.status === "cancelled" || appt.status === "completed") {
|
||||||
|
return c.json({ error: "Appointment is already cancelled or completed" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [updated] = await db
|
||||||
|
.update(appointments)
|
||||||
|
.set({ status: "cancelled", confirmationStatus: "cancelled", cancelledAt: new Date(), updatedAt: new Date() })
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.returning();
|
||||||
|
|
||||||
|
if (!updated) {
|
||||||
|
return c.json({ error: "Not found" }, 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.json({
|
||||||
|
id: updated!.id,
|
||||||
|
status: updated!.status,
|
||||||
|
confirmationStatus: updated!.confirmationStatus,
|
||||||
|
cancelledAt: updated!.cancelledAt,
|
||||||
|
updatedAt: updated!.updatedAt,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// ─── Client-facing waitlist routes ───────────────────────────────────────────
|
// ─── Client-facing waitlist routes ───────────────────────────────────────────
|
||||||
|
|
||||||
const createWaitlistEntrySchema = z.object({
|
const createWaitlistEntrySchema = z.object({
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ test.describe("ImpersonationBanner", () => {
|
|||||||
test("clicking End Session calls API and redirects", async ({ page }) => {
|
test("clicking End Session calls API and redirects", async ({ page }) => {
|
||||||
await page.goto("/?sessionId=session-1");
|
await page.goto("/?sessionId=session-1");
|
||||||
await page.getByRole("button", { name: /End Session/ }).click();
|
await page.getByRole("button", { name: /End Session/ }).click();
|
||||||
await expect(page.getByText("STAFF VIEW")).not.toBeVisible();
|
await expect(page.getByTestId("impersonation-banner")).not.toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("Extend button appears when time is low and not extended", async ({ page }) => {
|
test("Extend button appears when time is low and not extended", async ({ page }) => {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { describe, it, expect, vi, beforeEach } from "vitest";
|
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
||||||
import { render, screen, fireEvent, waitFor } from "@testing-library/react";
|
import { render, screen, fireEvent, waitFor } from "@testing-library/react";
|
||||||
import type { Appointment } from "../portal/mockData.js";
|
import type { Appointment } from "../portal/mockData.js";
|
||||||
import { parseTimeTo24Hour, isUpcoming, CustomerNotesSection } from "../portal/sections/Appointments.js";
|
import { parseTimeTo24Hour, isUpcoming, CustomerNotesSection, ConfirmationSection } from "../portal/sections/Appointments.js";
|
||||||
|
|
||||||
const UPCOMING_APPT: Appointment = {
|
const UPCOMING_APPT: Appointment = {
|
||||||
id: "appt-1",
|
id: "appt-1",
|
||||||
@@ -18,6 +18,7 @@ const UPCOMING_APPT: Appointment = {
|
|||||||
status: "confirmed",
|
status: "confirmed",
|
||||||
notes: "",
|
notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
|
confirmationStatus: "pending",
|
||||||
};
|
};
|
||||||
|
|
||||||
const PAST_APPT: Appointment = {
|
const PAST_APPT: Appointment = {
|
||||||
@@ -191,4 +192,191 @@ describe("CustomerNotesSection", () => {
|
|||||||
render(<CustomerNotesSection appointment={{ ...UPCOMING_APPT, status: "cancelled" }} sessionId="test-session-id" />);
|
render(<CustomerNotesSection appointment={{ ...UPCOMING_APPT, status: "cancelled" }} sessionId="test-session-id" />);
|
||||||
expect(screen.queryByRole("button", { name: /Save Notes/i })).not.toBeInTheDocument();
|
expect(screen.queryByRole("button", { name: /Save Notes/i })).not.toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("ConfirmationSection", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.clearAllMocks();
|
||||||
|
global.fetch = vi.fn();
|
||||||
|
vi.stubGlobal("confirm", vi.fn(() => true));
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
vi.restoreAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders pending badge when confirmationStatus is pending", () => {
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
expect(screen.getByText("Pending confirmation")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders confirmed badge when confirmationStatus is confirmed", () => {
|
||||||
|
render(<ConfirmationSection appointment={{ ...UPCOMING_APPT, confirmationStatus: "confirmed" }} sessionId="test-session-id" />);
|
||||||
|
expect(screen.getByText("✓ Confirmed")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders cancelled badge when confirmationStatus is cancelled", () => {
|
||||||
|
render(<ConfirmationSection appointment={{ ...UPCOMING_APPT, confirmationStatus: "cancelled" }} sessionId="test-session-id" />);
|
||||||
|
expect(screen.getByText("Cancelled")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("shows Confirm Appointment button when status is pending", () => {
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
expect(screen.getByRole("button", { name: /Confirm Appointment/i })).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does not show Confirm button when already confirmed", () => {
|
||||||
|
render(<ConfirmationSection appointment={{ ...UPCOMING_APPT, confirmationStatus: "confirmed" }} sessionId="test-session-id" />);
|
||||||
|
expect(screen.queryByRole("button", { name: /Confirm Appointment/i })).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does not show Confirm button when cancelled", () => {
|
||||||
|
render(<ConfirmationSection appointment={{ ...UPCOMING_APPT, confirmationStatus: "cancelled" }} sessionId="test-session-id" />);
|
||||||
|
expect(screen.queryByRole("button", { name: /Confirm Appointment/i })).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("calls confirm API and updates local status on success", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({ id: "appt-1", confirmationStatus: "confirmed" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(global.fetch).toHaveBeenCalledWith(
|
||||||
|
"/api/portal/appointments/appt-1/confirm",
|
||||||
|
expect.objectContaining({ method: "POST" })
|
||||||
|
);
|
||||||
|
});
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(screen.getByText("✓ Confirmed")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("sends X-Impersonation-Session-Id header when session exists", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({ id: "appt-1", confirmationStatus: "confirmed" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(global.fetch).toHaveBeenCalledWith(
|
||||||
|
"/api/portal/appointments/appt-1/confirm",
|
||||||
|
expect.objectContaining({
|
||||||
|
headers: expect.objectContaining({
|
||||||
|
"X-Impersonation-Session-Id": "test-session-id",
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does not send X-Impersonation-Session-Id header when sessionId is null", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({ id: "appt-1", confirmationStatus: "confirmed" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId={null} />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(global.fetch).toHaveBeenCalledWith(
|
||||||
|
"/api/portal/appointments/appt-1/confirm",
|
||||||
|
expect.objectContaining({
|
||||||
|
headers: expect.not.objectContaining({
|
||||||
|
"X-Impersonation-Session-Id": expect.anything(),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("shows error message when confirm API returns 401", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: false,
|
||||||
|
status: 401,
|
||||||
|
json: async () => ({ error: "Unauthorized" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(screen.getByText(/Unauthorized/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("shows error message when confirm API returns 403", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: false,
|
||||||
|
status: 403,
|
||||||
|
json: async () => ({ error: "Forbidden" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(screen.getByText(/Forbidden/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("shows error message when confirm API returns 422 (invalid state)", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: false,
|
||||||
|
status: 422,
|
||||||
|
json: async () => ({ error: "Cannot confirm - appointment is not in pending state" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(screen.getByText(/Cannot confirm/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does not call confirm API if user cancels the confirmation dialog", async () => {
|
||||||
|
vi.stubGlobal("confirm", vi.fn(() => false));
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
expect(global.fetch).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("shows loading state while confirming", async () => {
|
||||||
|
vi.mocked(global.fetch).mockReturnValue(new Promise(() => {})); // Never resolves
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
// Get button reference before clicking
|
||||||
|
const btn = screen.getByRole("button", { name: /Confirm Appointment/i });
|
||||||
|
fireEvent.click(btn);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(screen.getByText(/Confirming.../i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
// Button is disabled while loading
|
||||||
|
expect(btn).toBeDisabled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("shows success message briefly after confirm", async () => {
|
||||||
|
vi.mocked(global.fetch).mockResolvedValue({
|
||||||
|
ok: true,
|
||||||
|
json: async () => ({ id: "appt-1", confirmationStatus: "confirmed" }),
|
||||||
|
} as Response);
|
||||||
|
|
||||||
|
render(<ConfirmationSection appointment={UPCOMING_APPT} sessionId="test-session-id" />);
|
||||||
|
fireEvent.click(screen.getByRole("button", { name: /Confirm Appointment/i }));
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(screen.getByText(/Confirmed!/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
@@ -41,6 +41,7 @@ export interface Appointment {
|
|||||||
duration: number;
|
duration: number;
|
||||||
price: number;
|
price: number;
|
||||||
status: "confirmed" | "pending" | "waitlisted" | "completed" | "cancelled";
|
status: "confirmed" | "pending" | "waitlisted" | "completed" | "cancelled";
|
||||||
|
confirmationStatus: "pending" | "confirmed" | "cancelled";
|
||||||
notes: string;
|
notes: string;
|
||||||
customerNotes: string;
|
customerNotes: string;
|
||||||
reportCardId?: string;
|
reportCardId?: string;
|
||||||
@@ -177,21 +178,21 @@ export const UPCOMING_APPOINTMENTS: Appointment[] = [
|
|||||||
id: "a1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "a1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Full Groom"], addOns: ["De-shedding Treatment"],
|
services: ["Full Groom"], addOns: ["De-shedding Treatment"],
|
||||||
date: "2026-03-21", time: "10:00 AM", duration: 120, price: 145,
|
date: "2026-03-21", time: "10:00 AM", duration: 120, price: 145,
|
||||||
status: "confirmed", notes: "Spring shed is heavy — extra undercoat work needed",
|
status: "confirmed", confirmationStatus: "confirmed", notes: "Spring shed is heavy — extra undercoat work needed",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "a2", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
id: "a2", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
||||||
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
||||||
date: "2026-03-25", time: "2:00 PM", duration: 100, price: 90,
|
date: "2026-03-25", time: "2:00 PM", duration: 100, price: 90,
|
||||||
status: "confirmed", notes: "First visit with Morgan — patient with anxious pets",
|
status: "confirmed", confirmationStatus: "confirmed", notes: "First visit with Morgan — patient with anxious pets",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "a3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "a3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Bath & Brush"], addOns: [],
|
services: ["Bath & Brush"], addOns: [],
|
||||||
date: "2026-04-18", time: "11:00 AM", duration: 45, price: 55,
|
date: "2026-04-18", time: "11:00 AM", duration: 45, price: 55,
|
||||||
status: "pending", notes: "",
|
status: "pending", confirmationStatus: "pending", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
@@ -201,56 +202,56 @@ export const PAST_APPOINTMENTS: Appointment[] = [
|
|||||||
id: "pa1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Full Groom"], addOns: ["De-shedding Treatment", "Blueberry Facial"],
|
services: ["Full Groom"], addOns: ["De-shedding Treatment", "Blueberry Facial"],
|
||||||
date: "2026-02-15", time: "10:00 AM", duration: 130, price: 160,
|
date: "2026-02-15", time: "10:00 AM", duration: 130, price: 160,
|
||||||
status: "completed", notes: "", reportCardId: "rc1",
|
status: "completed", confirmationStatus: "confirmed", notes: "", reportCardId: "rc1",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa2", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
id: "pa2", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
||||||
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
||||||
date: "2026-02-20", time: "1:00 PM", duration: 100, price: 88,
|
date: "2026-02-20", time: "1:00 PM", duration: 100, price: 88,
|
||||||
status: "completed", notes: "", reportCardId: "rc2",
|
status: "completed", confirmationStatus: "confirmed", notes: "", reportCardId: "rc2",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Bath & Brush"], addOns: [],
|
services: ["Bath & Brush"], addOns: [],
|
||||||
date: "2026-01-18", time: "9:00 AM", duration: 45, price: 55,
|
date: "2026-01-18", time: "9:00 AM", duration: 45, price: 55,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa4", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
id: "pa4", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
||||||
services: ["Puppy's First Groom"], addOns: [],
|
services: ["Puppy's First Groom"], addOns: [],
|
||||||
date: "2026-01-10", time: "3:00 PM", duration: 60, price: 62,
|
date: "2026-01-10", time: "3:00 PM", duration: 60, price: 62,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa5", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa5", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Full Groom"], addOns: ["Nail Grinding"],
|
services: ["Full Groom"], addOns: ["Nail Grinding"],
|
||||||
date: "2025-12-20", time: "10:00 AM", duration: 105, price: 132,
|
date: "2025-12-20", time: "10:00 AM", duration: 105, price: 132,
|
||||||
status: "completed", notes: "Holiday groom",
|
status: "completed", confirmationStatus: "confirmed", notes: "Holiday groom",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa6", petId: "p1", petName: "Biscuit", groomerId: "g2", groomerName: "Alex",
|
id: "pa6", petId: "p1", petName: "Biscuit", groomerId: "g2", groomerName: "Alex",
|
||||||
services: ["Full Groom"], addOns: [],
|
services: ["Full Groom"], addOns: [],
|
||||||
date: "2025-11-15", time: "11:00 AM", duration: 90, price: 110,
|
date: "2025-11-15", time: "11:00 AM", duration: 90, price: 110,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa7", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
id: "pa7", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
||||||
services: ["Bath & Brush"], addOns: [],
|
services: ["Bath & Brush"], addOns: [],
|
||||||
date: "2025-11-08", time: "2:00 PM", duration: 45, price: 48,
|
date: "2025-11-08", time: "2:00 PM", duration: 45, price: 48,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa8", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa8", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Bath & Brush"], addOns: ["De-shedding Treatment"],
|
services: ["Bath & Brush"], addOns: ["De-shedding Treatment"],
|
||||||
date: "2025-10-12", time: "10:00 AM", duration: 75, price: 85,
|
date: "2025-10-12", time: "10:00 AM", duration: 75, price: 85,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -41,6 +41,12 @@ const STATUS_COLORS: Record<string, string> = {
|
|||||||
cancelled: "bg-red-100 text-red-600",
|
cancelled: "bg-red-100 text-red-600",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const CONFIRMATION_STATUS_COLORS: Record<string, string> = {
|
||||||
|
confirmed: "bg-green-100 text-green-700",
|
||||||
|
pending: "bg-amber-100 text-amber-700",
|
||||||
|
cancelled: "bg-red-100 text-red-600",
|
||||||
|
};
|
||||||
|
|
||||||
export function AppointmentsSection({ readOnly, sessionId }: Props) {
|
export function AppointmentsSection({ readOnly, sessionId }: Props) {
|
||||||
const [showBooking, setShowBooking] = useState(false);
|
const [showBooking, setShowBooking] = useState(false);
|
||||||
const [expandedId, setExpandedId] = useState<string | null>(null);
|
const [expandedId, setExpandedId] = useState<string | null>(null);
|
||||||
@@ -165,14 +171,15 @@ function AppointmentCard({
|
|||||||
{isUpcoming(appt) && !readOnly && (
|
{isUpcoming(appt) && !readOnly && (
|
||||||
<CustomerNotesSection appointment={appt} sessionId={sessionId} />
|
<CustomerNotesSection appointment={appt} sessionId={sessionId} />
|
||||||
)}
|
)}
|
||||||
|
{isUpcoming(appt) && (
|
||||||
|
<ConfirmationSection appointment={appt} sessionId={sessionId} />
|
||||||
|
)}
|
||||||
{appt.status !== "completed" && appt.status !== "cancelled" && !readOnly && (
|
{appt.status !== "completed" && appt.status !== "cancelled" && !readOnly && (
|
||||||
<div className="flex gap-2 mt-3">
|
<div className="flex gap-2 mt-3">
|
||||||
<button className="text-xs px-3 py-1.5 border border-stone-200 rounded-lg text-stone-600 hover:bg-stone-50">
|
<button className="text-xs px-3 py-1.5 border border-stone-200 rounded-lg text-stone-600 hover:bg-stone-50">
|
||||||
Reschedule
|
Reschedule
|
||||||
</button>
|
</button>
|
||||||
<button className="text-xs px-3 py-1.5 border border-red-200 rounded-lg text-red-600 hover:bg-red-50">
|
<CancelAppointmentButton appointment={appt} sessionId={sessionId} />
|
||||||
Cancel
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
{appt.reportCardId && (
|
{appt.reportCardId && (
|
||||||
@@ -188,6 +195,116 @@ function AppointmentCard({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function ConfirmationSection({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
||||||
|
const [confirming, setConfirming] = useState(false);
|
||||||
|
const [confirmError, setConfirmError] = useState<string | null>(null);
|
||||||
|
const [confirmSuccess, setConfirmSuccess] = useState(false);
|
||||||
|
// Local state mirrors confirmationStatus so the badge updates immediately after confirm
|
||||||
|
const [localStatus, setLocalStatus] = useState(appt.confirmationStatus);
|
||||||
|
|
||||||
|
async function handleConfirm() {
|
||||||
|
if (!window.confirm("Confirm this appointment?")) return;
|
||||||
|
setConfirming(true);
|
||||||
|
setConfirmError(null);
|
||||||
|
try {
|
||||||
|
const headers: Record<string, string> = {};
|
||||||
|
if (sessionId) {
|
||||||
|
headers["X-Impersonation-Session-Id"] = sessionId;
|
||||||
|
}
|
||||||
|
const res = await fetch(`/api/portal/appointments/${appt.id}/confirm`, {
|
||||||
|
method: "POST",
|
||||||
|
headers,
|
||||||
|
});
|
||||||
|
if (!res.ok) {
|
||||||
|
const err = await res.json().catch(() => ({ error: "Failed to confirm" }));
|
||||||
|
throw new Error(err.error || `HTTP ${res.status}`);
|
||||||
|
}
|
||||||
|
setLocalStatus("confirmed");
|
||||||
|
setConfirmSuccess(true);
|
||||||
|
setTimeout(() => setConfirmSuccess(false), 2000);
|
||||||
|
} catch (e) {
|
||||||
|
setConfirmError(e instanceof Error ? e.message : "Failed to confirm");
|
||||||
|
} finally {
|
||||||
|
setConfirming(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const currentStatus = localStatus ?? appt.confirmationStatus;
|
||||||
|
const statusLabel = currentStatus === "confirmed"
|
||||||
|
? "✓ Confirmed"
|
||||||
|
: currentStatus === "pending"
|
||||||
|
? "Pending confirmation"
|
||||||
|
: "Cancelled";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mt-3 p-3 bg-stone-50 rounded-lg">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className={`px-2 py-0.5 rounded-full text-xs font-medium ${CONFIRMATION_STATUS_COLORS[currentStatus] || ""}`}>
|
||||||
|
{statusLabel}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{!confirmSuccess && currentStatus === "pending" && (
|
||||||
|
<button
|
||||||
|
onClick={handleConfirm}
|
||||||
|
disabled={confirming}
|
||||||
|
className="flex items-center gap-1.5 text-xs px-3 py-1.5 bg-green-600 text-white rounded-lg font-medium hover:bg-green-700 disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{confirming && <Loader2 size={12} className="animate-spin" />}
|
||||||
|
{confirming ? "Confirming..." : "Confirm Appointment"}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
{confirmSuccess && (
|
||||||
|
<span className="text-xs text-green-600 font-medium">Confirmed!</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{confirmError && <p className="text-xs text-red-500 mt-1">{confirmError}</p>}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function CancelAppointmentButton({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
||||||
|
const [cancelling, setCancelling] = useState(false);
|
||||||
|
const [cancelError, setCancelError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
async function handleCancel() {
|
||||||
|
if (!window.confirm("Cancel this appointment? This cannot be undone.")) return;
|
||||||
|
setCancelling(true);
|
||||||
|
setCancelError(null);
|
||||||
|
try {
|
||||||
|
const headers: Record<string, string> = {};
|
||||||
|
if (sessionId) {
|
||||||
|
headers["X-Impersonation-Session-Id"] = sessionId;
|
||||||
|
}
|
||||||
|
const res = await fetch(`/api/portal/appointments/${appt.id}/cancel`, {
|
||||||
|
method: "POST",
|
||||||
|
headers,
|
||||||
|
});
|
||||||
|
if (!res.ok) {
|
||||||
|
const err = await res.json().catch(() => ({ error: "Failed to cancel" }));
|
||||||
|
throw new Error(err.error || `HTTP ${res.status}`);
|
||||||
|
}
|
||||||
|
window.location.reload();
|
||||||
|
} catch (e) {
|
||||||
|
setCancelError(e instanceof Error ? e.message : "Failed to cancel");
|
||||||
|
setCancelling(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<button
|
||||||
|
onClick={handleCancel}
|
||||||
|
disabled={cancelling}
|
||||||
|
className="text-xs px-3 py-1.5 border border-red-200 rounded-lg text-red-600 hover:bg-red-50 disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{cancelling ? "Cancelling..." : "Cancel"}
|
||||||
|
</button>
|
||||||
|
{cancelError && <p className="text-xs text-red-500 mt-1">{cancelError}</p>}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
export function CustomerNotesSection({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
export function CustomerNotesSection({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
||||||
const [notes, setNotes] = useState(appt.customerNotes || "");
|
const [notes, setNotes] = useState(appt.customerNotes || "");
|
||||||
const [saving, setSaving] = useState(false);
|
const [saving, setSaving] = useState(false);
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import postgres from "postgres";
|
|||||||
import * as schema from "./schema.js";
|
import * as schema from "./schema.js";
|
||||||
|
|
||||||
export * from "./schema.js";
|
export * from "./schema.js";
|
||||||
export { and, asc, desc, eq, gte, gt, ilike, lt, lte, ne, or, sql } from "drizzle-orm";
|
export { and, asc, desc, eq, exists, gte, gt, ilike, lt, lte, ne, or, sql } from "drizzle-orm";
|
||||||
|
|
||||||
let _db: ReturnType<typeof drizzle> | null = null;
|
let _db: ReturnType<typeof drizzle> | null = null;
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user