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 modal from 'sentry/actionCreators/modal';
import * as LineChart from 'sentry/components/charts/lineChart';
import SimpleTableChart from 'sentry/components/charts/simpleTableChart';
import {MINUTE, SECOND} from 'sentry/utils/formatters';
import {MEPSettingProvider} from 'sentry/utils/performance/contexts/metricsEnhancedSetting';
import type {Widget} from 'sentry/views/dashboards/types';
import {DisplayType, WidgetType} from 'sentry/views/dashboards/types';
import WidgetCard from 'sentry/views/dashboards/widgetCard';
import ReleaseWidgetQueries from 'sentry/views/dashboards/widgetCard/releaseWidgetQueries';
jest.mock('sentry/components/charts/simpleTableChart', () => jest.fn(() =>
));
jest.mock('sentry/views/dashboards/widgetCard/releaseWidgetQueries');
describe('Dashboards > WidgetCard', function () {
const {router, organization, routerContext} = initializeOrg({
organization: OrganizationFixture({
features: ['dashboards-edit', 'discover-basic'],
projects: [ProjectFixture()],
}),
router: {orgId: 'orgId'},
} as Parameters[0]);
const renderWithProviders = (component: React.ReactNode) =>
render(
{component},
{organization, router, context: routerContext}
);
const multipleQueryWidget: Widget = {
title: 'Errors',
description: 'Valid widget description',
interval: '5m',
displayType: DisplayType.LINE,
widgetType: WidgetType.DISCOVER,
queries: [
{
conditions: 'event.type:error',
fields: ['count()', 'failure_count()'],
aggregates: ['count()', 'failure_count()'],
columns: [],
name: 'errors',
orderby: '',
},
{
conditions: 'event.type:default',
fields: ['count()', 'failure_count()'],
aggregates: ['count()', 'failure_count()'],
columns: [],
name: 'default',
orderby: '',
},
],
};
const selection = {
projects: [1],
environments: ['prod'],
datetime: {
period: '14d',
start: null,
end: null,
utc: false,
},
};
const api = new MockApiClient();
let eventsMock;
beforeEach(function () {
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-stats/',
body: {meta: {isMetricsData: false}},
});
eventsMock = MockApiClient.addMockResponse({
url: '/organizations/org-slug/events/',
body: {
meta: {fields: {title: 'string'}},
data: [{title: 'title'}],
},
});
});
afterEach(function () {
MockApiClient.clearMockResponses();
});
it('renders with Open in Discover button and opens the Query Selector Modal when clicked', async function () {
const spy = jest.spyOn(modal, 'openDashboardWidgetQuerySelectorModal');
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Open in Discover')).toBeInTheDocument();
await userEvent.click(screen.getByText('Open in Discover'));
expect(spy).toHaveBeenCalledWith({
isMetricsData: false,
organization,
widget: multipleQueryWidget,
});
});
it('renders with Open in Discover button and opens in Discover when clicked', async function () {
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Open in Discover')).toBeInTheDocument();
await userEvent.click(screen.getByText('Open in Discover'));
expect(router.push).toHaveBeenCalledWith(
'/organizations/org-slug/discover/results/?environment=prod&field=count%28%29&field=failure_count%28%29&name=Errors&project=1&query=event.type%3Aerror&statsPeriod=14d&yAxis=count%28%29&yAxis=failure_count%28%29'
);
});
it('renders widget description in dashboard', async function () {
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
expect(await screen.findByText('Valid widget description')).toBeInTheDocument();
});
it('Opens in Discover with prepended fields pulled from equations', async function () {
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Open in Discover')).toBeInTheDocument();
await userEvent.click(screen.getByText('Open in Discover'));
expect(router.push).toHaveBeenCalledWith(
'/organizations/org-slug/discover/results/?environment=prod&field=count_if%28transaction.duration%2Cequals%2C300%29&field=failure_count%28%29&field=count%28%29&field=equation%7C%28count%28%29%20%2B%20failure_count%28%29%29%20%2F%20count_if%28transaction.duration%2Cequals%2C300%29&name=Errors&project=1&query=event.type%3Aerror&statsPeriod=14d&yAxis=equation%7C%28count%28%29%20%2B%20failure_count%28%29%29%20%2F%20count_if%28transaction.duration%2Cequals%2C300%29'
);
});
it('Opens in Discover with Top N', async function () {
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Open in Discover')).toBeInTheDocument();
await userEvent.click(screen.getByText('Open in Discover'));
expect(router.push).toHaveBeenCalledWith(
'/organizations/org-slug/discover/results/?display=top5&environment=prod&field=transaction&field=count%28%29&name=Errors&project=1&query=event.type%3Aerror&statsPeriod=14d&yAxis=count%28%29'
);
});
it('allows Open in Discover when the widget contains custom measurements', async function () {
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Open in Discover')).toBeInTheDocument();
await userEvent.click(screen.getByText('Open in Discover'));
expect(router.push).toHaveBeenCalledWith(
'/organizations/org-slug/discover/results/?environment=prod&field=p99%28measurements.custom.measurement%29&name=Errors&project=1&query=&statsPeriod=14d&yAxis=p99%28measurements.custom.measurement%29'
);
});
it('calls onDuplicate when Duplicate Widget is clicked', async function () {
const mock = jest.fn();
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={mock}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Duplicate Widget')).toBeInTheDocument();
await userEvent.click(screen.getByText('Duplicate Widget'));
expect(mock).toHaveBeenCalledTimes(1);
});
it('does not add duplicate widgets if max widget is reached', async function () {
const mock = jest.fn();
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={mock}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Duplicate Widget')).toBeInTheDocument();
await userEvent.click(screen.getByText('Duplicate Widget'));
expect(mock).toHaveBeenCalledTimes(0);
});
it('calls onEdit when Edit Widget is clicked', async function () {
const mock = jest.fn();
renderWithProviders(
undefined}
onEdit={mock}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Edit Widget')).toBeInTheDocument();
await userEvent.click(screen.getByText('Edit Widget'));
expect(mock).toHaveBeenCalledTimes(1);
});
it('renders delete widget option', async function () {
const mock = jest.fn();
renderWithProviders(
undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await userEvent.click(await screen.findByLabelText('Widget actions'));
expect(screen.getByText('Delete Widget')).toBeInTheDocument();
await userEvent.click(screen.getByText('Delete Widget'));
// Confirm Modal
renderGlobalModal();
await screen.findByRole('dialog');
await userEvent.click(screen.getByTestId('confirm-button'));
expect(mock).toHaveBeenCalled();
});
it('calls events with a limit of 20 items', async function () {
const mock = jest.fn();
renderWithProviders(
undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
tableItemLimit={20}
/>
);
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
'/organizations/org-slug/events/',
expect.objectContaining({
query: expect.objectContaining({
per_page: 20,
}),
})
);
});
});
it('calls events with a default limit of 5 items', async function () {
const mock = jest.fn();
renderWithProviders(
undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
'/organizations/org-slug/events/',
expect.objectContaining({
query: expect.objectContaining({
per_page: 5,
}),
})
);
});
});
it('has sticky table headers', async function () {
const tableWidget: Widget = {
title: 'Table Widget',
interval: '5m',
displayType: DisplayType.TABLE,
widgetType: WidgetType.DISCOVER,
queries: [
{
conditions: '',
fields: ['transaction', 'count()'],
columns: ['transaction'],
aggregates: ['count()'],
name: 'Table',
orderby: '',
},
],
};
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
tableItemLimit={20}
/>
);
await waitFor(() => expect(eventsMock).toHaveBeenCalled());
await waitFor(() =>
expect(SimpleTableChart).toHaveBeenCalledWith(
expect.objectContaining({stickyHeaders: true}),
expect.anything()
)
);
});
it('calls release queries', function () {
const widget: Widget = {
title: 'Release Widget',
interval: '5m',
displayType: DisplayType.LINE,
widgetType: WidgetType.RELEASE,
queries: [],
};
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
tableItemLimit={20}
/>
);
expect(ReleaseWidgetQueries).toHaveBeenCalledTimes(1);
});
it('opens the widget viewer modal when a widget has no id', async () => {
const widget: Widget = {
title: 'Widget',
interval: '5m',
displayType: DisplayType.LINE,
widgetType: WidgetType.DISCOVER,
queries: [],
};
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
index="10"
isPreview
/>
);
await userEvent.click(await screen.findByLabelText('Open Widget Viewer'));
expect(router.push).toHaveBeenCalledWith(
expect.objectContaining({pathname: '/mock-pathname/widget/10/'})
);
});
it('renders stored data disclaimer', async function () {
MockApiClient.addMockResponse({
url: '/organizations/org-slug/events/',
body: {
meta: {title: 'string', isMetricsData: false},
data: [{title: 'title'}],
},
});
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
showStoredAlert
/>
);
// Badge in the widget header
expect(await screen.findByText('Indexed')).toBeInTheDocument();
expect(
// Alert below the widget
await screen.findByText(/we've automatically adjusted your results/i)
).toBeInTheDocument();
});
it('renders chart using axis and tooltip formatters from custom measurement meta', async function () {
const spy = jest.spyOn(LineChart, 'LineChart');
const eventsStatsMock = MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-stats/',
body: {
data: [
[
1658262600,
[
{
count: 24,
},
],
],
],
meta: {
fields: {
time: 'date',
p95_measurements_custom: 'duration',
},
units: {
time: null,
p95_measurements_custom: 'millisecond',
},
isMetricsData: true,
tips: {},
},
},
});
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await waitFor(function () {
expect(eventsStatsMock).toHaveBeenCalled();
});
const {tooltip, yAxis} = spy.mock.calls.pop()?.[0] ?? {};
expect(tooltip).toBeDefined();
expect(yAxis).toBeDefined();
// @ts-expect-error
expect(yAxis.axisLabel.formatter(24, 'p95(measurements.custom)')).toEqual('24ms');
});
it('renders label in seconds when there is a transition from seconds to minutes in the y axis', async function () {
const spy = jest.spyOn(LineChart, 'LineChart');
const eventsStatsMock = MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-stats/',
body: {
data: [
[
1658262600,
[
{
count: 40 * SECOND,
},
],
],
[
1658262601,
[
{
count: 50 * SECOND,
},
],
],
[
1658262602,
[
{
count: MINUTE,
},
],
],
[
1658262603,
[
{
count: 1.3 * MINUTE,
},
],
],
],
meta: {
fields: {
time: 'date',
p50_transaction_duration: 'duration',
},
units: {
time: null,
p50_transaction_duration: 'millisecond',
},
isMetricsData: false,
tips: {},
},
},
});
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
/>
);
await waitFor(function () {
expect(eventsStatsMock).toHaveBeenCalled();
});
const {yAxis} = spy.mock.calls.pop()?.[0] ?? {};
expect(yAxis).toBeDefined();
// @ts-expect-error
expect(yAxis.axisLabel.formatter(60000, 'p50(transaction.duration)')).toEqual('60s');
expect((yAxis as any).minInterval).toEqual(SECOND);
});
it('displays indexed badge in preview mode', async function () {
renderWithProviders(
undefined}
onEdit={() => undefined}
onDuplicate={() => undefined}
renderErrorMessage={() => undefined}
showContextMenu
widgetLimitReached={false}
isPreview
/>
);
expect(await screen.findByText('Indexed')).toBeInTheDocument();
});
});