import { type LoaderFunctionArgs, redirect } from 'react-router'; import { Outlet, useLoaderData, useNavigation } from 'react-router'; import { ProgressBar } from 'react-aria-components'; import { ErrorPopup } from '~/components/Error'; import Header from '~/components/Header'; import Footer from '~/components/Footer'; import Link from '~/components/Link'; import { cn } from '~/utils/cn'; import { loadContext } from '~/utils/config/headplane'; import { HeadscaleError, pull } from '~/utils/headscale'; import { destroySession, getSession } from '~/utils/sessions.server'; export async function loader({ request }: LoaderFunctionArgs) { const session = await getSession(request.headers.get('Cookie')); if (!session.has('hsApiKey')) { return redirect('/login'); } try { await pull('v1/apikey', session.get('hsApiKey')!); } catch (error) { if (error instanceof HeadscaleError) { // Safest to just redirect to login if we can't pull return redirect('/login', { headers: { 'Set-Cookie': await destroySession(session), }, }); } // Otherwise propagate to boundary throw error; } const context = await loadContext(); return { config: context.config, url: context.headscalePublicUrl ?? context.headscaleUrl, debug: context.debug, user: session.get('user'), }; } export default function Layout() { const data = useLoaderData(); const nav = useNavigation(); return ( <>