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;