Promote dev → uat: GRO-2513 Settings role-gate #83
Reference in New Issue
Block a user
Delete Branch "dev"
Deleting a branch is permanent. Although the deleted branch may continue to exist for a short time before it actually gets removed, it CANNOT be undone in most cases. Continue?
Promote dev → uat: GRO-2513 Settings role-gate
What's in this promotion:
Changes merged to dev:
src/App.tsx— Settings tab in admin nav now hidden for groomer/receptionist;/admin/settingsroute guard redirects non-managers to/adminsrc/pages/Settings.tsx—GET /api/admin/settingsonly fires for manager/super-user roles; groomers never trigger the 403UAT_PLAYBOOK.md§5.14 — 8 new role-path test cases (TC-WEB-5.14.1–8)PR: groombook/web#82 (feature→dev, squash merged, CI ✓)
The post-auth handler in CustomerPortal previously rendered the "Portal access not configured" card when the SSO bridge returned 404 (no client row for the user's email). That trapped first-time SSO users on a dead-end screen with no path to portal creation. This change routes the 404 to a new OOBE component (src/portal/OOBE.tsx) that drives portal creation: * Mounts inline inside CustomerPortal so the post-auth flow stays inside the portal render tree (no App-level router needed). * Also reachable as a direct deep-link via the new /onboarding route in App.tsx (for grooming admins or recovery flows). * Submits to a new POST /api/portal/clients-from-auth endpoint in groombook-api (companion commit) that creates a fresh client row bound to the Better Auth email. 409 means the email already has a portal record — the OOBE shows a portal-selection message. * Uses the canonical shared signOut() from lib/auth-client (GRO-2358 invariant) for the Sign out button. * Full window.location.href reload on submit success to reset the bridge's cached state and land the user in their portal. The no-access card itself is preserved for the deep-link deleted-portal case (a customer whose portal was disabled/deleted), signalled via ?noAccess=deleted-portal on a portal sub-route. The OOBE handles the first-time-creation case; the no-access card handles the "had a portal but lost it" case. Test coverage: * "routes to /onboarding when session-from-auth returns 404 (GRO-2359)" — proves the post-auth 404 mounts the OOBE inline, not the legacy no-access card. * 6 new OOBE tests: render from direct link, name prefill, form submission, 409 portal-selection, required-name validation, shared signOut(), redirect on no-session. * P1 no-access tests reworked to use ?noAccess=deleted-portal so the GRO-2358 signOut invariant is still verified on the only surviving path to the no-access card. UAT_PLAYBOOK §5.25.5–6e rewritten to cover the OOBE flow (form submit, 409, deep-link mount, deleted-portal no-access card). Paired with the api PR on feature/2357-p2-portal-clients-from-auth. Co-Authored-By: Paperclip <noreply@paperclip.ing>QA code review APPROVED ✅ (dev → uat, Phase 2)
GRO-2513 (scoped review) — verified correct against spec:
src/App.tsx: Settings nav tab filtered viacanSettings = staffUser !== null && (role === "manager" || isSuperUser);/admin/settingsroute rendersnullwhilestaffUser === null(fail-closed during load), then<SettingsPage />for managers/super-users or<Navigate to="/admin" replace />for everyone else. Matches spec.src/pages/Settings.tsx: effect now fetches/api/staff/mefirst, gatesGET /api/admin/settingsonmanager || isSuperUserandGET /api/admin/auth-provideronisSuperUser. Groomers/receptionists fire neither admin fetch → no 403. All.catch()paths clearloaded/authLoaded, so no infinite-load on error.UAT_PLAYBOOK.md §5.14(TC-WEB-5.14.1–8): present and covers manager/super-user/groomer/receptionist paths incl. the "zero 403" assertion.Promotion-wide checks:
2ce7966: Test, Lint & Typecheck, and Docker Build all ✅ (push + pull_request). PR mergeable.Engineer is clear to self-merge.