grafana/public/test/test-utils.tsx

104 lines
3.6 KiB
TypeScript
Raw Normal View History

import { Store } from '@reduxjs/toolkit';
import { render, RenderOptions } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { createMemoryHistory, MemoryHistoryBuildOptions } from 'history';
import React, { Fragment, PropsWithChildren } from 'react';
import { Provider } from 'react-redux';
import { Router } from 'react-router-dom';
import { getGrafanaContextMock } from 'test/mocks/getGrafanaContextMock';
import { HistoryWrapper, setLocationService } from '@grafana/runtime';
import { GrafanaContext, GrafanaContextType } from 'app/core/context/GrafanaContext';
import { ModalsContextProvider } from 'app/core/context/ModalsContextProvider';
import { configureStore } from 'app/store/configureStore';
import { StoreState } from 'app/types/store';
interface ExtendedRenderOptions extends RenderOptions {
/**
* Optional store to use for rendering. If not provided, a fresh store will be generated
* via `configureStore` method
*/
store?: Store<StoreState>;
/**
* Partial state to use for preloading store when rendering tests
*/
preloadedState?: Partial<StoreState>;
/**
* Should the wrapper be generated with a wrapping Router component?
* Useful if you're testing something that needs more nuanced routing behaviour
* and you want full control over it instead
*/
renderWithRouter?: boolean;
/**
* Props to pass to `createMemoryHistory`, if being used
*/
historyOptions?: MemoryHistoryBuildOptions;
}
/**
* Get a wrapper component that implements all of the providers that components
* within the app will need
*/
const getWrapper = ({
store,
renderWithRouter,
historyOptions,
grafanaContext,
}: ExtendedRenderOptions & {
grafanaContext?: GrafanaContextType;
}) => {
const reduxStore = store || configureStore();
/**
* Conditional router - either a MemoryRouter or just a Fragment
*/
const PotentialRouter = renderWithRouter ? Router : Fragment;
// Create a fresh location service for each test - otherwise we run the risk
// of it being stateful in between runs
const history = createMemoryHistory(historyOptions);
const locationService = new HistoryWrapper(history);
setLocationService(locationService);
const context = {
...getGrafanaContextMock(),
...grafanaContext,
};
/**
* Returns a wrapper that should (eventually?) match the main `AppWrapper`, so any tests are rendering
* in mostly the same providers as a "real" hierarchy
*/
return function Wrapper({ children }: PropsWithChildren) {
return (
<Provider store={reduxStore}>
<GrafanaContext.Provider value={context}>
<PotentialRouter history={history}>
<ModalsContextProvider>{children}</ModalsContextProvider>
</PotentialRouter>
</GrafanaContext.Provider>
</Provider>
);
};
};
/**
* Extended [@testing-library/react render](https://testing-library.com/docs/react-testing-library/api/#render)
* method which wraps the passed element in all of the necessary Providers,
* so it can render correctly in the context of the application
*/
const customRender = (
ui: React.ReactElement,
{ renderWithRouter = true, ...renderOptions }: ExtendedRenderOptions = {}
) => {
const store = renderOptions.preloadedState ? configureStore(renderOptions?.preloadedState) : undefined;
const AllTheProviders = renderOptions.wrapper || getWrapper({ store, renderWithRouter, ...renderOptions });
return {
...render(ui, { wrapper: AllTheProviders, ...renderOptions }),
store,
};
};
export * from '@testing-library/react';
export { customRender as render, getWrapper, userEvent };