test_performance_summary.py 7.4 KB

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