import React from 'react'; import PropTypes from 'prop-types'; import { FormattedMessage } from 'react-intl'; import { source_url } from 'flavours/glitch/initial_state'; import { preferencesLink } from 'flavours/glitch/utils/backend_links'; import StackTrace from 'stacktrace-js'; import { Helmet } from 'react-helmet'; export default class ErrorBoundary extends React.PureComponent { static propTypes = { children: PropTypes.node, }; state = { hasError: false, errorMessage: undefined, stackTrace: undefined, mappedStackTrace: undefined, componentStack: undefined, } componentDidCatch(error, info) { this.setState({ hasError: true, errorMessage: error.toString(), stackTrace: error.stack, componentStack: info && info.componentStack, mappedStackTrace: undefined, }); StackTrace.fromError(error).then((stackframes) => { this.setState({ mappedStackTrace: stackframes.map((sf) => sf.toString()).join('\n'), }); }).catch(() => { this.setState({ mappedStackTrace: undefined, }); }); } handleReload(e) { e.preventDefault(); window.location.reload(); } render() { const { hasError, errorMessage, stackTrace, mappedStackTrace, componentStack } = this.state; if (!hasError) return this.props.children; const likelyBrowserAddonIssue = errorMessage && errorMessage.includes('NotFoundError'); let debugInfo = ''; if (stackTrace) { debugInfo += 'Stack trace\n-----------\n\n```\n' + errorMessage + '\n' + stackTrace.toString() + '\n```'; } if (mappedStackTrace) { debugInfo += 'Mapped stack trace\n-----------\n\n```\n' + errorMessage + '\n' + mappedStackTrace.toString() + '\n```'; } if (componentStack) { if (debugInfo) { debugInfo += '\n\n\n'; } debugInfo += 'React component stack\n---------------------\n\n```\n' + componentStack.toString() + '\n```'; } let issueTracker = source_url; if (source_url.match(/^https:\/\/github\.com\/[^/]+\/[^/]+\/?$/)) { issueTracker = source_url + '/issues'; } return ( <div tabIndex='-1'> <div className='error-boundary'> <h1><FormattedMessage id='web_app_crash.title' defaultMessage="We're sorry, but something went wrong with the Mastodon app." /></h1> <p> <FormattedMessage id='web_app_crash.content' defaultMessage='You could try any of the following:' /> </p> <ul> { likelyBrowserAddonIssue && ( <li> <FormattedMessage id='web_app_crash.disable_addons' defaultMessage='Disable browser add-ons or built-in translation tools' /> </li> ) } <li> <FormattedMessage id='web_app_crash.report_issue' defaultMessage='Report a bug in the {issuetracker}' values={{ issuetracker: <a href={issueTracker} rel='noopener noreferrer' target='_blank'><FormattedMessage id='web_app_crash.issue_tracker' defaultMessage='issue tracker' /></a> }} /> { debugInfo !== '' && ( <details> <summary><FormattedMessage id='web_app_crash.debug_info' defaultMessage='Debug information' /></summary> <textarea className='web_app_crash-stacktrace' value={debugInfo} rows='10' readOnly /> </details> )} </li> <li> <FormattedMessage id='web_app_crash.reload_page' defaultMessage='{reload} the current page' values={{ reload: <a href='#' onClick={this.handleReload}><FormattedMessage id='web_app_crash.reload' defaultMessage='Reload' /></a> }} /> </li> { preferencesLink !== undefined && ( <li> <FormattedMessage id='web_app_crash.change_your_settings' defaultMessage='Change your {settings}' values={{ settings: <a href={preferencesLink}><FormattedMessage id='web_app_crash.settings' defaultMessage='settings' /></a> }} /> </li> )} </ul> </div> <Helmet> <meta name='robots' content='noindex' /> </Helmet> </div> ); } }