test_performance_summary.py 9.6 KB

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