import {type RouteObject, RouterProvider, useRouteError} from 'react-router-dom'; import {cache} from '@emotion/css'; // eslint-disable-line @emotion/no-vanilla import {CacheProvider, ThemeProvider} from '@emotion/react'; import {createMemoryHistory, createRouter} from '@remix-run/router'; import * as rtl from '@testing-library/react'; // eslint-disable-line no-restricted-imports import userEvent from '@testing-library/user-event'; // eslint-disable-line no-restricted-imports import * as qs from 'query-string'; import {makeTestQueryClient} from 'sentry-test/queryClient'; import {GlobalDrawer} from 'sentry/components/globalDrawer'; import GlobalModal from 'sentry/components/globalModal'; import type {InjectedRouter} from 'sentry/types/legacyReactRouter'; import type {Organization} from 'sentry/types/organization'; import {DANGEROUS_SET_TEST_HISTORY} from 'sentry/utils/browserHistory'; import {QueryClientProvider} from 'sentry/utils/queryClient'; import {lightTheme} from 'sentry/utils/theme'; import {OrganizationContext} from 'sentry/views/organizationContext'; import {TestRouteContext} from 'sentry/views/routeContext'; import {instrumentUserEvent} from '../instrumentedEnv/userEventIntegration'; import {initializeOrg} from './initializeOrg'; interface ProviderOptions { /** * Do not shim the router use{Routes,Router,Navigate,Location} functions, and * instead allow them to work as normal, rendering inside of a memory router. * * When enabling this passing a `router` object *will do nothing*! */ disableRouterMocks?: boolean; /** * Sets the OrganizationContext. You may pass null to provide no organization */ organization?: Partial | null; /** * Sets the RouterContext. */ router?: Partial; } interface Options extends ProviderOptions, rtl.RenderOptions {} function makeAllTheProviders(options: ProviderOptions) { const {organization, router} = initializeOrg({ organization: options.organization === null ? undefined : options.organization, router: options.router, }); // In some cases we may want to not provide an organization at all const optionalOrganization = options.organization === null ? null : organization; return function ({children}: {children?: React.ReactNode}) { const content = ( {children} ); const wrappedContent = options.disableRouterMocks ? ( content ) : ( {content} ); const history = createMemoryHistory(); // Inject legacy react-router 3 style router mocked navigation functions // into the memory history used in react router 6 // // TODO(epurkhiser): In a world without react-router 3 we should figure out // how to write our tests in a simpler way without all these shims if (!options.disableRouterMocks) { Object.defineProperty(history, 'location', {get: () => router.location}); history.replace = router.replace; history.push = (path: any) => { if (typeof path === 'object' && path.search) { path.query = qs.parse(path.search); delete path.search; delete path.hash; delete path.state; delete path.key; } // XXX(epurkhiser): This is a hack for react-router 3 to 6. react-router // 6 will not convert objects into strings before pushing. We can detect // this by looking for an empty hash, which we normally do not set for // our browserHistory.push calls if (typeof path === 'object' && path.hash === '') { const queryString = path.query ? qs.stringify(path.query) : null; path = `${path.pathname}${queryString ? `?${queryString}` : ''}`; } router.push(path); }; } DANGEROUS_SET_TEST_HISTORY({ goBack: router.goBack, push: router.push, replace: router.replace, listen: jest.fn(() => {}), listenBefore: jest.fn(), getCurrentLocation: jest.fn(() => ({pathname: '', query: {}})), }); // By default react-router 6 catches exceptions and displays the stack // trace. For tests we want them to bubble out function ErrorBoundary(): React.ReactNode { throw useRouteError(); } const routes: RouteObject[] = [ { path: '*', element: wrappedContent, errorElement: , }, ]; const memoryRouter = createRouter({ future: {v7_prependBasename: true}, history, routes, }).initialize(); return ( ); }; } /** * Try avoiding unnecessary context and just mount your component. If it works, * then you dont need anything else. * * render(); * * If your component requires additional context you can pass it in the * options. */ function render( ui: React.ReactElement, {router, organization, disableRouterMocks, ...rtlOptions}: Options = {} ) { const AllTheProviders = makeAllTheProviders({ organization, router, disableRouterMocks, }); return rtl.render(ui, {wrapper: AllTheProviders, ...rtlOptions}); } /** * @deprecated * Use userEvent over fireEvent where possible. * More details: https://kentcdodds.com/blog/common-mistakes-with-react-testing-library#not-using-testing-libraryuser-event */ const fireEvent = rtl.fireEvent; function renderGlobalModal(options?: Options) { const result = render(, options); /** * Helper that waits for the modal to be removed from the DOM. You may need to * wait for the modal to be removed to avoid any act warnings. */ function waitForModalToHide() { return rtl.waitFor(() => { expect(rtl.screen.queryByRole('dialog')).not.toBeInTheDocument(); }); } return {...result, waitForModalToHide}; } /** * Helper that waits for the drawer to be hidden from the DOM. You may need to * wait for the drawer to be removed to avoid any act warnings. */ function waitForDrawerToHide(ariaLabel: string) { return rtl.waitFor(() => { expect( rtl.screen.queryByRole('complementary', {name: ariaLabel}) ).not.toBeInTheDocument(); }); } /** * This cannot be implemented as a Sentry Integration because Jest creates an * isolated environment for each test suite. This means that if we were to apply * the monkey patching ahead of time, it would be shadowed by Jest. */ instrumentUserEvent(); // eslint-disable-next-line no-restricted-imports, import/export export * from '@testing-library/react'; export { // eslint-disable-next-line import/export render, renderGlobalModal, userEvent, // eslint-disable-next-line import/export fireEvent, waitForDrawerToHide, makeAllTheProviders, };