7c0d422228
Ground-up TypeScript rewrite of the Vector hardware parts inventory
system. Ships the full roadmap (Phases 0-8) in one initial commit:
- pnpm + Turbo monorepo: apps/{api,web,e2e}, packages/{db,shared,ui,config}
- Express 5 + Prisma 5 + zod validation + JWT w/ refresh-token rotation
- React 19 + Vite + shadcn/ui + TanStack Query/Table + nuqs URL state
- Repair/RMA, tags, bulk ops, saved views, CSV audit export
- Analytics dashboard on Recharts + EOL tracking
- Signed webhook subscriptions (HMAC-SHA256) with in-process emitter
- Vitest unit tests (shared schemas, api services/helpers) + Playwright skeleton
- Gitea Actions CI (lint, typecheck, test+coverage, build) + Renovate
Deferred follow-ups: Postgres cutover (data-migration script ready),
BullMQ worker for webhook delivery, @react-pdf PDF export, CSV import wizard.
31 lines
855 B
TypeScript
31 lines
855 B
TypeScript
import { Navigate, useLocation } from 'react-router-dom';
|
|
import type { ReactNode } from 'react';
|
|
import { Skeleton } from '@vector/ui';
|
|
import { useAuth } from '../../contexts/AuthContext.js';
|
|
import type { Role } from '@vector/shared';
|
|
|
|
interface RequireAuthProps {
|
|
children: ReactNode;
|
|
role?: Role;
|
|
}
|
|
|
|
export function RequireAuth({ children, role }: RequireAuthProps) {
|
|
const { user, status } = useAuth();
|
|
const location = useLocation();
|
|
|
|
if (status === 'loading') {
|
|
return (
|
|
<div className="flex min-h-screen items-center justify-center">
|
|
<Skeleton className="h-20 w-72" />
|
|
</div>
|
|
);
|
|
}
|
|
if (status === 'anonymous' || !user) {
|
|
return <Navigate to="/login" state={{ from: location }} replace />;
|
|
}
|
|
if (role && user.role !== role) {
|
|
return <Navigate to="/" replace />;
|
|
}
|
|
return <>{children}</>;
|
|
}
|