import {LocationFixture} from 'sentry-fixture/locationFixture';
import {OrganizationFixture} from 'sentry-fixture/organization';
import {ProjectFixture} from 'sentry-fixture/project';
import {initializeOrg} from 'sentry-test/initializeOrg';
import {
render,
renderGlobalModal,
screen,
userEvent,
waitFor,
} from 'sentry-test/reactTestingLibrary';
import * as pageFilterUtils from 'sentry/components/organizations/pageFilters/persistence';
import ProjectsStore from 'sentry/stores/projectsStore';
import {browserHistory} from 'sentry/utils/browserHistory';
import EventView from 'sentry/utils/discover/eventView';
import {DEFAULT_EVENT_VIEW} from './data';
import Homepage from './homepage';
describe('Discover > Homepage', () => {
const features = ['global-views', 'discover-query'];
let initialData, organization, mockHomepage, measurementsMetaMock;
beforeEach(() => {
organization = OrganizationFixture({
features,
});
initialData = initializeOrg({
organization,
router: {
location: LocationFixture(),
},
});
ProjectsStore.loadInitialData(initialData.projects);
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events/',
body: [],
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-meta/',
body: {
count: 2,
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-stats/',
body: {data: [[123, []]]},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/tags/',
body: [],
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/releases/stats/',
body: [],
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/dynamic-sampling/custom-rules/',
body: '',
});
mockHomepage = MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
body: {
id: '2',
name: 'homepage query',
projects: [],
version: 2,
expired: false,
dateCreated: '2021-04-08T17:53:25.195782Z',
dateUpdated: '2021-04-09T12:13:18.567264Z',
createdBy: {
id: '2',
},
environment: ['alpha'],
fields: ['environment'],
widths: ['-1'],
range: '24h',
orderby: '-environment',
display: 'previous',
query: 'event.type:error',
queryDataset: 'discover',
},
});
measurementsMetaMock = MockApiClient.addMockResponse({
url: '/organizations/org-slug/measurements-meta/',
method: 'GET',
body: {},
});
});
it('renders the Discover banner', async () => {
render(
,
{router: initialData.router, organization: initialData.organization}
);
await screen.findByText('Discover Trends');
screen.getByText('Get a Tour');
expect(screen.queryByText('Build a new query')).not.toBeInTheDocument();
});
it('fetches from the homepage URL and renders fields, async page filters, async and chart information', async () => {
render(
,
{router: initialData.router, organization: initialData.organization}
);
expect(mockHomepage).toHaveBeenCalled();
await screen.findByText('environment');
// Only the environment field
expect(screen.getAllByTestId('grid-head-cell').length).toEqual(1);
screen.getByText('Previous Period');
screen.getByText('event.type:error');
expect(screen.queryByText('Dataset')).not.toBeInTheDocument();
});
it('renders event view from URL params over homepage query', async () => {
initialData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
field: ['project'],
},
},
},
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
expect(mockHomepage).toHaveBeenCalled();
await screen.findByText('project');
// This is the field in the mocked response for the homepage
expect(screen.queryByText('environment')).not.toBeInTheDocument();
});
it('applies URL changes with the homepage pathname', async () => {
render(
,
{router: initialData.router, organization: initialData.organization}
);
renderGlobalModal();
await userEvent.click(await screen.findByText('Columns'));
await userEvent.click(screen.getByTestId('label'));
await userEvent.click(screen.getByText('event.type'));
await userEvent.click(screen.getByText('Apply'));
expect(browserHistory.push).toHaveBeenCalledWith(
expect.objectContaining({
pathname: '/organizations/org-slug/discover/homepage/',
query: expect.objectContaining({
field: ['event.type'],
}),
})
);
});
it('does not show an editable header or author information', async () => {
render(
,
{router: initialData.router, organization: initialData.organization}
);
await waitFor(() => {
expect(measurementsMetaMock).toHaveBeenCalled();
});
// 'Discover' is the header for the homepage
expect(screen.getByText('Discover')).toBeInTheDocument();
expect(screen.queryByText(/Created by:/)).not.toBeInTheDocument();
expect(screen.queryByText(/Last edited:/)).not.toBeInTheDocument();
});
it('shows the Remove Default button on initial load', async () => {
MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
body: {
id: '2',
name: 'homepage query',
projects: [],
version: 2,
expired: false,
dateCreated: '2021-04-08T17:53:25.195782Z',
dateUpdated: '2021-04-09T12:13:18.567264Z',
createdBy: {
id: '2',
},
environment: [],
fields: ['environment'],
widths: ['-1'],
range: '14d',
orderby: '-environment',
display: 'previous',
query: 'event.type:error',
topEvents: '5',
},
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
expect(await screen.findByText('Remove Default')).toBeInTheDocument();
expect(screen.queryByText('Set as Default')).not.toBeInTheDocument();
});
it('Disables the Set as Default button when no saved homepage', async () => {
initialData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
},
},
},
});
mockHomepage = MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
expect(mockHomepage).toHaveBeenCalled();
expect(screen.getByRole('button', {name: /set as default/i})).toBeDisabled();
await waitFor(() => {
expect(measurementsMetaMock).toHaveBeenCalled();
});
});
it('follows absolute date selection', async () => {
initialData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
},
},
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
await userEvent.click(await screen.findByText('24H'));
await userEvent.click(await screen.findByText('Absolute date'));
await userEvent.click(screen.getByText('Apply'));
expect(screen.queryByText('14D')).not.toBeInTheDocument();
});
it('renders changes to the discover query when no homepage', async () => {
initialData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
field: ['title'],
},
},
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
body: '',
});
const {rerender} = render(
,
{router: initialData.router, organization: initialData.organization}
);
renderGlobalModal();
// Simulate an update to the columns by changing the URL params
const rerenderData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
field: ['event.type'],
},
},
},
});
rerender(
);
await waitFor(() => {
expect(measurementsMetaMock).toHaveBeenCalled();
});
expect(screen.getByText('event.type')).toBeInTheDocument();
});
it('renders changes to the discover query when loaded with valid event view in url params', async () => {
initialData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
field: ['title'],
},
},
},
});
const {rerender} = render(
,
{router: initialData.router, organization: initialData.organization}
);
renderGlobalModal();
// Simulate an update to the columns by changing the URL params
const rerenderData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
field: ['event.type'],
},
},
},
});
rerender(
);
await waitFor(() => {
expect(measurementsMetaMock).toHaveBeenCalled();
});
expect(screen.getByText('event.type')).toBeInTheDocument();
});
it('overrides homepage filters with pinned filters if they exist', async () => {
ProjectsStore.loadInitialData([ProjectFixture({id: '1'}), ProjectFixture({id: '2'})]);
jest.spyOn(pageFilterUtils, 'getPageFilterStorage').mockReturnValueOnce({
pinnedFilters: new Set(['projects']),
state: {
project: [2],
environment: [],
start: null,
end: null,
period: '14d',
utc: null,
},
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
await waitFor(() => {
expect(measurementsMetaMock).toHaveBeenCalled();
});
expect(screen.getByText('project-slug')).toBeInTheDocument();
});
it('allows users to set the All Events query as default', async () => {
initialData = initializeOrg({
organization,
router: {
location: {
...LocationFixture(),
query: {
...EventView.fromSavedQuery(DEFAULT_EVENT_VIEW).generateQueryStringObject(),
},
},
},
});
mockHomepage = MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
await waitFor(() => expect(screen.getByTestId('set-as-default')).toBeEnabled());
});
it('uses split decision for homepage query', async () => {
organization = OrganizationFixture({
features: [
'discover-basic',
'discover-query',
'performance-discover-dataset-selector',
],
});
initialData = initializeOrg({
organization: organization,
router: {
location: LocationFixture(),
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events/',
body: {
meta: {
discoverSplitDecision: 'error-events',
},
data: [],
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
body: {
id: '2',
name: 'homepage query',
projects: [],
version: 2,
expired: false,
dateCreated: '2021-04-08T17:53:25.195782Z',
dateUpdated: '2021-04-09T12:13:18.567264Z',
createdBy: {
id: '2',
},
environment: [],
fields: ['environment'],
widths: ['-1'],
range: '14d',
orderby: '-environment',
display: 'previous',
query: 'event.type:error',
topEvents: '5',
queryDataset: 'discover',
},
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
expect(await screen.findByText('Remove Default')).toBeInTheDocument();
expect(screen.queryByText('Set as Default')).not.toBeInTheDocument();
await screen.findByText('environment');
expect(screen.getAllByTestId('grid-head-cell').length).toEqual(1);
screen.getByText('event.type:error');
expect(screen.getByRole('button', {name: 'Dataset Errors'})).toBeInTheDocument();
expect(
screen.getByText(
"We're splitting our datasets up to make it a bit easier to digest. We defaulted this query to Errors. Edit as you see fit."
)
).toBeInTheDocument();
});
it('saves homepage with dataset selection', async () => {
organization = OrganizationFixture({
features: [
'discover-basic',
'discover-query',
'performance-discover-dataset-selector',
],
});
initialData = initializeOrg({
organization: organization,
router: {
location: LocationFixture(),
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events/',
body: {
meta: {
discoverSplitDecision: 'error-events',
},
data: [],
},
});
MockApiClient.addMockResponse({
url: '/organizations/org-slug/discover/homepage/',
method: 'GET',
statusCode: 200,
body: {
id: '2',
name: 'homepage query',
projects: [],
version: 2,
expired: false,
dateCreated: '2021-04-08T17:53:25.195782Z',
dateUpdated: '2021-04-09T12:13:18.567264Z',
createdBy: {
id: '2',
},
environment: [],
fields: ['environment'],
widths: ['-1'],
range: '14d',
orderby: '-environment',
display: 'previous',
query: 'event.type:error',
topEvents: '5',
queryDataset: 'discover',
},
});
render(
,
{router: initialData.router, organization: initialData.organization}
);
expect(await screen.findByText('Remove Default')).toBeInTheDocument();
expect(screen.queryByText('Set as Default')).not.toBeInTheDocument();
await userEvent.click(screen.getByText('Dataset'));
await userEvent.click(screen.getByRole('option', {name: 'Transactions'}));
expect(initialData.router.push).toHaveBeenCalledWith(
expect.objectContaining({
query: expect.objectContaining({
dataset: 'transactions',
name: 'homepage query',
project: [],
query: 'event.type:error',
queryDataset: 'transaction-like',
}),
})
);
});
});