test_performance_summary.py 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. from urllib.parse import urlencode
  2. import pytz
  3. from sentry.testutils import AcceptanceTestCase, SnubaTestCase
  4. from sentry.testutils.helpers.datetime import before_now, iso_format
  5. from sentry.utils.compat.mock import patch
  6. from sentry.utils.samples import load_data
  7. from .page_objects.transaction_summary import TransactionSummaryPage
  8. FEATURE_NAMES = ("organizations:performance-view",)
  9. def make_event(event_data):
  10. event_data["event_id"] = "c" * 32
  11. event_data["contexts"]["trace"]["trace_id"] = "a" * 32
  12. return event_data
  13. class PerformanceSummaryTest(AcceptanceTestCase, SnubaTestCase):
  14. def setUp(self):
  15. super().setUp()
  16. self.org = self.create_organization(owner=self.user, name="Rowdy Tiger")
  17. self.team = self.create_team(
  18. organization=self.org, name="Mariachi Band", members=[self.user]
  19. )
  20. self.project = self.create_project(organization=self.org, teams=[self.team], name="Bengal")
  21. self.group = self.create_group(project=self.project)
  22. self.login_as(self.user)
  23. self.path = "/organizations/{}/performance/summary/?{}".format(
  24. self.org.slug,
  25. urlencode({"transaction": "/country_by_code/", "project": self.project.id}),
  26. )
  27. self.page = TransactionSummaryPage(self.browser)
  28. @patch("django.utils.timezone.now")
  29. def test_with_data(self, mock_now):
  30. mock_now.return_value = before_now().replace(tzinfo=pytz.utc)
  31. # Create a transaction
  32. event = make_event(load_data("transaction", timestamp=before_now(minutes=3)))
  33. self.store_event(data=event, project_id=self.project.id)
  34. self.store_event(
  35. data={
  36. "transaction": "/country_by_code/",
  37. "message": "This is bad",
  38. "event_id": "b" * 32,
  39. "timestamp": iso_format(before_now(minutes=1)),
  40. },
  41. project_id=self.project.id,
  42. )
  43. with self.feature(FEATURE_NAMES):
  44. self.browser.get(self.path)
  45. self.page.wait_until_loaded()
  46. # This test is flakey in that we sometimes load this page before the event is processed
  47. # depend on pytest-retry to reload the page
  48. self.browser.wait_until_not(
  49. '[data-test-id="grid-editable"] [data-test-id="empty-state"]', timeout=2
  50. )
  51. # We have to wait for this again because there are loaders inside of the table
  52. self.page.wait_until_loaded()
  53. self.browser.snapshot("performance summary - with data")
  54. @patch("django.utils.timezone.now")
  55. def test_view_details_from_summary(self, mock_now):
  56. mock_now.return_value = before_now().replace(tzinfo=pytz.utc)
  57. event = make_event(
  58. load_data(
  59. "transaction", timestamp=before_now(minutes=3), trace="a" * 32, span_id="ab" * 8
  60. )
  61. )
  62. self.store_event(data=event, project_id=self.project.id)
  63. with self.feature(FEATURE_NAMES):
  64. self.browser.get(self.path)
  65. self.page.wait_until_loaded()
  66. # View the first event details.
  67. self.browser.element('[data-test-id="view-id"]').click()
  68. self.page.wait_until_loaded()
  69. self.browser.snapshot("performance event details")
  70. @patch("django.utils.timezone.now")
  71. def test_tags_page(self, mock_now):
  72. mock_now.return_value = before_now().replace(tzinfo=pytz.utc)
  73. tags_path = "/organizations/{}/performance/summary/tags/?{}".format(
  74. self.org.slug,
  75. urlencode({"transaction": "/country_by_code/", "project": self.project.id}),
  76. )
  77. # Create a transaction
  78. event_data = load_data("transaction", timestamp=before_now(minutes=3))
  79. event = make_event(event_data)
  80. self.store_event(data=event, project_id=self.project.id)
  81. with self.feature(FEATURE_NAMES + ("organizations:performance-tag-page",)):
  82. self.browser.get(tags_path)
  83. self.page.wait_until_loaded()
  84. self.browser.snapshot("transaction summary tags page")
  85. @patch("django.utils.timezone.now")
  86. def test_transaction_vitals(self, mock_now):
  87. mock_now.return_value = before_now().replace(tzinfo=pytz.utc)
  88. vitals_path = "/organizations/{}/performance/summary/vitals/?{}".format(
  89. self.org.slug,
  90. urlencode({"transaction": "/country_by_code/", "project": self.project.id}),
  91. )
  92. # Create a transaction
  93. event_data = load_data("transaction", timestamp=before_now(minutes=3))
  94. # only frontend pageload transactions can be shown on the vitals tab
  95. event_data["contexts"]["trace"]["op"] = "pageload"
  96. event_data["measurements"]["fp"]["value"] = 5000
  97. event = make_event(event_data)
  98. self.store_event(data=event, project_id=self.project.id)
  99. with self.feature(FEATURE_NAMES):
  100. self.browser.get(vitals_path)
  101. self.page.wait_until_loaded()
  102. self.browser.snapshot("real user monitoring")
  103. @patch("django.utils.timezone.now")
  104. def test_transaction_vitals_filtering(self, mock_now):
  105. mock_now.return_value = before_now().replace(tzinfo=pytz.utc)
  106. vitals_path = "/organizations/{}/performance/summary/vitals/?{}".format(
  107. self.org.slug,
  108. urlencode(
  109. {
  110. "transaction": "/country_by_code/",
  111. "project": self.project.id,
  112. "dataFilter": "exclude_outliers",
  113. }
  114. ),
  115. )
  116. # Create transactions
  117. for seconds in range(3):
  118. event_data = load_data("transaction", timestamp=before_now(minutes=3))
  119. event_data["contexts"]["trace"]["op"] = "pageload"
  120. event_data["contexts"]["trace"]["id"] = ("c" * 31) + hex(seconds)[2:]
  121. event_data["event_id"] = ("c" * 31) + hex(seconds)[2:]
  122. event_data["measurements"]["fp"]["value"] = seconds * 10
  123. event_data["measurements"]["fcp"]["value"] = seconds * 10
  124. event_data["measurements"]["lcp"]["value"] = seconds * 10
  125. event_data["measurements"]["fid"]["value"] = seconds * 10
  126. event_data["measurements"]["cls"]["value"] = seconds / 10.0
  127. self.store_event(data=event_data, project_id=self.project.id)
  128. # add anchor point
  129. event_data = load_data("transaction", timestamp=before_now(minutes=2))
  130. event_data["contexts"]["trace"]["op"] = "pageload"
  131. event_data["contexts"]["trace"]["id"] = "a" * 32
  132. event_data["event_id"] = "a" * 32
  133. event_data["measurements"]["fp"]["value"] = 3000
  134. event_data["measurements"]["fcp"]["value"] = 3000
  135. event_data["measurements"]["lcp"]["value"] = 3000
  136. event_data["measurements"]["fid"]["value"] = 3000
  137. event_data["measurements"]["cls"]["value"] = 0.3
  138. self.store_event(data=event_data, project_id=self.project.id)
  139. # add outlier
  140. event_data = load_data("transaction", timestamp=before_now(minutes=2))
  141. event_data["contexts"]["trace"]["op"] = "pageload"
  142. event_data["contexts"]["trace"]["id"] = "b" * 32
  143. event_data["event_id"] = "b" * 32
  144. event_data["measurements"]["fp"]["value"] = 3000000000
  145. event_data["measurements"]["fcp"]["value"] = 3000000000
  146. event_data["measurements"]["lcp"]["value"] = 3000000000
  147. event_data["measurements"]["fid"]["value"] = 3000000000
  148. event_data["measurements"]["cls"]["value"] = 3000000000
  149. self.store_event(data=event_data, project_id=self.project.id)
  150. with self.feature(FEATURE_NAMES):
  151. self.browser.get(vitals_path)
  152. self.page.wait_until_loaded()
  153. self.browser.snapshot("real user monitoring - exclude outliers")
  154. self.browser.element(xpath="//button//span[contains(text(), 'Exclude')]").click()
  155. self.browser.element(xpath="//li//span[contains(text(), 'Include')]").click()
  156. self.page.wait_until_loaded()
  157. self.browser.snapshot("real user monitoring - view all data")
  158. @patch("django.utils.timezone.now")
  159. def test_transaction_threshold_modal(self, mock_now):
  160. mock_now.return_value = before_now().replace(tzinfo=pytz.utc)
  161. # Create a transaction
  162. event = make_event(load_data("transaction", timestamp=before_now(minutes=3)))
  163. self.store_event(data=event, project_id=self.project.id)
  164. self.store_event(
  165. data={
  166. "transaction": "/country_by_code/",
  167. "message": "This is bad",
  168. "event_id": "b" * 32,
  169. "timestamp": iso_format(before_now(minutes=3)),
  170. },
  171. project_id=self.project.id,
  172. )
  173. with self.feature(
  174. (
  175. "organizations:performance-view",
  176. "organizations:project-transaction-threshold-override",
  177. "organizations:project-transaction-threshold",
  178. )
  179. ):
  180. self.browser.get(self.path)
  181. self.page.wait_until_loaded()
  182. # This test is flakey in that we sometimes load this page before the event is processed
  183. # depend on pytest-retry to reload the page
  184. self.browser.wait_until_not(
  185. '[data-test-id="grid-editable"] [data-test-id="empty-state"]', timeout=2
  186. )
  187. # We have to wait for this again because there are loaders inside of the table
  188. self.page.wait_until_loaded()
  189. self.browser.click('[data-test-id="set-transaction-threshold"]')
  190. self.browser.snapshot("transaction threshold modal")