* custom error component
* upload sourcemaps to sentry * report caught errors to sentry
This commit is contained in:
63
app/core/components/error-component.tsx
Normal file
63
app/core/components/error-component.tsx
Normal file
@ -0,0 +1,63 @@
|
||||
import type { ErrorProps } from "next/error";
|
||||
import { ErrorComponent as DefaultErrorComponent } from "blitz";
|
||||
|
||||
import Sentry from "../../../integrations/sentry";
|
||||
|
||||
type ExtraProps = {
|
||||
hasGetInitialPropsRun?: boolean;
|
||||
err?: any;
|
||||
};
|
||||
|
||||
class ErrorComponent extends DefaultErrorComponent<ExtraProps> {
|
||||
render() {
|
||||
const { statusCode, hasGetInitialPropsRun, err } = this.props;
|
||||
if (!hasGetInitialPropsRun && err) {
|
||||
// getInitialProps is not called in case of
|
||||
// https://github.com/vercel/next.js/issues/8592. As a workaround, we pass
|
||||
// err via _app.js so it can be captured
|
||||
Sentry.captureException(err);
|
||||
}
|
||||
|
||||
return <DefaultErrorComponent statusCode={statusCode} />;
|
||||
}
|
||||
}
|
||||
|
||||
ErrorComponent.getInitialProps = async (ctx) => {
|
||||
const errorInitialProps: ErrorProps & ExtraProps = await DefaultErrorComponent.getInitialProps(ctx);
|
||||
|
||||
// Workaround for https://github.com/vercel/next.js/issues/8592, mark when
|
||||
// getInitialProps has run
|
||||
errorInitialProps.hasGetInitialPropsRun = true;
|
||||
|
||||
// Running on the server, the response object (`res`) is available.
|
||||
// Next.js will pass an err on the server if a page's data fetching methods
|
||||
// threw or returned a Promise that rejected
|
||||
//
|
||||
// Running on the client (browser), Next.js will provide an err if:
|
||||
// - a page's `getInitialProps` threw or returned a Promise that rejected
|
||||
// - an exception was thrown somewhere in the React lifecycle (render,
|
||||
// componentDidMount, etc) that was caught by Next.js's React Error
|
||||
// Boundary. Read more about what types of exceptions are caught by Error
|
||||
// Boundaries: https://reactjs.org/docs/error-boundaries.html
|
||||
|
||||
if (ctx.res?.statusCode === 404) {
|
||||
// Opinionated: do not record an exception in Sentry for 404
|
||||
return { statusCode: 404 };
|
||||
}
|
||||
|
||||
if (ctx.err) {
|
||||
Sentry.captureException(ctx.err);
|
||||
await Sentry.flush(2000);
|
||||
return errorInitialProps;
|
||||
}
|
||||
|
||||
// If this point is reached, getInitialProps was called without any
|
||||
// information about what the error might be. This is unexpected and may
|
||||
// indicate a bug introduced in Next.js, so record it in Sentry
|
||||
Sentry.captureException(new Error(`_error.js getInitialProps missing data at path: ${ctx.asPath}`));
|
||||
await Sentry.flush(2000);
|
||||
|
||||
return errorInitialProps;
|
||||
};
|
||||
|
||||
export default ErrorComponent;
|
@ -1,4 +1,6 @@
|
||||
import { Head, ErrorComponent } from "blitz";
|
||||
import { Head } from "blitz";
|
||||
|
||||
import ErrorComponent from "../core/components/error-component";
|
||||
|
||||
// ------------------------------------------------------
|
||||
// This page is rendered if a route match is not found
|
||||
|
@ -1,15 +1,17 @@
|
||||
import { Suspense } from "react";
|
||||
import { Suspense, useEffect } from "react";
|
||||
import {
|
||||
AppProps,
|
||||
ErrorBoundary,
|
||||
ErrorComponent,
|
||||
AuthenticationError,
|
||||
AuthorizationError,
|
||||
ErrorFallbackProps,
|
||||
useQueryErrorResetBoundary,
|
||||
getConfig,
|
||||
useSession,
|
||||
} from "blitz";
|
||||
|
||||
import Sentry from "../../integrations/sentry";
|
||||
import ErrorComponent from "../core/components/error-component";
|
||||
import LoginForm from "../auth/components/login-form";
|
||||
import { usePanelbear } from "../core/hooks/use-panelbear";
|
||||
|
||||
@ -18,12 +20,27 @@ import "app/core/styles/index.css";
|
||||
const { publicRuntimeConfig } = getConfig();
|
||||
|
||||
export default function App({ Component, pageProps }: AppProps) {
|
||||
const session = useSession();
|
||||
usePanelbear(publicRuntimeConfig.panelBear.siteId);
|
||||
useEffect(() => {
|
||||
if (session.userId) {
|
||||
Sentry.setUser({
|
||||
id: session.userId.toString(),
|
||||
orgId: session.orgId,
|
||||
});
|
||||
}
|
||||
}, [session]);
|
||||
|
||||
const getLayout = Component.getLayout || ((page) => page);
|
||||
|
||||
return (
|
||||
<ErrorBoundary FallbackComponent={RootErrorFallback} onReset={useQueryErrorResetBoundary().reset}>
|
||||
<ErrorBoundary
|
||||
onError={(error, componentStack) =>
|
||||
Sentry.captureException(error, { contexts: { react: { componentStack } } })
|
||||
}
|
||||
FallbackComponent={RootErrorFallback}
|
||||
onReset={useQueryErrorResetBoundary().reset}
|
||||
>
|
||||
<Suspense fallback="Silence, ca pousse">{getLayout(<Component {...pageProps} />)}</Suspense>
|
||||
</ErrorBoundary>
|
||||
);
|
||||
|
1
app/pages/_error.tsx
Normal file
1
app/pages/_error.tsx
Normal file
@ -0,0 +1 @@
|
||||
export { default } from "../core/components/error-component";
|
Reference in New Issue
Block a user