widgetContainer.spec.tsx 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215
  1. import {
  2. initializeData as _initializeData,
  3. InitializeDataSettings,
  4. } from 'sentry-test/performance/initializePerformanceData';
  5. import {render, screen, userEvent} from 'sentry-test/reactTestingLibrary';
  6. import {MetricsCardinalityProvider} from 'sentry/utils/performance/contexts/metricsCardinality';
  7. import {MEPSettingProvider} from 'sentry/utils/performance/contexts/metricsEnhancedSetting';
  8. import {PageAlert, PageAlertProvider} from 'sentry/utils/performance/contexts/pageAlert';
  9. import {PerformanceDisplayProvider} from 'sentry/utils/performance/contexts/performanceDisplayContext';
  10. import {OrganizationContext} from 'sentry/views/organizationContext';
  11. import WidgetContainer from 'sentry/views/performance/landing/widgets/components/widgetContainer';
  12. import {PerformanceWidgetSetting} from 'sentry/views/performance/landing/widgets/widgetDefinitions';
  13. import {ProjectPerformanceType} from 'sentry/views/performance/utils';
  14. import {QUERY_LIMIT_PARAM} from '../utils';
  15. const initializeData = (query = {}, rest: InitializeDataSettings = {}) => {
  16. const data = _initializeData({
  17. query: {statsPeriod: '7d', environment: ['prod'], project: [-42], ...query},
  18. ...rest,
  19. });
  20. data.eventView.additionalConditions.addFilterValues('transaction.op', ['pageload']);
  21. return data;
  22. };
  23. function WrappedComponent({data, withStaticFilters = false, ...rest}) {
  24. return (
  25. <OrganizationContext.Provider value={data.organization}>
  26. <MetricsCardinalityProvider
  27. location={data.router.location}
  28. organization={data.organization}
  29. >
  30. <MEPSettingProvider forceTransactions>
  31. <PerformanceDisplayProvider
  32. value={{performanceType: ProjectPerformanceType.ANY}}
  33. >
  34. <WidgetContainer
  35. chartHeight={100}
  36. allowedCharts={[
  37. PerformanceWidgetSetting.TPM_AREA,
  38. PerformanceWidgetSetting.FAILURE_RATE_AREA,
  39. PerformanceWidgetSetting.USER_MISERY_AREA,
  40. PerformanceWidgetSetting.DURATION_HISTOGRAM,
  41. ]}
  42. rowChartSettings={[]}
  43. withStaticFilters={withStaticFilters}
  44. forceDefaultChartSetting
  45. {...data}
  46. {...rest}
  47. />
  48. </PerformanceDisplayProvider>
  49. </MEPSettingProvider>
  50. </MetricsCardinalityProvider>
  51. </OrganizationContext.Provider>
  52. );
  53. }
  54. const issuesPredicate = (url, options) =>
  55. url.includes('events') && options.query?.query.includes('error');
  56. describe('Performance > Widgets > WidgetContainer', function () {
  57. let wrapper;
  58. let eventStatsMock;
  59. let eventsTrendsStats;
  60. let eventsMock;
  61. let issuesListMock;
  62. beforeEach(function () {
  63. eventStatsMock = MockApiClient.addMockResponse({
  64. method: 'GET',
  65. url: `/organizations/org-slug/events-stats/`,
  66. body: [],
  67. });
  68. eventsMock = MockApiClient.addMockResponse({
  69. method: 'GET',
  70. url: `/organizations/org-slug/events/`,
  71. body: {
  72. data: [{}],
  73. meta: {},
  74. },
  75. match: [(...args) => !issuesPredicate(...args)],
  76. });
  77. issuesListMock = MockApiClient.addMockResponse({
  78. method: 'GET',
  79. url: `/organizations/org-slug/events/`,
  80. body: {
  81. data: [
  82. {
  83. 'issue.id': 123,
  84. transaction: '/issue/:id/',
  85. title: 'Error: Something is broken.',
  86. 'project.id': 1,
  87. count: 3100,
  88. issue: 'JAVASCRIPT-ABCD',
  89. },
  90. ],
  91. },
  92. match: [(...args) => issuesPredicate(...args)],
  93. });
  94. eventsTrendsStats = MockApiClient.addMockResponse({
  95. method: 'GET',
  96. url: '/organizations/org-slug/events-trends-stats/',
  97. body: [],
  98. });
  99. MockApiClient.addMockResponse({
  100. method: 'GET',
  101. url: `/organizations/org-slug/metrics-compatibility/`,
  102. body: [],
  103. });
  104. MockApiClient.addMockResponse({
  105. method: 'GET',
  106. url: `/organizations/org-slug/metrics-compatibility-sums/`,
  107. body: [],
  108. });
  109. });
  110. afterEach(function () {
  111. if (wrapper) {
  112. wrapper.unmount();
  113. wrapper = undefined;
  114. }
  115. });
  116. it('Check requests when changing widget props', function () {
  117. const data = initializeData();
  118. wrapper = render(
  119. <WrappedComponent
  120. data={data}
  121. defaultChartSetting={PerformanceWidgetSetting.TPM_AREA}
  122. />
  123. );
  124. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  125. // Change eventView reference
  126. data.eventView = data.eventView.clone();
  127. wrapper.rerender(
  128. <WrappedComponent
  129. data={data}
  130. defaultChartSetting={PerformanceWidgetSetting.TPM_AREA}
  131. />
  132. );
  133. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  134. // Change eventView statsperiod
  135. const modifiedData = initializeData({
  136. statsPeriod: '14d',
  137. });
  138. wrapper.rerender(
  139. <WrappedComponent
  140. data={modifiedData}
  141. defaultChartSetting={PerformanceWidgetSetting.TPM_AREA}
  142. />
  143. );
  144. expect(eventStatsMock).toHaveBeenCalledTimes(2);
  145. expect(eventStatsMock).toHaveBeenNthCalledWith(
  146. 2,
  147. expect.anything(),
  148. expect.objectContaining({
  149. query: expect.objectContaining({
  150. interval: '1h',
  151. partial: '1',
  152. query: 'transaction.op:pageload',
  153. statsPeriod: '28d',
  154. yAxis: 'tpm()',
  155. }),
  156. })
  157. );
  158. });
  159. it('Check requests when changing widget props for GenericDiscoverQuery based widget', function () {
  160. const data = initializeData();
  161. wrapper = render(
  162. <MEPSettingProvider forceTransactions>
  163. <WrappedComponent
  164. data={data}
  165. defaultChartSetting={PerformanceWidgetSetting.MOST_IMPROVED}
  166. />
  167. </MEPSettingProvider>
  168. );
  169. expect(eventsTrendsStats).toHaveBeenCalledTimes(1);
  170. // Change eventView reference
  171. data.eventView = data.eventView.clone();
  172. wrapper.rerender(
  173. <MEPSettingProvider forceTransactions>
  174. <WrappedComponent
  175. data={data}
  176. defaultChartSetting={PerformanceWidgetSetting.MOST_IMPROVED}
  177. />
  178. </MEPSettingProvider>
  179. );
  180. expect(eventsTrendsStats).toHaveBeenCalledTimes(1);
  181. // Change eventView statsperiod
  182. const modifiedData = initializeData({
  183. statsPeriod: '14d',
  184. });
  185. wrapper.rerender(
  186. <MEPSettingProvider forceTransactions>
  187. <WrappedComponent
  188. data={modifiedData}
  189. defaultChartSetting={PerformanceWidgetSetting.MOST_IMPROVED}
  190. />
  191. </MEPSettingProvider>
  192. );
  193. expect(eventsTrendsStats).toHaveBeenCalledTimes(2);
  194. expect(eventsTrendsStats).toHaveBeenNthCalledWith(
  195. 2,
  196. expect.anything(),
  197. expect.objectContaining({
  198. query: expect.objectContaining({
  199. cursor: '0:0:1',
  200. environment: ['prod'],
  201. field: ['transaction', 'project'],
  202. interval: undefined,
  203. middle: undefined,
  204. noPagination: true,
  205. per_page: QUERY_LIMIT_PARAM,
  206. project: ['-42'],
  207. query:
  208. 'transaction.op:pageload tpm():>0.01 count_percentage():>0.25 count_percentage():<4 trend_percentage():>0% confidence():>6',
  209. sort: 'trend_percentage()',
  210. statsPeriod: '14d',
  211. trendFunction: 'p50(transaction.duration)',
  212. trendType: 'improved',
  213. }),
  214. })
  215. );
  216. });
  217. it('should call PageError Provider when errors are present', async function () {
  218. const data = initializeData();
  219. eventStatsMock = MockApiClient.addMockResponse({
  220. method: 'GET',
  221. url: `/organizations/org-slug/events-stats/`,
  222. statusCode: 400,
  223. body: {
  224. detail: 'Request did not work :(',
  225. },
  226. });
  227. wrapper = render(
  228. <PageAlertProvider>
  229. <PageAlert />
  230. <WrappedComponent
  231. data={data}
  232. defaultChartSetting={PerformanceWidgetSetting.TPM_AREA}
  233. />
  234. </PageAlertProvider>
  235. );
  236. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  237. 'Transactions Per Minute'
  238. );
  239. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  240. expect(await screen.findByTestId('page-error-alert')).toHaveTextContent(
  241. 'Request did not work :('
  242. );
  243. });
  244. it('TPM Widget', async function () {
  245. const data = initializeData();
  246. wrapper = render(
  247. <WrappedComponent
  248. data={data}
  249. defaultChartSetting={PerformanceWidgetSetting.TPM_AREA}
  250. />
  251. );
  252. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  253. 'Transactions Per Minute'
  254. );
  255. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  256. expect(eventStatsMock).toHaveBeenNthCalledWith(
  257. 1,
  258. expect.anything(),
  259. expect.objectContaining({
  260. query: expect.objectContaining({
  261. interval: '1h',
  262. partial: '1',
  263. query: 'transaction.op:pageload',
  264. statsPeriod: '14d',
  265. yAxis: 'tpm()',
  266. }),
  267. })
  268. );
  269. });
  270. it('Failure Rate Widget', async function () {
  271. const data = initializeData();
  272. wrapper = render(
  273. <WrappedComponent
  274. data={data}
  275. defaultChartSetting={PerformanceWidgetSetting.FAILURE_RATE_AREA}
  276. />
  277. );
  278. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  279. 'Failure Rate'
  280. );
  281. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  282. expect(eventStatsMock).toHaveBeenNthCalledWith(
  283. 1,
  284. expect.anything(),
  285. expect.objectContaining({
  286. query: expect.objectContaining({
  287. interval: '1h',
  288. partial: '1',
  289. query: 'transaction.op:pageload',
  290. statsPeriod: '14d',
  291. yAxis: 'failure_rate()',
  292. }),
  293. })
  294. );
  295. });
  296. it('Widget with MEP enabled and metric meta set to true', async function () {
  297. const data = initializeData(
  298. {},
  299. {
  300. features: ['performance-use-metrics'],
  301. }
  302. );
  303. eventStatsMock = MockApiClient.addMockResponse({
  304. method: 'GET',
  305. url: `/organizations/org-slug/events-stats/`,
  306. body: {
  307. data: [],
  308. isMetricsData: true,
  309. },
  310. });
  311. eventsMock = MockApiClient.addMockResponse({
  312. method: 'GET',
  313. url: `/organizations/org-slug/events/`,
  314. body: {
  315. data: [{}],
  316. meta: {isMetricsData: true},
  317. },
  318. });
  319. wrapper = render(
  320. <WrappedComponent
  321. data={data}
  322. defaultChartSetting={PerformanceWidgetSetting.FAILURE_RATE_AREA}
  323. />
  324. );
  325. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  326. expect(eventStatsMock).toHaveBeenNthCalledWith(
  327. 1,
  328. expect.anything(),
  329. expect.objectContaining({
  330. query: expect.objectContaining({dataset: 'metrics'}),
  331. })
  332. );
  333. expect(eventsMock).toHaveBeenCalledTimes(1);
  334. expect(eventsMock).toHaveBeenCalledWith(
  335. expect.anything(),
  336. expect.objectContaining({
  337. query: expect.objectContaining({dataset: 'metrics'}),
  338. })
  339. );
  340. expect(await screen.findByTestId('has-metrics-data-tag')).toHaveTextContent(
  341. 'processed'
  342. );
  343. });
  344. it('Widget with MEP enabled and metric meta set to undefined', async function () {
  345. const data = initializeData(
  346. {},
  347. {
  348. features: ['performance-use-metrics'],
  349. }
  350. );
  351. eventStatsMock = MockApiClient.addMockResponse({
  352. method: 'GET',
  353. url: `/organizations/org-slug/events-stats/`,
  354. body: {
  355. data: [],
  356. isMetricsData: undefined,
  357. },
  358. });
  359. wrapper = render(
  360. <WrappedComponent
  361. data={data}
  362. defaultChartSetting={PerformanceWidgetSetting.FAILURE_RATE_AREA}
  363. />
  364. );
  365. expect(await screen.findByTestId('no-metrics-data-tag')).toBeInTheDocument();
  366. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  367. expect(eventStatsMock).toHaveBeenNthCalledWith(
  368. 1,
  369. expect.anything(),
  370. expect.objectContaining({
  371. query: expect.objectContaining({dataset: 'metrics'}),
  372. })
  373. );
  374. });
  375. it('Widget with MEP enabled and metric meta set to false', async function () {
  376. const data = initializeData(
  377. {},
  378. {
  379. features: ['performance-use-metrics'],
  380. }
  381. );
  382. eventStatsMock = MockApiClient.addMockResponse({
  383. method: 'GET',
  384. url: `/organizations/org-slug/events-stats/`,
  385. body: {
  386. data: [],
  387. isMetricsData: false,
  388. },
  389. });
  390. eventsMock = MockApiClient.addMockResponse({
  391. method: 'GET',
  392. url: `/organizations/org-slug/events/`,
  393. body: {
  394. data: [{}],
  395. meta: {isMetricsData: false},
  396. },
  397. });
  398. wrapper = render(
  399. <WrappedComponent
  400. data={data}
  401. defaultChartSetting={PerformanceWidgetSetting.FAILURE_RATE_AREA}
  402. />
  403. );
  404. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  405. expect(eventStatsMock).toHaveBeenNthCalledWith(
  406. 1,
  407. expect.anything(),
  408. expect.objectContaining({
  409. query: expect.objectContaining({dataset: 'metrics'}),
  410. })
  411. );
  412. expect(eventsMock).toHaveBeenCalledTimes(1);
  413. expect(eventsMock).toHaveBeenCalledWith(
  414. expect.anything(),
  415. expect.objectContaining({
  416. query: expect.objectContaining({dataset: 'metrics'}),
  417. })
  418. );
  419. expect(await screen.findByTestId('has-metrics-data-tag')).toHaveTextContent(
  420. 'indexed'
  421. );
  422. });
  423. it('User misery Widget', async function () {
  424. const data = initializeData();
  425. wrapper = render(
  426. <WrappedComponent
  427. data={data}
  428. defaultChartSetting={PerformanceWidgetSetting.USER_MISERY_AREA}
  429. />
  430. );
  431. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  432. 'User Misery'
  433. );
  434. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  435. expect(eventStatsMock).toHaveBeenNthCalledWith(
  436. 1,
  437. expect.anything(),
  438. expect.objectContaining({
  439. query: expect.objectContaining({
  440. interval: '1h',
  441. partial: '1',
  442. query: 'transaction.op:pageload',
  443. statsPeriod: '14d',
  444. yAxis: 'user_misery()',
  445. }),
  446. })
  447. );
  448. });
  449. it('Worst LCP widget', async function () {
  450. const data = initializeData();
  451. wrapper = render(
  452. <WrappedComponent
  453. data={data}
  454. defaultChartSetting={PerformanceWidgetSetting.WORST_LCP_VITALS}
  455. />
  456. );
  457. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  458. 'Worst LCP Web Vitals'
  459. );
  460. expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All');
  461. expect(eventsMock).toHaveBeenCalledTimes(1);
  462. expect(eventsMock).toHaveBeenNthCalledWith(
  463. 1,
  464. expect.anything(),
  465. expect.objectContaining({
  466. query: expect.objectContaining({
  467. environment: ['prod'],
  468. field: [
  469. 'transaction',
  470. 'title',
  471. 'project.id',
  472. 'count_web_vitals(measurements.lcp, poor)',
  473. 'count_web_vitals(measurements.lcp, meh)',
  474. 'count_web_vitals(measurements.lcp, good)',
  475. ],
  476. per_page: 4,
  477. project: ['-42'],
  478. query: 'transaction.op:pageload',
  479. sort: '-count_web_vitals(measurements.lcp, poor)',
  480. statsPeriod: '7d',
  481. }),
  482. })
  483. );
  484. });
  485. it('Worst LCP widget - MEP', async function () {
  486. const data = initializeData(
  487. {},
  488. {
  489. features: ['performance-use-metrics'],
  490. }
  491. );
  492. wrapper = render(
  493. <WrappedComponent
  494. data={data}
  495. defaultChartSetting={PerformanceWidgetSetting.WORST_LCP_VITALS}
  496. />
  497. );
  498. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  499. 'Worst LCP Web Vitals'
  500. );
  501. expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All');
  502. expect(eventsMock).toHaveBeenCalledTimes(1);
  503. expect(eventsMock).toHaveBeenNthCalledWith(
  504. 1,
  505. expect.anything(),
  506. expect.objectContaining({
  507. query: expect.objectContaining({
  508. environment: ['prod'],
  509. field: [
  510. 'transaction',
  511. 'title',
  512. 'project.id',
  513. 'count_web_vitals(measurements.lcp, poor)',
  514. 'count_web_vitals(measurements.lcp, meh)',
  515. 'count_web_vitals(measurements.lcp, good)',
  516. ],
  517. per_page: 4,
  518. project: ['-42'],
  519. query: 'transaction.op:pageload !transaction:"<< unparameterized >>"',
  520. sort: '-count_web_vitals(measurements.lcp, poor)',
  521. statsPeriod: '7d',
  522. }),
  523. })
  524. );
  525. });
  526. it('Worst FCP widget', async function () {
  527. const data = initializeData();
  528. wrapper = render(
  529. <WrappedComponent
  530. data={data}
  531. defaultChartSetting={PerformanceWidgetSetting.WORST_FCP_VITALS}
  532. />
  533. );
  534. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  535. 'Worst FCP Web Vitals'
  536. );
  537. expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All');
  538. expect(eventsMock).toHaveBeenCalledTimes(1);
  539. expect(eventsMock).toHaveBeenNthCalledWith(
  540. 1,
  541. expect.anything(),
  542. expect.objectContaining({
  543. query: expect.objectContaining({
  544. environment: ['prod'],
  545. field: [
  546. 'transaction',
  547. 'title',
  548. 'project.id',
  549. 'count_web_vitals(measurements.fcp, poor)',
  550. 'count_web_vitals(measurements.fcp, meh)',
  551. 'count_web_vitals(measurements.fcp, good)',
  552. ],
  553. per_page: 4,
  554. project: ['-42'],
  555. query: 'transaction.op:pageload',
  556. sort: '-count_web_vitals(measurements.fcp, poor)',
  557. statsPeriod: '7d',
  558. }),
  559. })
  560. );
  561. });
  562. it('Worst FID widget', async function () {
  563. const data = initializeData();
  564. wrapper = render(
  565. <WrappedComponent
  566. data={data}
  567. defaultChartSetting={PerformanceWidgetSetting.WORST_FID_VITALS}
  568. />
  569. );
  570. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  571. 'Worst FID Web Vitals'
  572. );
  573. expect(await screen.findByTestId('view-all-button')).toHaveTextContent('View All');
  574. expect(eventsMock).toHaveBeenCalledTimes(1);
  575. expect(eventsMock).toHaveBeenNthCalledWith(
  576. 1,
  577. expect.anything(),
  578. expect.objectContaining({
  579. query: expect.objectContaining({
  580. environment: ['prod'],
  581. field: [
  582. 'transaction',
  583. 'title',
  584. 'project.id',
  585. 'count_web_vitals(measurements.fid, poor)',
  586. 'count_web_vitals(measurements.fid, meh)',
  587. 'count_web_vitals(measurements.fid, good)',
  588. ],
  589. per_page: 4,
  590. project: ['-42'],
  591. query: 'transaction.op:pageload',
  592. sort: '-count_web_vitals(measurements.fid, poor)',
  593. statsPeriod: '7d',
  594. }),
  595. })
  596. );
  597. });
  598. it('LCP Histogram Widget', async function () {
  599. const data = initializeData();
  600. wrapper = render(
  601. <WrappedComponent
  602. data={data}
  603. defaultChartSetting={PerformanceWidgetSetting.LCP_HISTOGRAM}
  604. />
  605. );
  606. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  607. 'LCP Distribution'
  608. );
  609. // TODO(k-fish): Add histogram mock
  610. });
  611. it('FCP Histogram Widget', async function () {
  612. const data = initializeData();
  613. wrapper = render(
  614. <WrappedComponent
  615. data={data}
  616. defaultChartSetting={PerformanceWidgetSetting.FCP_HISTOGRAM}
  617. />
  618. );
  619. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  620. 'FCP Distribution'
  621. );
  622. // TODO(k-fish): Add histogram mock
  623. });
  624. it('Most errors widget', async function () {
  625. const data = initializeData();
  626. wrapper = render(
  627. <WrappedComponent
  628. data={data}
  629. defaultChartSetting={PerformanceWidgetSetting.MOST_RELATED_ERRORS}
  630. />
  631. );
  632. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  633. 'Most Related Errors'
  634. );
  635. expect(eventsMock).toHaveBeenCalledTimes(1);
  636. expect(eventsMock).toHaveBeenNthCalledWith(
  637. 1,
  638. expect.anything(),
  639. expect.objectContaining({
  640. query: expect.objectContaining({
  641. environment: ['prod'],
  642. field: ['transaction', 'project.id', 'failure_count()'],
  643. per_page: QUERY_LIMIT_PARAM,
  644. project: ['-42'],
  645. query: 'transaction.op:pageload failure_count():>0',
  646. sort: '-failure_count()',
  647. statsPeriod: '7d',
  648. }),
  649. })
  650. );
  651. });
  652. it('Most related issues widget', async function () {
  653. const data = initializeData();
  654. wrapper = render(
  655. <WrappedComponent
  656. data={data}
  657. defaultChartSetting={PerformanceWidgetSetting.MOST_RELATED_ISSUES}
  658. />
  659. );
  660. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  661. 'Most Related Issues'
  662. );
  663. expect(issuesListMock).toHaveBeenCalledTimes(1);
  664. expect(issuesListMock).toHaveBeenNthCalledWith(
  665. 1,
  666. expect.anything(),
  667. expect.objectContaining({
  668. query: expect.objectContaining({
  669. environment: ['prod'],
  670. field: ['issue', 'transaction', 'title', 'project.id', 'count()'],
  671. per_page: QUERY_LIMIT_PARAM,
  672. project: ['-42'],
  673. query: 'event.type:error !tags[transaction]:"" count():>0',
  674. sort: '-count()',
  675. statsPeriod: '7d',
  676. }),
  677. })
  678. );
  679. });
  680. it('Switching from issues to errors widget', async function () {
  681. const data = initializeData();
  682. wrapper = render(
  683. <WrappedComponent
  684. data={data}
  685. defaultChartSetting={PerformanceWidgetSetting.MOST_RELATED_ISSUES}
  686. />
  687. );
  688. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  689. 'Most Related Issues'
  690. );
  691. expect(issuesListMock).toHaveBeenCalledTimes(1);
  692. wrapper.rerender(
  693. <WrappedComponent
  694. data={data}
  695. defaultChartSetting={PerformanceWidgetSetting.MOST_RELATED_ERRORS}
  696. />
  697. );
  698. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  699. 'Most Related Errors'
  700. );
  701. expect(eventsMock).toHaveBeenCalledTimes(1);
  702. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  703. });
  704. it('Most improved trends widget', async function () {
  705. const data = initializeData();
  706. wrapper = render(
  707. <MEPSettingProvider forceTransactions>
  708. <WrappedComponent
  709. data={data}
  710. defaultChartSetting={PerformanceWidgetSetting.MOST_IMPROVED}
  711. />
  712. </MEPSettingProvider>
  713. );
  714. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  715. 'Most Improved'
  716. );
  717. expect(eventsTrendsStats).toHaveBeenCalledTimes(1);
  718. expect(eventsTrendsStats).toHaveBeenNthCalledWith(
  719. 1,
  720. expect.anything(),
  721. expect.objectContaining({
  722. query: expect.objectContaining({
  723. environment: ['prod'],
  724. field: ['transaction', 'project'],
  725. interval: undefined,
  726. middle: undefined,
  727. per_page: QUERY_LIMIT_PARAM,
  728. project: ['-42'],
  729. query:
  730. 'transaction.op:pageload tpm():>0.01 count_percentage():>0.25 count_percentage():<4 trend_percentage():>0% confidence():>6',
  731. sort: 'trend_percentage()',
  732. statsPeriod: '7d',
  733. trendFunction: 'p50(transaction.duration)',
  734. trendType: 'improved',
  735. }),
  736. })
  737. );
  738. });
  739. it('Most time spent in db queries widget', async function () {
  740. const data = initializeData();
  741. wrapper = render(
  742. <MEPSettingProvider forceTransactions>
  743. <WrappedComponent
  744. data={data}
  745. defaultChartSetting={PerformanceWidgetSetting.MOST_TIME_SPENT_DB_QUERIES}
  746. />
  747. </MEPSettingProvider>
  748. );
  749. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  750. 'Most Time-Consuming Queries'
  751. );
  752. expect(eventsMock).toHaveBeenCalledTimes(1);
  753. expect(eventsMock).toHaveBeenNthCalledWith(
  754. 1,
  755. expect.anything(),
  756. expect.objectContaining({
  757. query: expect.objectContaining({
  758. dataset: 'spansMetrics',
  759. environment: ['prod'],
  760. field: [
  761. 'span.op',
  762. 'span.group',
  763. 'project.id',
  764. 'span.description',
  765. 'sum(span.self_time)',
  766. 'avg(span.self_time)',
  767. 'time_spent_percentage()',
  768. ],
  769. per_page: QUERY_LIMIT_PARAM,
  770. project: ['-42'],
  771. query: 'has:span.description span.module:db transaction.op:pageload',
  772. sort: '-time_spent_percentage()',
  773. statsPeriod: '7d',
  774. }),
  775. })
  776. );
  777. });
  778. it('Most time consuming resources widget', async function () {
  779. const data = initializeData();
  780. wrapper = render(
  781. <MEPSettingProvider forceTransactions>
  782. <WrappedComponent
  783. data={data}
  784. defaultChartSetting={PerformanceWidgetSetting.MOST_TIME_CONSUMING_RESOURCES}
  785. />
  786. </MEPSettingProvider>
  787. );
  788. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  789. 'Most Time Consuming Resources'
  790. );
  791. expect(eventsMock).toHaveBeenCalledTimes(1);
  792. expect(eventsMock).toHaveBeenNthCalledWith(
  793. 1,
  794. expect.anything(),
  795. expect.objectContaining({
  796. query: expect.objectContaining({
  797. dataset: 'spansMetrics',
  798. environment: ['prod'],
  799. field: [
  800. 'span.description',
  801. 'span.op',
  802. 'project.id',
  803. 'span.group',
  804. 'sum(span.self_time)',
  805. 'avg(span.self_time)',
  806. 'time_spent_percentage()',
  807. ],
  808. per_page: QUERY_LIMIT_PARAM,
  809. project: ['-42'],
  810. query:
  811. '!span.description:browser-extension://* resource.render_blocking_status:blocking ( span.op:resource.script OR file_extension:css OR file_extension:[woff,woff2,ttf,otf,eot] OR file_extension:[jpg,jpeg,png,gif,svg,webp,apng,avif] OR span.op:resource.img ) transaction.op:pageload',
  812. sort: '-time_spent_percentage()',
  813. statsPeriod: '7d',
  814. }),
  815. })
  816. );
  817. });
  818. it('Best Page Opportunities widget', async function () {
  819. const data = initializeData();
  820. wrapper = render(
  821. <MEPSettingProvider forceTransactions>
  822. <WrappedComponent
  823. data={data}
  824. defaultChartSetting={PerformanceWidgetSetting.HIGHEST_OPPORTUNITY_PAGES}
  825. />
  826. </MEPSettingProvider>
  827. );
  828. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  829. 'Best Page Opportunities'
  830. );
  831. expect(eventsMock).toHaveBeenCalledTimes(2);
  832. expect(eventsMock).toHaveBeenNthCalledWith(
  833. 1,
  834. expect.anything(),
  835. expect.objectContaining({
  836. query: expect.objectContaining({
  837. dataset: 'metrics',
  838. field: [
  839. 'p75(measurements.lcp)',
  840. 'p75(measurements.fcp)',
  841. 'p75(measurements.cls)',
  842. 'p75(measurements.ttfb)',
  843. 'p75(measurements.fid)',
  844. 'p75(transaction.duration)',
  845. 'count_web_vitals(measurements.lcp, any)',
  846. 'count_web_vitals(measurements.fcp, any)',
  847. 'count_web_vitals(measurements.cls, any)',
  848. 'count_web_vitals(measurements.fid, any)',
  849. 'count_web_vitals(measurements.ttfb, any)',
  850. 'count()',
  851. ],
  852. query: 'transaction.op:pageload',
  853. }),
  854. })
  855. );
  856. expect(eventsMock).toHaveBeenNthCalledWith(
  857. 2,
  858. expect.anything(),
  859. expect.objectContaining({
  860. query: expect.objectContaining({
  861. dataset: 'metrics',
  862. field: [
  863. 'transaction',
  864. 'transaction.op',
  865. 'p75(measurements.lcp)',
  866. 'p75(measurements.fcp)',
  867. 'p75(measurements.cls)',
  868. 'p75(measurements.ttfb)',
  869. 'p75(measurements.fid)',
  870. 'count_web_vitals(measurements.lcp, any)',
  871. 'count_web_vitals(measurements.fcp, any)',
  872. 'count_web_vitals(measurements.cls, any)',
  873. 'count_web_vitals(measurements.ttfb, any)',
  874. 'count_web_vitals(measurements.fid, any)',
  875. 'count()',
  876. ],
  877. per_page: 4,
  878. query: 'transaction.op:pageload',
  879. sort: '-count()',
  880. }),
  881. })
  882. );
  883. });
  884. it('Most regressed trends widget', async function () {
  885. const data = initializeData();
  886. wrapper = render(
  887. <WrappedComponent
  888. data={data}
  889. defaultChartSetting={PerformanceWidgetSetting.MOST_REGRESSED}
  890. />
  891. );
  892. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  893. 'Most Regressed'
  894. );
  895. expect(eventsTrendsStats).toHaveBeenCalledTimes(1);
  896. expect(eventsTrendsStats).toHaveBeenNthCalledWith(
  897. 1,
  898. expect.anything(),
  899. expect.objectContaining({
  900. query: expect.objectContaining({
  901. environment: ['prod'],
  902. field: ['transaction', 'project'],
  903. interval: undefined,
  904. middle: undefined,
  905. per_page: QUERY_LIMIT_PARAM,
  906. project: ['-42'],
  907. query:
  908. 'transaction.op:pageload tpm():>0.01 count_percentage():>0.25 count_percentage():<4 trend_percentage():>0% confidence():>6',
  909. sort: '-trend_percentage()',
  910. statsPeriod: '7d',
  911. trendFunction: 'p50(transaction.duration)',
  912. trendType: 'regression',
  913. }),
  914. })
  915. );
  916. });
  917. it('Most slow frames widget', async function () {
  918. const data = initializeData();
  919. wrapper = render(
  920. <WrappedComponent
  921. data={data}
  922. defaultChartSetting={PerformanceWidgetSetting.MOST_SLOW_FRAMES}
  923. />
  924. );
  925. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  926. 'Most Slow Frames'
  927. );
  928. expect(eventsMock).toHaveBeenCalledTimes(1);
  929. expect(eventsMock).toHaveBeenNthCalledWith(
  930. 1,
  931. expect.anything(),
  932. expect.objectContaining({
  933. query: expect.objectContaining({
  934. cursor: '0:0:1',
  935. environment: ['prod'],
  936. field: ['transaction', 'project.id', 'epm()', 'avg(measurements.frames_slow)'],
  937. noPagination: true,
  938. per_page: QUERY_LIMIT_PARAM,
  939. project: ['-42'],
  940. query: 'transaction.op:pageload epm():>0.01 avg(measurements.frames_slow):>0',
  941. sort: '-avg(measurements.frames_slow)',
  942. statsPeriod: '7d',
  943. }),
  944. })
  945. );
  946. expect(await screen.findByTestId('empty-state')).toBeInTheDocument();
  947. });
  948. it('Most slow frames widget - MEP', async function () {
  949. const data = initializeData(
  950. {},
  951. {
  952. features: ['performance-use-metrics'],
  953. }
  954. );
  955. wrapper = render(
  956. <WrappedComponent
  957. data={data}
  958. defaultChartSetting={PerformanceWidgetSetting.MOST_SLOW_FRAMES}
  959. />
  960. );
  961. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  962. 'Most Slow Frames'
  963. );
  964. expect(eventsMock).toHaveBeenCalledTimes(1);
  965. expect(eventsMock).toHaveBeenNthCalledWith(
  966. 1,
  967. expect.anything(),
  968. expect.objectContaining({
  969. query: expect.objectContaining({
  970. cursor: '0:0:1',
  971. environment: ['prod'],
  972. field: ['transaction', 'project.id', 'epm()', 'avg(measurements.frames_slow)'],
  973. noPagination: true,
  974. per_page: QUERY_LIMIT_PARAM,
  975. project: ['-42'],
  976. query: 'transaction.op:pageload epm():>0.01 avg(measurements.frames_slow):>0',
  977. sort: '-avg(measurements.frames_slow)',
  978. statsPeriod: '7d',
  979. }),
  980. })
  981. );
  982. expect(await screen.findByTestId('empty-state')).toBeInTheDocument();
  983. });
  984. it('Most frozen frames widget', async function () {
  985. const data = initializeData();
  986. wrapper = render(
  987. <WrappedComponent
  988. data={data}
  989. defaultChartSetting={PerformanceWidgetSetting.MOST_FROZEN_FRAMES}
  990. />
  991. );
  992. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  993. 'Most Frozen Frames'
  994. );
  995. expect(eventsMock).toHaveBeenCalledTimes(1);
  996. expect(eventsMock).toHaveBeenNthCalledWith(
  997. 1,
  998. expect.anything(),
  999. expect.objectContaining({
  1000. query: expect.objectContaining({
  1001. cursor: '0:0:1',
  1002. environment: ['prod'],
  1003. field: [
  1004. 'transaction',
  1005. 'project.id',
  1006. 'epm()',
  1007. 'avg(measurements.frames_frozen)',
  1008. ],
  1009. noPagination: true,
  1010. per_page: QUERY_LIMIT_PARAM,
  1011. project: ['-42'],
  1012. query: 'transaction.op:pageload epm():>0.01 avg(measurements.frames_frozen):>0',
  1013. sort: '-avg(measurements.frames_frozen)',
  1014. statsPeriod: '7d',
  1015. }),
  1016. })
  1017. );
  1018. expect(await screen.findByTestId('empty-state')).toBeInTheDocument();
  1019. });
  1020. it('Able to change widget type from menu', async function () {
  1021. const data = initializeData();
  1022. const setRowChartSettings = jest.fn(() => {});
  1023. wrapper = render(
  1024. <WrappedComponent
  1025. data={data}
  1026. defaultChartSetting={PerformanceWidgetSetting.FAILURE_RATE_AREA}
  1027. setRowChartSettings={setRowChartSettings}
  1028. />
  1029. );
  1030. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  1031. 'Failure Rate'
  1032. );
  1033. expect(eventStatsMock).toHaveBeenCalledTimes(1);
  1034. expect(setRowChartSettings).toHaveBeenCalledTimes(0);
  1035. await userEvent.click(await screen.findByLabelText('More'));
  1036. await userEvent.click(await screen.findByText('User Misery'));
  1037. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  1038. 'User Misery'
  1039. );
  1040. expect(eventStatsMock).toHaveBeenCalledTimes(2);
  1041. expect(setRowChartSettings).toHaveBeenCalledTimes(1);
  1042. });
  1043. it('Chart settings passed from the row are disabled in the menu', async function () {
  1044. const data = initializeData();
  1045. const setRowChartSettings = jest.fn(() => {});
  1046. wrapper = render(
  1047. <WrappedComponent
  1048. data={data}
  1049. defaultChartSetting={PerformanceWidgetSetting.FAILURE_RATE_AREA}
  1050. setRowChartSettings={setRowChartSettings}
  1051. rowChartSettings={[
  1052. PerformanceWidgetSetting.FAILURE_RATE_AREA,
  1053. PerformanceWidgetSetting.USER_MISERY_AREA,
  1054. ]}
  1055. />
  1056. );
  1057. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  1058. 'Failure Rate'
  1059. );
  1060. // Open context menu
  1061. await userEvent.click(await screen.findByLabelText('More'));
  1062. // Check that the the "User Misery" option is disabled by clicking on it,
  1063. // expecting that the selected option doesn't change
  1064. const userMiseryOption = await screen.findByRole('option', {name: 'User Misery'});
  1065. await userEvent.click(userMiseryOption);
  1066. expect(await screen.findByTestId('performance-widget-title')).toHaveTextContent(
  1067. 'Failure Rate'
  1068. );
  1069. });
  1070. });