fix: handle headscale unavailability gracefully
This commit is contained in:
parent
e01fd8d50c
commit
1b45b0917f
@ -8,7 +8,6 @@ interface Props {
|
|||||||
type?: 'full' | 'embedded';
|
type?: 'full' | 'embedded';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function getMessage(error: Error | unknown) {
|
function getMessage(error: Error | unknown) {
|
||||||
if (!(error instanceof Error)) {
|
if (!(error instanceof Error)) {
|
||||||
return "An unknown error occurred";
|
return "An unknown error occurred";
|
||||||
@ -18,9 +17,9 @@ function getMessage(error: Error | unknown) {
|
|||||||
|
|
||||||
// Traverse the error chain to find the root cause
|
// Traverse the error chain to find the root cause
|
||||||
if (error.cause) {
|
if (error.cause) {
|
||||||
rootError = error.cause;
|
rootError = error.cause as Error;
|
||||||
while (rootError.cause) {
|
while (rootError.cause) {
|
||||||
rootError = rootError.cause;
|
rootError = rootError.cause as Error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -15,7 +15,7 @@ export default function Footer({ url, debug }: FooterProps) {
|
|||||||
return (
|
return (
|
||||||
<footer
|
<footer
|
||||||
className={cn(
|
className={cn(
|
||||||
'fixed bottom-0 left-0 z-50 w-full h-14',
|
'fixed bottom-0 left-0 z-40 w-full h-14',
|
||||||
'bg-ui-100 dark:bg-ui-900 text-ui-500',
|
'bg-ui-100 dark:bg-ui-900 text-ui-500',
|
||||||
'flex flex-col justify-center gap-1',
|
'flex flex-col justify-center gap-1',
|
||||||
'border-t border-ui-200 dark:border-ui-800',
|
'border-t border-ui-200 dark:border-ui-800',
|
||||||
|
|||||||
@ -17,10 +17,8 @@ import Menu from './Menu';
|
|||||||
import TabLink from './TabLink';
|
import TabLink from './TabLink';
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
data?: {
|
|
||||||
config: HeadplaneContext['config'];
|
config: HeadplaneContext['config'];
|
||||||
user?: SessionData['user'];
|
user?: SessionData['user'];
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface LinkProps {
|
interface LinkProps {
|
||||||
@ -45,7 +43,7 @@ function Link({ href, text, isMenu }: LinkProps) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Header({ data }: Props) {
|
export default function Header(data: Props) {
|
||||||
return (
|
return (
|
||||||
<header className="bg-main-700 dark:bg-main-800 text-ui-50">
|
<header className="bg-main-700 dark:bg-main-800 text-ui-50">
|
||||||
<div className="container flex items-center justify-between py-4">
|
<div className="container flex items-center justify-between py-4">
|
||||||
@ -57,7 +55,7 @@ export default function Header({ data }: Props) {
|
|||||||
<Link href="https://tailscale.com/download" text="Download" />
|
<Link href="https://tailscale.com/download" text="Download" />
|
||||||
<Link href="https://github.com/tale/headplane" text="GitHub" />
|
<Link href="https://github.com/tale/headplane" text="GitHub" />
|
||||||
<Link href="https://github.com/juanfont/headscale" text="Headscale" />
|
<Link href="https://github.com/juanfont/headscale" text="Headscale" />
|
||||||
{data?.user ? (
|
{data.user ? (
|
||||||
<Menu>
|
<Menu>
|
||||||
<Menu.Button
|
<Menu.Button
|
||||||
className={cn(
|
className={cn(
|
||||||
@ -122,7 +120,7 @@ export default function Header({ data }: Props) {
|
|||||||
name="Access Control"
|
name="Access Control"
|
||||||
icon={<LockIcon className="w-4 h-4" />}
|
icon={<LockIcon className="w-4 h-4" />}
|
||||||
/>
|
/>
|
||||||
{data?.config.read ? (
|
{data.config.read ? (
|
||||||
<>
|
<>
|
||||||
<TabLink
|
<TabLink
|
||||||
to="/dns"
|
to="/dns"
|
||||||
|
|||||||
@ -1,77 +1,77 @@
|
|||||||
import { type LoaderFunctionArgs, redirect } from 'react-router';
|
import { type LoaderFunctionArgs, redirect } from 'react-router';
|
||||||
import { Outlet, useLoaderData, useNavigation } from 'react-router';
|
import { Outlet, useLoaderData } from 'react-router';
|
||||||
import { ProgressBar } from 'react-aria-components';
|
import { useEffect } from 'react'
|
||||||
|
|
||||||
import { ErrorPopup } from '~/components/Error';
|
import { ErrorPopup } from '~/components/Error';
|
||||||
import Header from '~/components/Header';
|
import Header from '~/components/Header';
|
||||||
|
import { toast } from '~/components/Toaster';
|
||||||
import Footer from '~/components/Footer';
|
import Footer from '~/components/Footer';
|
||||||
import Link from '~/components/Link';
|
import Link from '~/components/Link';
|
||||||
|
import { useLiveData } from '~/utils/useLiveData'
|
||||||
import { cn } from '~/utils/cn';
|
import { cn } from '~/utils/cn';
|
||||||
import { loadContext } from '~/utils/config/headplane';
|
import { loadContext } from '~/utils/config/headplane';
|
||||||
import { HeadscaleError, pull } from '~/utils/headscale';
|
import { HeadscaleError, pull, healthcheck } from '~/utils/headscale';
|
||||||
import { destroySession, getSession } from '~/utils/sessions.server';
|
import { destroySession, getSession } from '~/utils/sessions.server';
|
||||||
|
import { XCircleFillIcon } from '@primer/octicons-react';
|
||||||
|
import log from '~/utils/log';
|
||||||
|
|
||||||
export async function loader({ request }: LoaderFunctionArgs) {
|
export async function loader({ request }: LoaderFunctionArgs) {
|
||||||
const session = await getSession(request.headers.get('Cookie'));
|
let healthy = false;
|
||||||
if (!session.has('hsApiKey')) {
|
try {
|
||||||
return redirect('/login');
|
healthy = await healthcheck();
|
||||||
|
} catch (error) {
|
||||||
|
log.debug('APIC', 'Healthcheck failed %o', error);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We shouldn't session invalidate if Headscale is down
|
||||||
|
if (healthy) {
|
||||||
|
// We can assert because shell ensures this is set
|
||||||
|
const session = await getSession(request.headers.get('Cookie'));
|
||||||
|
const apiKey = session.get('hsApiKey')!;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await pull('v1/apikey', session.get('hsApiKey')!);
|
await pull('v1/apikey', apiKey);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error instanceof HeadscaleError) {
|
if (error instanceof HeadscaleError) {
|
||||||
// Safest to just redirect to login if we can't pull
|
log.debug('APIC', 'API Key validation failed %o', error);
|
||||||
return redirect('/login', {
|
return redirect('/login', {
|
||||||
headers: {
|
headers: {
|
||||||
'Set-Cookie': await destroySession(session),
|
'Set-Cookie': await destroySession(session),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// Otherwise propagate to boundary
|
|
||||||
throw error;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const context = await loadContext();
|
|
||||||
return {
|
return {
|
||||||
config: context.config,
|
healthy,
|
||||||
url: context.headscalePublicUrl ?? context.headscaleUrl,
|
}
|
||||||
debug: context.debug,
|
|
||||||
user: session.get('user'),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Layout() {
|
export default function Layout() {
|
||||||
const data = useLoaderData<typeof loader>();
|
useLiveData({ interval: 3000 });
|
||||||
const nav = useNavigation();
|
const { healthy } = useLoaderData<typeof loader>()
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ProgressBar aria-label="Loading...">
|
{!healthy ? (
|
||||||
<div
|
<div className={cn(
|
||||||
className={cn(
|
'fixed bottom-0 right-0 z-50 w-fit h-14',
|
||||||
'fixed top-0 left-0 z-50 w-1/2 h-1',
|
'flex flex-col justify-center gap-1',
|
||||||
'bg-blue-500 dark:bg-blue-400 opacity-0',
|
)}>
|
||||||
nav.state === 'loading' && 'animate-loading opacity-100',
|
<div className={cn(
|
||||||
)}
|
'flex items-center gap-1.5 mr-1.5 py-2 px-1.5',
|
||||||
/>
|
'border rounded-lg text-white bg-red-500',
|
||||||
</ProgressBar>
|
'border-red-600 dark:border-red-400 shadow-sm',
|
||||||
<Header data={data} />
|
)}>
|
||||||
|
<XCircleFillIcon className="w-4 h-4 text-white" />
|
||||||
|
Headscale is unreachable
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : undefined}
|
||||||
<main className="container mx-auto overscroll-contain mt-4 mb-24">
|
<main className="container mx-auto overscroll-contain mt-4 mb-24">
|
||||||
<Outlet />
|
<Outlet />
|
||||||
</main>
|
</main>
|
||||||
<Footer {...data} />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function ErrorBoundary() {
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Header />
|
|
||||||
<ErrorPopup type="embedded" />
|
|
||||||
<Footer url="Unknown" debug={false} />
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
33
app/layouts/shell.tsx
Normal file
33
app/layouts/shell.tsx
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import Header from '~/components/Header';
|
||||||
|
import Footer from '~/components/Footer';
|
||||||
|
import { getSession } from '~/utils/sessions.server';
|
||||||
|
import { loadContext } from '~/utils/config/headplane';
|
||||||
|
import { useLoaderData, LoaderFunctionArgs, Outlet, redirect } from 'react-router';
|
||||||
|
|
||||||
|
// This loads the bare minimum for the application to function
|
||||||
|
// So we know that if context fails to load then well, oops?
|
||||||
|
export async function loader({ request }: LoaderFunctionArgs) {
|
||||||
|
const session = await getSession(request.headers.get('Cookie'));
|
||||||
|
if (!session.has('hsApiKey')) {
|
||||||
|
return redirect('/login');
|
||||||
|
}
|
||||||
|
|
||||||
|
const context = await loadContext();
|
||||||
|
return {
|
||||||
|
config: context.config,
|
||||||
|
url: context.headscalePublicUrl ?? context.headscaleUrl,
|
||||||
|
debug: context.debug,
|
||||||
|
user: session.get('user'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Shell() {
|
||||||
|
const data = useLoaderData<typeof loader>();
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Header {...data} />
|
||||||
|
<Outlet />
|
||||||
|
<Footer {...data} />
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
25
app/root.tsx
25
app/root.tsx
@ -1,9 +1,13 @@
|
|||||||
import type { LinksFunction, MetaFunction } from 'react-router';
|
import type { LoaderFunctionArgs, LinksFunction, MetaFunction } from 'react-router';
|
||||||
import { Links, Meta, Outlet, Scripts, ScrollRestoration } from 'react-router';
|
import { Links, Meta, Outlet, Scripts, ScrollRestoration, useNavigation } from 'react-router';
|
||||||
|
import { loadContext } from '~/utils/config/headplane';
|
||||||
|
|
||||||
|
import { ProgressBar } from 'react-aria-components';
|
||||||
import { ErrorPopup } from '~/components/Error';
|
import { ErrorPopup } from '~/components/Error';
|
||||||
|
// TODO: Make this a default export
|
||||||
import { Toaster } from '~/components/Toaster';
|
import { Toaster } from '~/components/Toaster';
|
||||||
import stylesheet from '~/tailwind.css?url';
|
import stylesheet from '~/tailwind.css?url';
|
||||||
|
import { cn } from '~/utils/cn';
|
||||||
|
|
||||||
export const meta: MetaFunction = () => [
|
export const meta: MetaFunction = () => [
|
||||||
{ title: 'Headplane' },
|
{ title: 'Headplane' },
|
||||||
@ -41,5 +45,20 @@ export function ErrorBoundary() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function App() {
|
export default function App() {
|
||||||
return <Outlet />;
|
const nav = useNavigation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ProgressBar aria-label="Loading...">
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'fixed top-0 left-0 z-50 w-1/2 h-1',
|
||||||
|
'bg-blue-500 dark:bg-blue-400 opacity-0',
|
||||||
|
nav.state === 'loading' && 'animate-loading opacity-100',
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</ProgressBar>
|
||||||
|
<Outlet />
|
||||||
|
</>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -12,6 +12,8 @@ export default [
|
|||||||
route('/oidc/start', 'routes/auth/oidc-start.ts'),
|
route('/oidc/start', 'routes/auth/oidc-start.ts'),
|
||||||
|
|
||||||
// All the main logged-in dashboard routes
|
// All the main logged-in dashboard routes
|
||||||
|
// Double nested to separate error propagations
|
||||||
|
layout('layouts/shell.tsx', [
|
||||||
layout('layouts/dashboard.tsx', [
|
layout('layouts/dashboard.tsx', [
|
||||||
...prefix('/machines', [
|
...prefix('/machines', [
|
||||||
index('routes/machines/overview.tsx'),
|
index('routes/machines/overview.tsx'),
|
||||||
@ -25,6 +27,8 @@ export default [
|
|||||||
...prefix('/settings', [
|
...prefix('/settings', [
|
||||||
index('routes/settings/overview.tsx'),
|
index('routes/settings/overview.tsx'),
|
||||||
route('/auth-keys', 'routes/settings/auth-keys.tsx'),
|
route('/auth-keys', 'routes/settings/auth-keys.tsx'),
|
||||||
|
route('/local-agent', 'routes/settings/local-agent.tsx'),
|
||||||
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
];
|
];
|
||||||
|
|||||||
@ -13,15 +13,10 @@ import TextField from '~/components/TextField';
|
|||||||
import type { Key } from '~/types';
|
import type { Key } from '~/types';
|
||||||
import { loadContext } from '~/utils/config/headplane';
|
import { loadContext } from '~/utils/config/headplane';
|
||||||
import { pull } from '~/utils/headscale';
|
import { pull } from '~/utils/headscale';
|
||||||
import {
|
import { beginAuthFlow, getRedirectUri } from '~/utils/oidc';
|
||||||
startOidc,
|
|
||||||
beginAuthFlow,
|
|
||||||
getRedirectUri
|
|
||||||
} from '~/utils/oidc';
|
|
||||||
import { commitSession, getSession } from '~/utils/sessions.server';
|
import { commitSession, getSession } from '~/utils/sessions.server';
|
||||||
|
|
||||||
export async function loader({ request }: LoaderFunctionArgs) {
|
export async function loader({ request }: LoaderFunctionArgs) {
|
||||||
|
|
||||||
const session = await getSession(request.headers.get('Cookie'));
|
const session = await getSession(request.headers.get('Cookie'));
|
||||||
if (session.has('hsApiKey')) {
|
if (session.has('hsApiKey')) {
|
||||||
return redirect('/machines', {
|
return redirect('/machines', {
|
||||||
|
|||||||
@ -13,6 +13,7 @@ import { getSession } from '~/utils/sessions.server';
|
|||||||
import { useLiveData } from '~/utils/useLiveData';
|
import { useLiveData } from '~/utils/useLiveData';
|
||||||
import type { Machine, Route, User } from '~/types';
|
import type { Machine, Route, User } from '~/types';
|
||||||
import { queryAgent, initAgentSocket } from '~/utils/ws-agent';
|
import { queryAgent, initAgentSocket } from '~/utils/ws-agent';
|
||||||
|
import { ErrorPopup } from '~/components/Error'
|
||||||
|
|
||||||
import { menuAction } from './action';
|
import { menuAction } from './action';
|
||||||
import MachineRow from './components/machine';
|
import MachineRow from './components/machine';
|
||||||
@ -139,3 +140,9 @@ export default function Page() {
|
|||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function ErrorBoundary() {
|
||||||
|
return (
|
||||||
|
<ErrorPopup type="embedded" />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user