123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- import {OrganizationFixture} from 'sentry-fixture/organization';
- import {initializeOrg} from 'sentry-test/initializeOrg';
- import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
- import MemberListStore from 'sentry/stores/memberListStore';
- import type {Widget} from 'sentry/views/dashboards/types';
- import {DisplayType, WidgetType} from 'sentry/views/dashboards/types';
- import WidgetCard from 'sentry/views/dashboards/widgetCard';
- import {IssueSortOptions} from 'sentry/views/issueList/utils';
- describe('Dashboards > IssueWidgetCard', function () {
- const {router, organization, routerContext} = initializeOrg({
- organization: OrganizationFixture({
- features: ['dashboards-edit'],
- }),
- router: {orgId: 'orgId'},
- } as Parameters<typeof initializeOrg>[0]);
- const widget: Widget = {
- title: 'Issues',
- interval: '5m',
- displayType: DisplayType.TABLE,
- widgetType: WidgetType.ISSUE,
- queries: [
- {
- conditions: 'event.type:default',
- fields: ['issue', 'assignee', 'title'],
- columns: ['issue', 'assignee', 'title'],
- aggregates: [],
- name: '',
- orderby: IssueSortOptions.FREQ,
- },
- ],
- };
- const selection = {
- projects: [1],
- environments: ['prod'],
- datetime: {
- period: '14d',
- start: null,
- end: null,
- utc: false,
- },
- };
- const api = new MockApiClient();
- beforeEach(function () {
- MockApiClient.addMockResponse({
- url: '/organizations/org-slug/issues/',
- body: [
- {
- id: '44444444',
- title: 'ChunkLoadError: Loading chunk app_bootstrap_index_tsx failed.',
- shortId: 'ISSUE',
- assignedTo: {
- type: 'user',
- id: '2222222',
- name: 'dashboard user',
- email: 'dashboarduser@sentry.io',
- },
- lifetime: {count: 10, userCount: 5},
- count: 6,
- userCount: 3,
- project: {id: 1},
- },
- ],
- });
- MockApiClient.addMockResponse({
- url: '/organizations/org-slug/users/',
- method: 'GET',
- body: [],
- });
- });
- afterEach(function () {
- MockApiClient.clearMockResponses();
- });
- it('renders with title and issues chart', async function () {
- MemberListStore.loadInitialData([]);
- render(
- <WidgetCard
- api={api}
- organization={organization}
- widget={widget}
- selection={selection}
- isEditingDashboard={false}
- onDelete={() => undefined}
- onEdit={() => undefined}
- onDuplicate={() => undefined}
- renderErrorMessage={() => undefined}
- showContextMenu
- widgetLimitReached={false}
- />
- );
- expect(await screen.findByText('Issues')).toBeInTheDocument();
- expect(await screen.findByText('assignee')).toBeInTheDocument();
- expect(screen.getByText('title')).toBeInTheDocument();
- expect(screen.getByText('issue')).toBeInTheDocument();
- expect(screen.getByText('DU')).toBeInTheDocument();
- expect(screen.getByText('ISSUE')).toBeInTheDocument();
- expect(
- screen.getByText('ChunkLoadError: Loading chunk app_bootstrap_index_tsx failed.')
- ).toBeInTheDocument();
- await userEvent.hover(screen.getByTitle('dashboard user'));
- expect(await screen.findByText('Assigned to dashboard user')).toBeInTheDocument();
- });
- it('opens in issues page', async function () {
- render(
- <WidgetCard
- api={api}
- organization={organization}
- widget={widget}
- selection={selection}
- isEditingDashboard={false}
- onDelete={() => undefined}
- onEdit={() => undefined}
- onDuplicate={() => undefined}
- renderErrorMessage={() => undefined}
- showContextMenu
- widgetLimitReached={false}
- />,
- {context: routerContext}
- );
- await userEvent.click(await screen.findByLabelText('Widget actions'));
- expect(screen.getByText('Duplicate Widget')).toBeInTheDocument();
- expect(screen.getByText('Open in Issues')).toBeInTheDocument();
- await userEvent.click(screen.getByText('Open in Issues'));
- expect(router.push).toHaveBeenCalledWith(
- '/organizations/org-slug/issues/?environment=prod&project=1&query=event.type%3Adefault&sort=freq&statsPeriod=14d'
- );
- });
- it('calls onDuplicate when Duplicate Widget is clicked', async function () {
- const mock = jest.fn();
- render(
- <WidgetCard
- api={api}
- organization={organization}
- widget={widget}
- selection={selection}
- isEditingDashboard={false}
- onDelete={() => 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('disables the duplicate widget button if max widgets is reached', async function () {
- const mock = jest.fn();
- render(
- <WidgetCard
- api={api}
- organization={organization}
- widget={widget}
- selection={selection}
- isEditingDashboard={false}
- onDelete={() => 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('maps lifetimeEvents and lifetimeUsers headers to more human readable', async function () {
- MemberListStore.loadInitialData([]);
- render(
- <WidgetCard
- api={api}
- organization={organization}
- widget={{
- ...widget,
- queries: [
- {
- ...widget.queries[0],
- fields: ['issue', 'assignee', 'title', 'lifetimeEvents', 'lifetimeUsers'],
- },
- ],
- }}
- selection={selection}
- isEditingDashboard={false}
- onDelete={() => undefined}
- onEdit={() => undefined}
- onDuplicate={() => undefined}
- renderErrorMessage={() => undefined}
- showContextMenu
- widgetLimitReached={false}
- />
- );
- expect(await screen.findByText('Lifetime Events')).toBeInTheDocument();
- expect(screen.getByText('Lifetime Users')).toBeInTheDocument();
- });
- });
|