test_organization_events_span_indexed.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. from tests.snuba.api.endpoints.test_organization_events import OrganizationEventsEndpointTestBase
  2. class OrganizationEventsSpanIndexedEndpointTest(OrganizationEventsEndpointTestBase):
  3. """Test the indexed spans dataset.
  4. To run this locally you may need to set the ENABLE_SPANS_CONSUMER flag to True in Snuba.
  5. A way to do this is
  6. 1. run: `sentry devservices down snuba`
  7. 2. clone snuba locally
  8. 3. run: `export ENABLE_SPANS_CONSUMER=True`
  9. 4. run snuba
  10. At this point tests should work locally
  11. Once span ingestion is on by default this will no longer need to be done
  12. """
  13. def setUp(self):
  14. super().setUp()
  15. self.features = {
  16. "organizations:starfish-view": True,
  17. }
  18. def test_simple(self):
  19. self.store_spans(
  20. [
  21. self.create_span({"description": "foo"}, start_ts=self.ten_mins_ago),
  22. self.create_span({"description": "bar"}, start_ts=self.ten_mins_ago),
  23. ]
  24. )
  25. response = self.do_request(
  26. {
  27. "field": ["description", "count()"],
  28. "query": "",
  29. "orderby": "description",
  30. "project": self.project.id,
  31. "dataset": "spansIndexed",
  32. }
  33. )
  34. assert response.status_code == 200, response.content
  35. data = response.data["data"]
  36. meta = response.data["meta"]
  37. assert len(data) == 2
  38. assert data[0]["description"] == "bar"
  39. assert data[1]["description"] == "foo"
  40. assert meta["dataset"] == "spansIndexed"
  41. def test_sentry_tags_vs_tags(self):
  42. self.store_spans(
  43. [
  44. self.create_span(
  45. {"sentry_tags": {"transaction.method": "foo"}}, start_ts=self.ten_mins_ago
  46. ),
  47. ]
  48. )
  49. response = self.do_request(
  50. {
  51. "field": ["transaction.method", "count()"],
  52. "query": "",
  53. "orderby": "count()",
  54. "project": self.project.id,
  55. "dataset": "spansIndexed",
  56. }
  57. )
  58. assert response.status_code == 200, response.content
  59. data = response.data["data"]
  60. meta = response.data["meta"]
  61. assert len(data) == 1
  62. assert data[0]["transaction.method"] == "foo"
  63. assert meta["dataset"] == "spansIndexed"
  64. def test_sentry_tags_syntax(self):
  65. self.store_spans(
  66. [
  67. self.create_span(
  68. {"sentry_tags": {"transaction.method": "foo"}}, start_ts=self.ten_mins_ago
  69. ),
  70. ]
  71. )
  72. response = self.do_request(
  73. {
  74. "field": ["sentry_tags[transaction.method]", "count()"],
  75. "query": "",
  76. "orderby": "count()",
  77. "project": self.project.id,
  78. "dataset": "spansIndexed",
  79. }
  80. )
  81. assert response.status_code == 200, response.content
  82. data = response.data["data"]
  83. meta = response.data["meta"]
  84. assert len(data) == 1
  85. assert data[0]["sentry_tags[transaction.method]"] == "foo"
  86. assert meta["dataset"] == "spansIndexed"
  87. def test_module_alias(self):
  88. # Delegates `span.module` to `sentry_tags[category]`. Maps `"db.redis"` spans to the `"cache"` module
  89. self.store_spans(
  90. [
  91. self.create_span(
  92. {
  93. "op": "db.redis",
  94. "description": "EXEC *",
  95. "sentry_tags": {
  96. "description": "EXEC *",
  97. "category": "db",
  98. "op": "db.redis",
  99. "transaction": "/app/index",
  100. },
  101. },
  102. start_ts=self.ten_mins_ago,
  103. ),
  104. ]
  105. )
  106. response = self.do_request(
  107. {
  108. "field": ["span.module", "span.description"],
  109. "query": "span.module:cache",
  110. "project": self.project.id,
  111. "dataset": "spansIndexed",
  112. }
  113. )
  114. assert response.status_code == 200, response.content
  115. data = response.data["data"]
  116. meta = response.data["meta"]
  117. assert len(data) == 1
  118. assert data[0]["span.module"] == "cache"
  119. assert data[0]["span.description"] == "EXEC *"
  120. assert meta["dataset"] == "spansIndexed"
  121. def test_device_class_filter_unknown(self):
  122. self.store_spans(
  123. [
  124. self.create_span({"sentry_tags": {"device.class": ""}}, start_ts=self.ten_mins_ago),
  125. ]
  126. )
  127. response = self.do_request(
  128. {
  129. "field": ["device.class", "count()"],
  130. "query": "device.class:Unknown",
  131. "orderby": "count()",
  132. "project": self.project.id,
  133. "dataset": "spansIndexed",
  134. }
  135. )
  136. assert response.status_code == 200, response.content
  137. data = response.data["data"]
  138. meta = response.data["meta"]
  139. assert len(data) == 1
  140. assert data[0]["device.class"] == "Unknown"
  141. assert meta["dataset"] == "spansIndexed"
  142. def test_network_span(self):
  143. self.store_spans(
  144. [
  145. self.create_span(
  146. {
  147. "sentry_tags": {
  148. "action": "GET",
  149. "category": "http",
  150. "description": "GET https://*.resource.com",
  151. "domain": "*.resource.com",
  152. "op": "http.client",
  153. "status_code": "200",
  154. "transaction": "/api/0/data/",
  155. "transaction.method": "GET",
  156. "transaction.op": "http.server",
  157. }
  158. },
  159. start_ts=self.ten_mins_ago,
  160. ),
  161. ]
  162. )
  163. response = self.do_request(
  164. {
  165. "field": ["span.op", "span.status_code"],
  166. "query": "span.module:http span.status_code:200",
  167. "project": self.project.id,
  168. "dataset": "spansIndexed",
  169. }
  170. )
  171. assert response.status_code == 200, response.content
  172. data = response.data["data"]
  173. meta = response.data["meta"]
  174. assert len(data) == 1
  175. assert data[0]["span.op"] == "http.client"
  176. assert data[0]["span.status_code"] == "200"
  177. assert meta["dataset"] == "spansIndexed"
  178. def test_inp_span(self):
  179. self.store_spans(
  180. [
  181. self.create_span(
  182. {
  183. "sentry_tags": {
  184. "replay_id": "abc123",
  185. "browser.name": "Chrome",
  186. "transaction": "/pageloads/",
  187. }
  188. },
  189. start_ts=self.ten_mins_ago,
  190. ),
  191. ]
  192. )
  193. response = self.do_request(
  194. {
  195. "field": ["replay.id", "browser.name", "origin.transaction", "count()"],
  196. "query": "replay.id:abc123 AND browser.name:Chrome AND origin.transaction:/pageloads/",
  197. "orderby": "count()",
  198. "project": self.project.id,
  199. "dataset": "spansIndexed",
  200. }
  201. )
  202. assert response.status_code == 200, response.content
  203. data = response.data["data"]
  204. meta = response.data["meta"]
  205. assert len(data) == 1
  206. assert data[0]["replay.id"] == "abc123"
  207. assert data[0]["browser.name"] == "Chrome"
  208. assert data[0]["origin.transaction"] == "/pageloads/"
  209. assert meta["dataset"] == "spansIndexed"
  210. def test_span_op_casing(self):
  211. self.store_spans(
  212. [
  213. self.create_span(
  214. {
  215. "sentry_tags": {
  216. "replay_id": "abc123",
  217. "browser.name": "Chrome",
  218. "transaction": "/pageloads/",
  219. "op": "this is a transaction",
  220. }
  221. },
  222. start_ts=self.ten_mins_ago,
  223. ),
  224. ]
  225. )
  226. response = self.do_request(
  227. {
  228. "field": ["span.op", "count()"],
  229. "query": 'span.op:"ThIs Is a TraNSActiON"',
  230. "orderby": "count()",
  231. "project": self.project.id,
  232. "dataset": "spansIndexed",
  233. }
  234. )
  235. assert response.status_code == 200, response.content
  236. data = response.data["data"]
  237. meta = response.data["meta"]
  238. assert len(data) == 1
  239. assert data[0]["span.op"] == "this is a transaction"
  240. assert meta["dataset"] == "spansIndexed"