import {TagsFixture} from 'sentry-fixture/tags'; import {initializeOrg} from 'sentry-test/initializeOrg'; import {act, render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary'; import {DEFAULT_DEBOUNCE_DURATION} from 'sentry/constants'; import ProjectsStore from 'sentry/stores/projectsStore'; import type {Organization} from 'sentry/types'; import {DashboardWidgetSource} from 'sentry/views/dashboards/types'; import WidgetBuilder from 'sentry/views/dashboards/widgetBuilder'; jest.unmock('lodash/debounce'); function mockRequests(orgSlug: Organization['slug']) { const eventsMock = MockApiClient.addMockResponse({ url: `/organizations/${orgSlug}/events/`, method: 'GET', statusCode: 200, body: { meta: {}, data: [], }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/tags/', method: 'GET', body: TagsFixture(), }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/users/', body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/projects/', method: 'GET', body: [], }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/measurements-meta/', method: 'GET', body: {'measurements.custom.measurement': {functions: ['p99']}}, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/metrics-compatibility/', method: 'GET', body: { incompatible_projects: [], compatible_projects: [1], }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/metrics-compatibility-sums/', method: 'GET', body: { sum: { metrics: 988803, metrics_null: 0, metrics_unparam: 132, }, }, }); MockApiClient.addMockResponse({ url: '/organizations/org-slug/releases/', body: [], }); return {eventsMock}; } describe('VisualizationStep', function () { const {organization, router, routerContext} = initializeOrg({ organization: { features: ['dashboards-edit', 'global-views', 'dashboards-mep'], }, router: { location: { query: { source: DashboardWidgetSource.DASHBOARDS, }, }, }, }); beforeEach(function () { ProjectsStore.loadInitialData(organization.projects); }); it('debounce works as expected and requests are not triggered often', async function () { const {eventsMock} = mockRequests(organization.slug); jest.useFakeTimers(); render( , { context: routerContext, organization, } ); await waitFor(() => expect(eventsMock).toHaveBeenCalledTimes(1)); await userEvent.type(await screen.findByPlaceholderText('Alias'), 'abc', { delay: null, }); act(() => jest.advanceTimersByTime(DEFAULT_DEBOUNCE_DURATION + 1)); jest.useRealTimers(); await waitFor(() => expect(eventsMock).toHaveBeenCalledTimes(1)); }); it('displays stored data alert', async function () { mockRequests(organization.slug); MockApiClient.addMockResponse({ url: `/organizations/${organization.slug}/events/`, method: 'GET', statusCode: 200, body: { meta: {isMetricsData: false}, data: [], }, }); render( , { context: routerContext, organization: { ...organization, features: [...organization.features, 'dynamic-sampling', 'mep-rollout-flag'], }, } ); await screen.findByText(/we've automatically adjusted your results/i); }); it('uses release from URL params when querying', async function () { const {eventsMock} = mockRequests(organization.slug); render( , { context: routerContext, organization, } ); await waitFor(() => expect(eventsMock).toHaveBeenCalledWith( '/organizations/org-slug/events/', expect.objectContaining({ query: expect.objectContaining({query: ' release:"v1" '}), }) ) ); }); it('does not trigger an extra events request when adding a column', async function () { const {eventsMock} = mockRequests(organization.slug); render( , { context: routerContext, organization, } ); await userEvent.click(screen.getByText('Add a Column')); // Only called once on the initial render await waitFor(() => expect(eventsMock).toHaveBeenCalledTimes(1)); }); });