httpSamplesPanel.spec.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. import {OrganizationFixture} from 'sentry-fixture/organization';
  2. import {
  3. render,
  4. screen,
  5. userEvent,
  6. waitForElementToBeRemoved,
  7. } from 'sentry-test/reactTestingLibrary';
  8. import {useLocation} from 'sentry/utils/useLocation';
  9. import useOrganization from 'sentry/utils/useOrganization';
  10. import usePageFilters from 'sentry/utils/usePageFilters';
  11. import {HTTPSamplesPanel} from 'sentry/views/performance/http/httpSamplesPanel';
  12. jest.mock('sentry/utils/useLocation');
  13. jest.mock('sentry/utils/usePageFilters');
  14. jest.mock('sentry/utils/useOrganization');
  15. describe('HTTPSamplesPanel', () => {
  16. const organization = OrganizationFixture();
  17. let ribbonRequestMock;
  18. jest.mocked(usePageFilters).mockReturnValue({
  19. isReady: true,
  20. desyncedFilters: new Set(),
  21. pinnedFilters: new Set(),
  22. shouldPersist: true,
  23. selection: {
  24. datetime: {
  25. period: '10d',
  26. start: null,
  27. end: null,
  28. utc: false,
  29. },
  30. environments: [],
  31. projects: [],
  32. },
  33. });
  34. jest.mocked(useLocation).mockReturnValue({
  35. pathname: '',
  36. search: '',
  37. query: {
  38. domain: '*.sentry.dev',
  39. statsPeriod: '10d',
  40. transaction: '/api/0/users',
  41. transactionMethod: 'GET',
  42. panel: 'status',
  43. },
  44. hash: '',
  45. state: undefined,
  46. action: 'PUSH',
  47. key: '',
  48. });
  49. jest.mocked(useOrganization).mockReturnValue(organization);
  50. beforeEach(() => {
  51. jest.clearAllMocks();
  52. ribbonRequestMock = MockApiClient.addMockResponse({
  53. url: `/organizations/${organization.slug}/events/`,
  54. method: 'GET',
  55. match: [
  56. MockApiClient.matchQuery({
  57. referrer: 'api.starfish.http-module-samples-panel-metrics-ribbon',
  58. }),
  59. ],
  60. body: {
  61. data: [
  62. {
  63. 'project.id': 1,
  64. 'spm()': 22.18,
  65. 'http_response_rate(3)': 0.01,
  66. 'http_response_rate(4)': 0.025,
  67. 'http_response_rate(5)': 0.015,
  68. 'avg(span.self_time)': 140.2,
  69. 'sum(span.self_time)': 2709238,
  70. },
  71. ],
  72. meta: {
  73. fields: {
  74. 'spm()': 'rate',
  75. 'avg(span.self_time)': 'duration',
  76. 'http_response_rate(3)': 'percentage',
  77. 'http_response_rate(4)': 'percentage',
  78. 'http_response_rate(5)': 'percentage',
  79. 'sum(span.self_time)': 'duration',
  80. },
  81. },
  82. },
  83. });
  84. });
  85. afterAll(() => {
  86. jest.resetAllMocks();
  87. });
  88. describe('status panel', () => {
  89. let chartRequestMock;
  90. beforeEach(() => {
  91. jest.mocked(useLocation).mockReturnValue({
  92. pathname: '',
  93. search: '',
  94. query: {
  95. domain: '*.sentry.dev',
  96. statsPeriod: '10d',
  97. transaction: '/api/0/users',
  98. transactionMethod: 'GET',
  99. panel: 'status',
  100. },
  101. hash: '',
  102. state: undefined,
  103. action: 'PUSH',
  104. key: '',
  105. });
  106. chartRequestMock = MockApiClient.addMockResponse({
  107. url: `/organizations/${organization.slug}/events-stats/`,
  108. method: 'GET',
  109. match: [
  110. MockApiClient.matchQuery({
  111. referrer: 'api.starfish.http-module-samples-panel-response-code-chart',
  112. }),
  113. ],
  114. body: {
  115. 'spm()': {
  116. data: [
  117. [1699907700, [{count: 7810.2}]],
  118. [1699908000, [{count: 1216.8}]],
  119. ],
  120. },
  121. },
  122. });
  123. });
  124. it('fetches panel data', async () => {
  125. render(<HTTPSamplesPanel />);
  126. expect(ribbonRequestMock).toHaveBeenNthCalledWith(
  127. 1,
  128. `/organizations/${organization.slug}/events/`,
  129. expect.objectContaining({
  130. method: 'GET',
  131. query: {
  132. dataset: 'spansMetrics',
  133. environment: [],
  134. field: [
  135. 'spm()',
  136. 'avg(span.self_time)',
  137. 'sum(span.self_time)',
  138. 'http_response_rate(3)',
  139. 'http_response_rate(4)',
  140. 'http_response_rate(5)',
  141. 'time_spent_percentage()',
  142. ],
  143. per_page: 50,
  144. project: [],
  145. query:
  146. 'span.module:http span.domain:"\\*.sentry.dev" transaction:/api/0/users',
  147. referrer: 'api.starfish.http-module-samples-panel-metrics-ribbon',
  148. statsPeriod: '10d',
  149. },
  150. })
  151. );
  152. expect(chartRequestMock).toHaveBeenNthCalledWith(
  153. 1,
  154. `/organizations/${organization.slug}/events-stats/`,
  155. expect.objectContaining({
  156. method: 'GET',
  157. query: {
  158. cursor: undefined,
  159. dataset: 'spansMetrics',
  160. environment: [],
  161. excludeOther: 0,
  162. field: [],
  163. interval: '30m',
  164. orderby: undefined,
  165. partial: 1,
  166. per_page: 50,
  167. project: [],
  168. query:
  169. 'span.module:http span.domain:"\\*.sentry.dev" transaction:/api/0/users',
  170. referrer: 'api.starfish.http-module-samples-panel-response-code-chart',
  171. statsPeriod: '10d',
  172. topEvents: undefined,
  173. yAxis: [
  174. 'http_response_rate(3)',
  175. 'http_response_rate(4)',
  176. 'http_response_rate(5)',
  177. ],
  178. },
  179. })
  180. );
  181. await waitForElementToBeRemoved(() => screen.queryAllByTestId('loading-indicator'));
  182. });
  183. it('shows basic transaction info', async () => {
  184. render(<HTTPSamplesPanel />);
  185. // Panel heading
  186. expect(screen.getByRole('heading', {name: 'GET /api/0/users'})).toBeInTheDocument();
  187. // Metrics ribbon
  188. await waitForElementToBeRemoved(() => screen.queryAllByTestId('loading-indicator'));
  189. expect(
  190. screen.getByRole('heading', {name: 'Requests Per Minute'})
  191. ).toBeInTheDocument();
  192. expect(screen.getByRole('heading', {name: 'Avg Duration'})).toBeInTheDocument();
  193. expect(screen.getByRole('heading', {name: '3XXs'})).toBeInTheDocument();
  194. expect(screen.getByRole('heading', {name: '4XXs'})).toBeInTheDocument();
  195. expect(screen.getByRole('heading', {name: '5XXs'})).toBeInTheDocument();
  196. expect(screen.getByRole('heading', {name: 'Time Spent'})).toBeInTheDocument();
  197. expect(screen.getByText('22.2/min')).toBeInTheDocument();
  198. expect(screen.getByText('140.20ms')).toBeInTheDocument();
  199. expect(screen.getByText('1%')).toBeInTheDocument();
  200. expect(screen.getByText('2.5%')).toBeInTheDocument();
  201. expect(screen.getByText('1.5%')).toBeInTheDocument();
  202. expect(screen.getByText('45.15min')).toBeInTheDocument();
  203. });
  204. });
  205. describe('Duration panel', () => {
  206. let chartRequestMock, samplesRequestMock;
  207. beforeEach(() => {
  208. jest.mocked(useLocation).mockReturnValue({
  209. pathname: '',
  210. search: '',
  211. query: {
  212. domain: '*.sentry.dev',
  213. statsPeriod: '10d',
  214. transaction: '/api/0/users',
  215. transactionMethod: 'GET',
  216. panel: 'duration',
  217. },
  218. hash: '',
  219. state: undefined,
  220. action: 'PUSH',
  221. key: '',
  222. });
  223. chartRequestMock = MockApiClient.addMockResponse({
  224. url: `/organizations/${organization.slug}/events-stats/`,
  225. method: 'GET',
  226. match: [
  227. MockApiClient.matchQuery({
  228. referrer: 'api.starfish.http-module-samples-panel-duration-chart',
  229. }),
  230. ],
  231. body: {data: [[1711393200, [{count: 900}]]]},
  232. });
  233. samplesRequestMock = MockApiClient.addMockResponse({
  234. url: `/api/0/organizations/${organization.slug}/spans-samples/`,
  235. method: 'GET',
  236. body: {
  237. data: [
  238. {
  239. span_id: 'b1bf1acde131623a',
  240. 'span.description':
  241. 'GET https://sentry.io/api/0/organizations/sentry/info/?projectId=1',
  242. project: 'javascript',
  243. timestamp: '2024-03-25T20:31:36+00:00',
  244. 'span.status_code': '200',
  245. 'transaction.id': '11c910c9c10b3ec4ecf8f209b8c6ce48',
  246. 'span.self_time': 320.300102,
  247. },
  248. ],
  249. },
  250. });
  251. });
  252. it('fetches panel data', async () => {
  253. render(<HTTPSamplesPanel />);
  254. await waitForElementToBeRemoved(() => screen.queryAllByTestId('loading-indicator'));
  255. expect(chartRequestMock).toHaveBeenNthCalledWith(
  256. 1,
  257. `/organizations/${organization.slug}/events-stats/`,
  258. expect.objectContaining({
  259. method: 'GET',
  260. query: expect.objectContaining({
  261. dataset: 'spansMetrics',
  262. environment: [],
  263. interval: '30m',
  264. per_page: 50,
  265. project: [],
  266. query:
  267. 'span.module:http span.domain:"\\*.sentry.dev" transaction:/api/0/users',
  268. referrer: 'api.starfish.http-module-samples-panel-duration-chart',
  269. statsPeriod: '10d',
  270. yAxis: 'avg(span.self_time)',
  271. }),
  272. })
  273. );
  274. expect(samplesRequestMock).toHaveBeenNthCalledWith(
  275. 1,
  276. `/api/0/organizations/${organization.slug}/spans-samples/`,
  277. expect.objectContaining({
  278. method: 'GET',
  279. query: expect.objectContaining({
  280. query:
  281. 'span.module:http span.domain:"\\*.sentry.dev" transaction:/api/0/users',
  282. project: [],
  283. additionalFields: ['transaction.id', 'span.description', 'span.status_code'],
  284. lowerBound: 0,
  285. firstBound: expect.closeTo(333.3333),
  286. secondBound: expect.closeTo(666.6666),
  287. upperBound: 1000,
  288. referrer: 'api.starfish.http-module-samples-panel-samples',
  289. statsPeriod: '10d',
  290. }),
  291. })
  292. );
  293. });
  294. it('show basic transaction info', async () => {
  295. render(<HTTPSamplesPanel />);
  296. // Panel heading
  297. expect(screen.getByRole('heading', {name: 'GET /api/0/users'})).toBeInTheDocument();
  298. // Metrics ribbon
  299. await waitForElementToBeRemoved(() => screen.queryAllByTestId('loading-indicator'));
  300. expect(
  301. screen.getByRole('heading', {name: 'Requests Per Minute'})
  302. ).toBeInTheDocument();
  303. expect(screen.getByRole('heading', {name: 'Avg Duration'})).toBeInTheDocument();
  304. expect(screen.getByRole('heading', {name: '3XXs'})).toBeInTheDocument();
  305. expect(screen.getByRole('heading', {name: '4XXs'})).toBeInTheDocument();
  306. expect(screen.getByRole('heading', {name: '5XXs'})).toBeInTheDocument();
  307. expect(screen.getByRole('heading', {name: 'Time Spent'})).toBeInTheDocument();
  308. expect(screen.getByText('22.2/min')).toBeInTheDocument();
  309. expect(screen.getByText('140.20ms')).toBeInTheDocument();
  310. expect(screen.getByText('1%')).toBeInTheDocument();
  311. expect(screen.getByText('2.5%')).toBeInTheDocument();
  312. expect(screen.getByText('1.5%')).toBeInTheDocument();
  313. expect(screen.getByText('45.15min')).toBeInTheDocument();
  314. // Samples table
  315. expect(screen.getByRole('table', {name: 'Span Samples'})).toBeInTheDocument();
  316. expect(screen.getByRole('columnheader', {name: 'Event ID'})).toBeInTheDocument();
  317. expect(screen.getByRole('columnheader', {name: 'Status'})).toBeInTheDocument();
  318. expect(screen.getByRole('columnheader', {name: 'URL'})).toBeInTheDocument();
  319. expect(screen.getByRole('cell', {name: '11c910c9'})).toBeInTheDocument();
  320. expect(screen.getByRole('link', {name: '11c910c9'})).toHaveAttribute(
  321. 'href',
  322. '/organizations/org-slug/performance/javascript:11c910c9c10b3ec4ecf8f209b8c6ce48#span-b1bf1acde131623a'
  323. );
  324. expect(screen.getByRole('cell', {name: '200'})).toBeInTheDocument();
  325. });
  326. it('re-fetches samples', async () => {
  327. render(<HTTPSamplesPanel />);
  328. await waitForElementToBeRemoved(() => screen.queryAllByTestId('loading-indicator'));
  329. expect(samplesRequestMock).toHaveBeenCalledTimes(1);
  330. await userEvent.click(screen.getByRole('button', {name: 'Try Different Samples'}));
  331. expect(samplesRequestMock).toHaveBeenCalledTimes(2);
  332. });
  333. });
  334. });