test_image_block_builder.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. import uuid
  2. from datetime import timedelta
  3. from unittest.mock import patch
  4. import pytest
  5. from django.core.cache import cache
  6. from sentry.integrations.slack.message_builder.image_block_builder import ImageBlockBuilder
  7. from sentry.issues.grouptype import (
  8. PerformanceHTTPOverheadGroupType,
  9. PerformanceP95EndpointRegressionGroupType,
  10. ProfileFunctionRegressionType,
  11. )
  12. from sentry.models.group import Group
  13. from sentry.testutils.cases import (
  14. AcceptanceTestCase,
  15. MetricsEnhancedPerformanceTestCase,
  16. ProfilesSnubaTestCase,
  17. )
  18. from sentry.testutils.helpers.datetime import before_now
  19. from sentry.testutils.helpers.features import with_feature
  20. from tests.sentry.issues.test_utils import OccurrenceTestMixin
  21. pytestmark = pytest.mark.sentry_metrics
  22. class TestSlackImageBlockBuilder(
  23. AcceptanceTestCase,
  24. MetricsEnhancedPerformanceTestCase,
  25. ProfilesSnubaTestCase,
  26. OccurrenceTestMixin,
  27. ):
  28. def setUp(self):
  29. super().setUp()
  30. cache.clear()
  31. def _create_endpoint_regression_issue(self):
  32. for i in range(10):
  33. event_id = uuid.uuid4().hex
  34. _ = self.process_occurrence(
  35. project_id=self.project.id,
  36. event_id=event_id,
  37. type=PerformanceP95EndpointRegressionGroupType.type_id,
  38. event_data={
  39. "fingerprint": ["group-1"],
  40. "timestamp": before_now(minutes=i + 10).isoformat(),
  41. "transaction": "/books/",
  42. },
  43. evidence_data={
  44. "breakpoint": before_now(minutes=i + 10).timestamp(),
  45. },
  46. )
  47. self.store_transaction_metric(
  48. metric="transaction.duration",
  49. tags={"transaction": "/books/"},
  50. value=1,
  51. timestamp=before_now(minutes=i + 10),
  52. project=self.project.id,
  53. )
  54. group = Group.objects.get()
  55. group.update(type=PerformanceP95EndpointRegressionGroupType.type_id)
  56. return group
  57. @with_feature("organizations:performance-use-metrics")
  58. def test_image_block_for_endpoint_regression(self):
  59. group = self._create_endpoint_regression_issue()
  60. image_block = ImageBlockBuilder(group=group).build_image_block()
  61. assert image_block and "type" in image_block and image_block["type"] == "image"
  62. assert "_media/" in image_block["image_url"]
  63. @with_feature("organizations:performance-use-metrics")
  64. @patch("sentry.utils.performance_issues.detectors.utils.escape_transaction")
  65. def test_caching(self, mock_escape_transaction):
  66. mock_escape_transaction.return_value = "Test Transaction"
  67. group = self._create_endpoint_regression_issue()
  68. image_blocks = []
  69. for _ in range(5):
  70. image_blocks.append(ImageBlockBuilder(group=group).build_image_block())
  71. assert mock_escape_transaction.call_count == 1
  72. assert len(image_blocks) == 5
  73. assert image_blocks[0]
  74. image_url = image_blocks[0]["image_url"]
  75. for image_block in image_blocks:
  76. assert image_block is not None
  77. assert image_block["image_url"] == image_url
  78. @with_feature("organizations:performance-use-metrics")
  79. def test_image_block_for_function_regression(self):
  80. hour_ago = (before_now(minutes=10) - timedelta(hours=1)).replace(
  81. minute=0, second=0, microsecond=0
  82. )
  83. for i in range(10):
  84. event_id = uuid.uuid4().hex
  85. _ = self.process_occurrence(
  86. project_id=self.project.id,
  87. event_id=event_id,
  88. type=ProfileFunctionRegressionType.type_id,
  89. event_data={
  90. "fingerprint": ["group-1"],
  91. "timestamp": before_now(minutes=i + 10).isoformat(),
  92. "function": "foo",
  93. },
  94. evidence_data={
  95. "breakpoint": before_now(minutes=i + 10).timestamp(),
  96. "fingerprint": self.function_fingerprint({"package": "foo", "function": "foo"}),
  97. "aggregate_range_1": 51588984.199999996,
  98. "aggregate_range_2": 839118611.8535699,
  99. },
  100. )
  101. self.store_functions(
  102. [
  103. {
  104. "self_times_ns": [100 for _ in range(100)],
  105. "package": "foo",
  106. "function": "foo",
  107. # only in app functions should
  108. # appear in the results
  109. "in_app": True,
  110. },
  111. ],
  112. project=self.project,
  113. timestamp=hour_ago,
  114. )
  115. group = Group.objects.get()
  116. image_block = ImageBlockBuilder(group=group).build_image_block()
  117. assert image_block and "type" in image_block and image_block["type"] == "image"
  118. assert "_media/" in image_block["image_url"]
  119. @patch("sentry_sdk.capture_exception")
  120. def test_image_not_generated_for_unsupported_issues(self, mock_capture_exception):
  121. group = self.create_group()
  122. group.update(type=PerformanceHTTPOverheadGroupType.type_id)
  123. image_block = ImageBlockBuilder(group=group).build_image_block()
  124. assert image_block is None
  125. assert mock_capture_exception.call_count == 0