test_organization_events_span_indexed.py 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551
  1. import uuid
  2. from unittest import mock
  3. import pytest
  4. from tests.snuba.api.endpoints.test_organization_events import OrganizationEventsEndpointTestBase
  5. class OrganizationEventsSpanIndexedEndpointTest(OrganizationEventsEndpointTestBase):
  6. is_eap = False
  7. use_rpc = False
  8. """Test the indexed spans dataset.
  9. To run this locally you may need to set the ENABLE_SPANS_CONSUMER flag to True in Snuba.
  10. A way to do this is
  11. 1. run: `sentry devservices down snuba`
  12. 2. clone snuba locally
  13. 3. run: `export ENABLE_SPANS_CONSUMER=True`
  14. 4. run snuba
  15. At this point tests should work locally
  16. Once span ingestion is on by default this will no longer need to be done
  17. """
  18. @property
  19. def dataset(self):
  20. if self.is_eap:
  21. return "spans"
  22. else:
  23. return "spansIndexed"
  24. def do_request(self, query, features=None, **kwargs):
  25. query["useRpc"] = "1" if self.use_rpc else "0"
  26. return super().do_request(query, features, **kwargs)
  27. def setUp(self):
  28. super().setUp()
  29. self.features = {
  30. "organizations:starfish-view": True,
  31. }
  32. @pytest.mark.querybuilder
  33. def test_simple(self):
  34. self.store_spans(
  35. [
  36. self.create_span(
  37. {"description": "foo", "sentry_tags": {"status": "success"}},
  38. start_ts=self.ten_mins_ago,
  39. ),
  40. self.create_span(
  41. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  42. start_ts=self.ten_mins_ago,
  43. ),
  44. ],
  45. is_eap=self.is_eap,
  46. )
  47. response = self.do_request(
  48. {
  49. "field": ["span.status", "description", "count()"],
  50. "query": "",
  51. "orderby": "description",
  52. "project": self.project.id,
  53. "dataset": self.dataset,
  54. }
  55. )
  56. assert response.status_code == 200, response.content
  57. data = response.data["data"]
  58. meta = response.data["meta"]
  59. assert len(data) == 2
  60. assert data == [
  61. {
  62. "span.status": "invalid_argument",
  63. "description": "bar",
  64. "count()": 1,
  65. },
  66. {
  67. "span.status": "ok",
  68. "description": "foo",
  69. "count()": 1,
  70. },
  71. ]
  72. assert meta["dataset"] == self.dataset
  73. def test_spm(self):
  74. self.store_spans(
  75. [
  76. self.create_span(
  77. {"description": "foo", "sentry_tags": {"status": "success"}},
  78. start_ts=self.ten_mins_ago,
  79. ),
  80. ],
  81. is_eap=self.is_eap,
  82. )
  83. response = self.do_request(
  84. {
  85. "field": ["description", "spm()"],
  86. "query": "",
  87. "orderby": "description",
  88. "project": self.project.id,
  89. "dataset": self.dataset,
  90. }
  91. )
  92. assert response.status_code == 200, response.content
  93. data = response.data["data"]
  94. meta = response.data["meta"]
  95. assert len(data) == 1
  96. assert data == [
  97. {
  98. "description": "foo",
  99. "spm()": 1 / (90 * 24 * 60),
  100. },
  101. ]
  102. assert meta["dataset"] == self.dataset
  103. def test_id_fields(self):
  104. self.store_spans(
  105. [
  106. self.create_span(
  107. {"description": "foo", "sentry_tags": {"status": "success"}},
  108. start_ts=self.ten_mins_ago,
  109. ),
  110. self.create_span(
  111. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  112. start_ts=self.ten_mins_ago,
  113. ),
  114. ],
  115. is_eap=self.is_eap,
  116. )
  117. response = self.do_request(
  118. {
  119. "field": ["id", "span_id"],
  120. "query": "",
  121. "orderby": "id",
  122. "project": self.project.id,
  123. "dataset": self.dataset,
  124. }
  125. )
  126. assert response.status_code == 200, response.content
  127. data = response.data["data"]
  128. meta = response.data["meta"]
  129. assert len(data) == 2
  130. for obj in data:
  131. assert obj["id"] == obj["span_id"]
  132. assert meta["dataset"] == self.dataset
  133. def test_sentry_tags_vs_tags(self):
  134. self.store_spans(
  135. [
  136. self.create_span(
  137. {"sentry_tags": {"transaction.method": "foo"}}, start_ts=self.ten_mins_ago
  138. ),
  139. ],
  140. is_eap=self.is_eap,
  141. )
  142. response = self.do_request(
  143. {
  144. "field": ["transaction.method", "count()"],
  145. "query": "",
  146. "orderby": "count()",
  147. "project": self.project.id,
  148. "dataset": self.dataset,
  149. }
  150. )
  151. assert response.status_code == 200, response.content
  152. data = response.data["data"]
  153. meta = response.data["meta"]
  154. assert len(data) == 1
  155. assert data[0]["transaction.method"] == "foo"
  156. assert meta["dataset"] == self.dataset
  157. def test_sentry_tags_syntax(self):
  158. self.store_spans(
  159. [
  160. self.create_span(
  161. {"sentry_tags": {"transaction.method": "foo"}}, start_ts=self.ten_mins_ago
  162. ),
  163. ],
  164. is_eap=self.is_eap,
  165. )
  166. response = self.do_request(
  167. {
  168. "field": ["sentry_tags[transaction.method]", "count()"],
  169. "query": "",
  170. "orderby": "count()",
  171. "project": self.project.id,
  172. "dataset": self.dataset,
  173. }
  174. )
  175. assert response.status_code == 200, response.content
  176. data = response.data["data"]
  177. meta = response.data["meta"]
  178. assert len(data) == 1
  179. assert data[0]["sentry_tags[transaction.method]"] == "foo"
  180. assert meta["dataset"] == self.dataset
  181. def test_module_alias(self):
  182. # Delegates `span.module` to `sentry_tags[category]`. Maps `"db.redis"` spans to the `"cache"` module
  183. self.store_spans(
  184. [
  185. self.create_span(
  186. {
  187. "op": "db.redis",
  188. "description": "EXEC *",
  189. "sentry_tags": {
  190. "description": "EXEC *",
  191. "category": "db",
  192. "op": "db.redis",
  193. "transaction": "/app/index",
  194. },
  195. },
  196. start_ts=self.ten_mins_ago,
  197. ),
  198. ],
  199. is_eap=self.is_eap,
  200. )
  201. response = self.do_request(
  202. {
  203. "field": ["span.module", "span.description"],
  204. "query": "span.module:cache",
  205. "project": self.project.id,
  206. "dataset": self.dataset,
  207. }
  208. )
  209. assert response.status_code == 200, response.content
  210. data = response.data["data"]
  211. meta = response.data["meta"]
  212. assert len(data) == 1
  213. assert data[0]["span.module"] == "cache"
  214. assert data[0]["span.description"] == "EXEC *"
  215. assert meta["dataset"] == self.dataset
  216. def test_device_class_filter_unknown(self):
  217. self.store_spans(
  218. [
  219. self.create_span({"sentry_tags": {"device.class": ""}}, start_ts=self.ten_mins_ago),
  220. ],
  221. is_eap=self.is_eap,
  222. )
  223. response = self.do_request(
  224. {
  225. "field": ["device.class", "count()"],
  226. "query": "device.class:Unknown",
  227. "orderby": "count()",
  228. "project": self.project.id,
  229. "dataset": self.dataset,
  230. }
  231. )
  232. assert response.status_code == 200, response.content
  233. data = response.data["data"]
  234. meta = response.data["meta"]
  235. assert len(data) == 1
  236. assert data[0]["device.class"] == "Unknown"
  237. assert meta["dataset"] == self.dataset
  238. def test_network_span(self):
  239. self.store_spans(
  240. [
  241. self.create_span(
  242. {
  243. "sentry_tags": {
  244. "action": "GET",
  245. "category": "http",
  246. "description": "GET https://*.resource.com",
  247. "domain": "*.resource.com",
  248. "op": "http.client",
  249. "status_code": "200",
  250. "transaction": "/api/0/data/",
  251. "transaction.method": "GET",
  252. "transaction.op": "http.server",
  253. }
  254. },
  255. start_ts=self.ten_mins_ago,
  256. ),
  257. ],
  258. is_eap=self.is_eap,
  259. )
  260. response = self.do_request(
  261. {
  262. "field": ["span.op", "span.status_code"],
  263. "query": "span.module:http span.status_code:200",
  264. "project": self.project.id,
  265. "dataset": self.dataset,
  266. }
  267. )
  268. assert response.status_code == 200, response.content
  269. data = response.data["data"]
  270. meta = response.data["meta"]
  271. assert len(data) == 1
  272. assert data[0]["span.op"] == "http.client"
  273. assert data[0]["span.status_code"] == "200"
  274. assert meta["dataset"] == self.dataset
  275. def test_other_category_span(self):
  276. self.store_spans(
  277. [
  278. self.create_span(
  279. {
  280. "sentry_tags": {
  281. "action": "GET",
  282. "category": "alternative",
  283. "description": "GET https://*.resource.com",
  284. "domain": "*.resource.com",
  285. "op": "alternative",
  286. "status_code": "200",
  287. "transaction": "/api/0/data/",
  288. "transaction.method": "GET",
  289. "transaction.op": "http.server",
  290. }
  291. },
  292. start_ts=self.ten_mins_ago,
  293. ),
  294. ],
  295. is_eap=self.is_eap,
  296. )
  297. response = self.do_request(
  298. {
  299. "field": ["span.op", "span.status_code"],
  300. "query": "span.module:other span.status_code:200",
  301. "project": self.project.id,
  302. "dataset": self.dataset,
  303. }
  304. )
  305. assert response.status_code == 200, response.content
  306. data = response.data["data"]
  307. meta = response.data["meta"]
  308. assert len(data) == 1
  309. assert data[0]["span.op"] == "alternative"
  310. assert data[0]["span.status_code"] == "200"
  311. assert meta["dataset"] == self.dataset
  312. def test_inp_span(self):
  313. replay_id = uuid.uuid4().hex
  314. self.store_spans(
  315. [
  316. self.create_span(
  317. {
  318. "sentry_tags": {
  319. "replay_id": replay_id,
  320. "browser.name": "Chrome",
  321. "transaction": "/pageloads/",
  322. }
  323. },
  324. start_ts=self.ten_mins_ago,
  325. ),
  326. ],
  327. is_eap=self.is_eap,
  328. )
  329. response = self.do_request(
  330. {
  331. "field": ["replay.id", "browser.name", "origin.transaction", "count()"],
  332. "query": f"replay.id:{replay_id} AND browser.name:Chrome AND origin.transaction:/pageloads/",
  333. "orderby": "count()",
  334. "project": self.project.id,
  335. "dataset": self.dataset,
  336. }
  337. )
  338. assert response.status_code == 200, response.content
  339. data = response.data["data"]
  340. meta = response.data["meta"]
  341. assert len(data) == 1
  342. assert data[0]["replay.id"] == replay_id
  343. assert data[0]["browser.name"] == "Chrome"
  344. assert data[0]["origin.transaction"] == "/pageloads/"
  345. assert meta["dataset"] == self.dataset
  346. def test_id_filtering(self):
  347. span = self.create_span({"description": "foo"}, start_ts=self.ten_mins_ago)
  348. self.store_span(span, is_eap=self.is_eap)
  349. response = self.do_request(
  350. {
  351. "field": ["description", "count()"],
  352. "query": f"id:{span['span_id']}",
  353. "orderby": "description",
  354. "project": self.project.id,
  355. "dataset": self.dataset,
  356. }
  357. )
  358. assert response.status_code == 200, response.content
  359. data = response.data["data"]
  360. meta = response.data["meta"]
  361. assert len(data) == 1
  362. assert data[0]["description"] == "foo"
  363. assert meta["dataset"] == self.dataset
  364. response = self.do_request(
  365. {
  366. "field": ["description", "count()"],
  367. "query": f"transaction.id:{span['event_id']}",
  368. "orderby": "description",
  369. "project": self.project.id,
  370. "dataset": self.dataset,
  371. }
  372. )
  373. assert response.status_code == 200, response.content
  374. data = response.data["data"]
  375. meta = response.data["meta"]
  376. assert len(data) == 1
  377. assert data[0]["description"] == "foo"
  378. assert meta["dataset"] == self.dataset
  379. def test_span_op_casing(self):
  380. self.store_spans(
  381. [
  382. self.create_span(
  383. {
  384. "sentry_tags": {
  385. "replay_id": "abc123",
  386. "browser.name": "Chrome",
  387. "transaction": "/pageloads/",
  388. "op": "this is a transaction",
  389. }
  390. },
  391. start_ts=self.ten_mins_ago,
  392. ),
  393. ],
  394. is_eap=self.is_eap,
  395. )
  396. response = self.do_request(
  397. {
  398. "field": ["span.op", "count()"],
  399. "query": 'span.op:"ThIs Is a TraNSActiON"',
  400. "orderby": "count()",
  401. "project": self.project.id,
  402. "dataset": self.dataset,
  403. }
  404. )
  405. assert response.status_code == 200, response.content
  406. data = response.data["data"]
  407. meta = response.data["meta"]
  408. assert len(data) == 1
  409. assert data[0]["span.op"] == "this is a transaction"
  410. assert meta["dataset"] == self.dataset
  411. def test_queue_span(self):
  412. self.store_spans(
  413. [
  414. self.create_span(
  415. {
  416. "measurements": {
  417. "messaging.message.body.size": {"value": 1024, "unit": "byte"},
  418. "messaging.message.receive.latency": {
  419. "value": 1000,
  420. "unit": "millisecond",
  421. },
  422. "messaging.message.retry.count": {"value": 2, "unit": "none"},
  423. },
  424. "sentry_tags": {
  425. "transaction": "queue-processor",
  426. "messaging.destination.name": "events",
  427. "messaging.message.id": "abc123",
  428. "trace.status": "ok",
  429. },
  430. },
  431. start_ts=self.ten_mins_ago,
  432. ),
  433. ],
  434. is_eap=self.is_eap,
  435. )
  436. response = self.do_request(
  437. {
  438. "field": [
  439. "transaction",
  440. "messaging.destination.name",
  441. "messaging.message.id",
  442. "measurements.messaging.message.receive.latency",
  443. "measurements.messaging.message.body.size",
  444. "measurements.messaging.message.retry.count",
  445. "trace.status",
  446. "count()",
  447. ],
  448. "query": 'messaging.destination.name:"events"',
  449. "orderby": "count()",
  450. "project": self.project.id,
  451. "dataset": self.dataset,
  452. }
  453. )
  454. assert response.status_code == 200, response.content
  455. data = response.data["data"]
  456. meta = response.data["meta"]
  457. assert len(data) == 1
  458. assert data[0]["transaction"] == "queue-processor"
  459. assert data[0]["messaging.destination.name"] == "events"
  460. assert data[0]["messaging.message.id"] == "abc123"
  461. assert data[0]["trace.status"] == "ok"
  462. assert data[0]["measurements.messaging.message.receive.latency"] == 1000
  463. assert data[0]["measurements.messaging.message.body.size"] == 1024
  464. assert data[0]["measurements.messaging.message.retry.count"] == 2
  465. assert meta["dataset"] == self.dataset
  466. def test_tag_wildcards(self):
  467. self.store_spans(
  468. [
  469. self.create_span(
  470. {"description": "foo", "tags": {"foo": "BaR"}},
  471. start_ts=self.ten_mins_ago,
  472. ),
  473. self.create_span(
  474. {"description": "qux", "tags": {"foo": "QuX"}},
  475. start_ts=self.ten_mins_ago,
  476. ),
  477. ],
  478. is_eap=self.is_eap,
  479. )
  480. for query in [
  481. "foo:b*",
  482. "foo:*r",
  483. "foo:*a*",
  484. "foo:b*r",
  485. ]:
  486. response = self.do_request(
  487. {
  488. "field": ["foo", "count()"],
  489. "query": query,
  490. "project": self.project.id,
  491. "dataset": self.dataset,
  492. }
  493. )
  494. assert response.status_code == 200, response.content
  495. assert response.data["data"] == [{"foo": "BaR", "count()": 1}]
  496. def test_query_for_missing_tag(self):
  497. self.store_spans(
  498. [
  499. self.create_span(
  500. {"description": "foo"},
  501. start_ts=self.ten_mins_ago,
  502. ),
  503. self.create_span(
  504. {"description": "qux", "tags": {"foo": "bar"}},
  505. start_ts=self.ten_mins_ago,
  506. ),
  507. ],
  508. is_eap=self.is_eap,
  509. )
  510. response = self.do_request(
  511. {
  512. "field": ["foo", "count()"],
  513. "query": 'foo:""',
  514. "project": self.project.id,
  515. "dataset": self.dataset,
  516. }
  517. )
  518. assert response.status_code == 200, response.content
  519. assert response.data["data"] == [{"foo": "", "count()": 1}]
  520. def test_count_field_type(self):
  521. response = self.do_request(
  522. {
  523. "field": ["count()"],
  524. "project": self.project.id,
  525. "dataset": self.dataset,
  526. }
  527. )
  528. assert response.status_code == 200, response.content
  529. assert response.data["meta"]["fields"] == {"count()": "integer"}
  530. assert response.data["meta"]["units"] == {"count()": None}
  531. assert response.data["data"] == [{"count()": 0}]
  532. def test_simple_measurements(self):
  533. keys = [
  534. ("app_start_cold", "duration", "millisecond"),
  535. ("app_start_warm", "duration", "millisecond"),
  536. ("frames_frozen", "number", None),
  537. ("frames_frozen_rate", "percentage", None),
  538. ("frames_slow", "number", None),
  539. ("frames_slow_rate", "percentage", None),
  540. ("frames_total", "number", None),
  541. ("time_to_initial_display", "duration", "millisecond"),
  542. ("time_to_full_display", "duration", "millisecond"),
  543. ("stall_count", "number", None),
  544. ("stall_percentage", "percentage", None),
  545. ("stall_stall_longest_time", "number", None),
  546. ("stall_stall_total_time", "number", None),
  547. ("cls", "number", None),
  548. ("fcp", "duration", "millisecond"),
  549. ("fid", "duration", "millisecond"),
  550. ("fp", "duration", "millisecond"),
  551. ("inp", "duration", "millisecond"),
  552. ("lcp", "duration", "millisecond"),
  553. ("ttfb", "duration", "millisecond"),
  554. ("ttfb.requesttime", "duration", "millisecond"),
  555. ("score.cls", "number", None),
  556. ("score.fcp", "number", None),
  557. ("score.fid", "number", None),
  558. ("score.inp", "number", None),
  559. ("score.lcp", "number", None),
  560. ("score.ttfb", "number", None),
  561. ("score.total", "number", None),
  562. ("score.weight.cls", "number", None),
  563. ("score.weight.fcp", "number", None),
  564. ("score.weight.fid", "number", None),
  565. ("score.weight.inp", "number", None),
  566. ("score.weight.lcp", "number", None),
  567. ("score.weight.ttfb", "number", None),
  568. ("cache.item_size", "number", None),
  569. ("messaging.message.body.size", "number", None),
  570. ("messaging.message.receive.latency", "number", None),
  571. ("messaging.message.retry.count", "number", None),
  572. ("http.response_content_length", "number", None),
  573. ]
  574. self.store_spans(
  575. [
  576. self.create_span(
  577. {
  578. "description": "foo",
  579. "sentry_tags": {"status": "success"},
  580. "tags": {"bar": "bar2"},
  581. },
  582. measurements={k: {"value": i + 1} for i, (k, _, _) in enumerate(keys)},
  583. start_ts=self.ten_mins_ago,
  584. ),
  585. ],
  586. is_eap=self.is_eap,
  587. )
  588. for i, (k, type, unit) in enumerate(keys):
  589. key = f"measurements.{k}"
  590. response = self.do_request(
  591. {
  592. "field": [key],
  593. "query": "description:foo",
  594. "project": self.project.id,
  595. "dataset": self.dataset,
  596. }
  597. )
  598. assert response.status_code == 200, response.content
  599. assert response.data["meta"] == {
  600. "dataset": mock.ANY,
  601. "datasetReason": "unchanged",
  602. "fields": {
  603. key: type,
  604. "id": "string",
  605. "project.name": "string",
  606. },
  607. "isMetricsData": False,
  608. "isMetricsExtractedData": False,
  609. "tips": {},
  610. "units": {
  611. key: unit,
  612. "id": None,
  613. "project.name": None,
  614. },
  615. }
  616. assert response.data["data"] == [
  617. {
  618. key: i + 1,
  619. "id": mock.ANY,
  620. "project.name": self.project.slug,
  621. }
  622. ]
  623. def test_environment(self):
  624. self.create_environment(self.project, name="prod")
  625. self.create_environment(self.project, name="test")
  626. self.store_spans(
  627. [
  628. self.create_span(
  629. {"description": "foo", "sentry_tags": {"environment": "prod"}},
  630. start_ts=self.ten_mins_ago,
  631. ),
  632. self.create_span(
  633. {"description": "foo", "sentry_tags": {"environment": "test"}},
  634. start_ts=self.ten_mins_ago,
  635. ),
  636. ],
  637. is_eap=self.is_eap,
  638. )
  639. response = self.do_request(
  640. {
  641. "field": ["environment", "count()"],
  642. "project": self.project.id,
  643. "environment": "prod",
  644. "dataset": self.dataset,
  645. }
  646. )
  647. assert response.status_code == 200, response.content
  648. assert response.data["data"] == [
  649. {"environment": "prod", "count()": 1},
  650. ]
  651. def test_transaction(self):
  652. self.store_spans(
  653. [
  654. self.create_span(
  655. {"description": "foo", "sentry_tags": {"transaction": "bar"}},
  656. start_ts=self.ten_mins_ago,
  657. ),
  658. ],
  659. is_eap=self.is_eap,
  660. )
  661. response = self.do_request(
  662. {
  663. "field": ["description", "count()"],
  664. "query": "transaction:bar",
  665. "orderby": "description",
  666. "project": self.project.id,
  667. "dataset": self.dataset,
  668. }
  669. )
  670. assert response.status_code == 200, response.content
  671. data = response.data["data"]
  672. meta = response.data["meta"]
  673. assert len(data) == 1
  674. assert data == [
  675. {
  676. "description": "foo",
  677. "count()": 1,
  678. },
  679. ]
  680. assert meta["dataset"] == self.dataset
  681. def test_orderby_alias(self):
  682. self.store_spans(
  683. [
  684. self.create_span(
  685. {"description": "foo", "sentry_tags": {"status": "success"}},
  686. start_ts=self.ten_mins_ago,
  687. ),
  688. self.create_span(
  689. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  690. duration=2000,
  691. start_ts=self.ten_mins_ago,
  692. ),
  693. ],
  694. is_eap=self.is_eap,
  695. )
  696. response = self.do_request(
  697. {
  698. "field": ["span.description", "sum(span.self_time)"],
  699. "query": "",
  700. "orderby": "sum_span_self_time",
  701. "project": self.project.id,
  702. "dataset": self.dataset,
  703. }
  704. )
  705. assert response.status_code == 200, response.content
  706. data = response.data["data"]
  707. meta = response.data["meta"]
  708. assert len(data) == 2
  709. assert data == [
  710. {
  711. "span.description": "foo",
  712. "sum(span.self_time)": 1000,
  713. },
  714. {
  715. "span.description": "bar",
  716. "sum(span.self_time)": 2000,
  717. },
  718. ]
  719. assert meta["dataset"] == self.dataset
  720. def test_span_status(self):
  721. self.store_spans(
  722. [
  723. self.create_span(
  724. {"description": "foo", "sentry_tags": {"status": "internal_error"}},
  725. start_ts=self.ten_mins_ago,
  726. ),
  727. ],
  728. is_eap=self.is_eap,
  729. )
  730. response = self.do_request(
  731. {
  732. "field": ["description", "count()"],
  733. "query": "span.status:internal_error",
  734. "orderby": "description",
  735. "project": self.project.id,
  736. "dataset": self.dataset,
  737. }
  738. )
  739. assert response.status_code == 200, response.content
  740. data = response.data["data"]
  741. meta = response.data["meta"]
  742. assert len(data) == 1
  743. assert data == [
  744. {
  745. "description": "foo",
  746. "count()": 1,
  747. },
  748. ]
  749. assert meta["dataset"] == self.dataset
  750. class OrganizationEventsEAPSpanEndpointTest(OrganizationEventsSpanIndexedEndpointTest):
  751. is_eap = True
  752. use_rpc = False
  753. def test_simple(self):
  754. self.store_spans(
  755. [
  756. self.create_span(
  757. {"description": "foo", "sentry_tags": {"status": "success"}},
  758. start_ts=self.ten_mins_ago,
  759. ),
  760. self.create_span(
  761. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  762. start_ts=self.ten_mins_ago,
  763. ),
  764. ],
  765. is_eap=self.is_eap,
  766. )
  767. response = self.do_request(
  768. {
  769. "field": ["span.status", "description", "count()"],
  770. "query": "",
  771. "orderby": "description",
  772. "project": self.project.id,
  773. "dataset": self.dataset,
  774. }
  775. )
  776. assert response.status_code == 200, response.content
  777. data = response.data["data"]
  778. meta = response.data["meta"]
  779. assert len(data) == 2
  780. assert data == [
  781. {
  782. "span.status": "invalid_argument",
  783. "description": "bar",
  784. "count()": 1,
  785. },
  786. {
  787. "span.status": "success",
  788. "description": "foo",
  789. "count()": 1,
  790. },
  791. ]
  792. assert meta["dataset"] == self.dataset
  793. @pytest.mark.xfail(reason="event_id isn't being written to the new table")
  794. def test_id_filtering(self):
  795. super().test_id_filtering()
  796. def test_span_duration(self):
  797. spans = [
  798. self.create_span(
  799. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  800. start_ts=self.ten_mins_ago,
  801. ),
  802. self.create_span(
  803. {"description": "foo", "sentry_tags": {"status": "success"}},
  804. start_ts=self.ten_mins_ago,
  805. ),
  806. ]
  807. self.store_spans(spans, is_eap=self.is_eap)
  808. response = self.do_request(
  809. {
  810. "field": ["span.duration", "description"],
  811. "query": "",
  812. "orderby": "description",
  813. "project": self.project.id,
  814. "dataset": self.dataset,
  815. }
  816. )
  817. assert response.status_code == 200, response.content
  818. data = response.data["data"]
  819. meta = response.data["meta"]
  820. assert len(data) == 2
  821. assert data == [
  822. {
  823. "span.duration": 1000.0,
  824. "description": "bar",
  825. "project.name": self.project.slug,
  826. "id": spans[0]["span_id"],
  827. },
  828. {
  829. "span.duration": 1000.0,
  830. "description": "foo",
  831. "project.name": self.project.slug,
  832. "id": spans[1]["span_id"],
  833. },
  834. ]
  835. assert meta["dataset"] == self.dataset
  836. def test_aggregate_numeric_attr_weighted(self):
  837. self.store_spans(
  838. [
  839. self.create_span(
  840. {
  841. "description": "foo",
  842. "sentry_tags": {"status": "success"},
  843. "tags": {"bar": "bar1"},
  844. },
  845. start_ts=self.ten_mins_ago,
  846. ),
  847. self.create_span(
  848. {
  849. "description": "foo",
  850. "sentry_tags": {"status": "success"},
  851. "tags": {"bar": "bar2"},
  852. },
  853. measurements={"foo": {"value": 5}},
  854. start_ts=self.ten_mins_ago,
  855. ),
  856. self.create_span(
  857. {
  858. "description": "foo",
  859. "sentry_tags": {"status": "success"},
  860. "tags": {"bar": "bar3"},
  861. },
  862. start_ts=self.ten_mins_ago,
  863. ),
  864. ],
  865. is_eap=self.is_eap,
  866. )
  867. response = self.do_request(
  868. {
  869. "field": [
  870. "description",
  871. "count_unique(bar)",
  872. "count_unique(tags[bar])",
  873. "count_unique(tags[bar,string])",
  874. "count()",
  875. "count(span.duration)",
  876. "count(tags[foo, number])",
  877. "sum(tags[foo,number])",
  878. "avg(tags[foo,number])",
  879. "p50(tags[foo,number])",
  880. "p75(tags[foo,number])",
  881. "p95(tags[foo,number])",
  882. "p99(tags[foo,number])",
  883. "p100(tags[foo,number])",
  884. "min(tags[foo,number])",
  885. "max(tags[foo,number])",
  886. ],
  887. "query": "",
  888. "orderby": "description",
  889. "project": self.project.id,
  890. "dataset": self.dataset,
  891. }
  892. )
  893. assert response.status_code == 200, response.content
  894. assert len(response.data["data"]) == 1
  895. data = response.data["data"]
  896. assert data[0] == {
  897. "description": "foo",
  898. "count_unique(bar)": 3,
  899. "count_unique(tags[bar])": 3,
  900. "count_unique(tags[bar,string])": 3,
  901. "count()": 3,
  902. "count(span.duration)": 3,
  903. "count(tags[foo, number])": 1,
  904. "sum(tags[foo,number])": 5.0,
  905. "avg(tags[foo,number])": 5.0,
  906. "p50(tags[foo,number])": 5.0,
  907. "p75(tags[foo,number])": 5.0,
  908. "p95(tags[foo,number])": 5.0,
  909. "p99(tags[foo,number])": 5.0,
  910. "p100(tags[foo,number])": 5.0,
  911. "min(tags[foo,number])": 5.0,
  912. "max(tags[foo,number])": 5.0,
  913. }
  914. def test_numeric_attr_without_space(self):
  915. self.store_spans(
  916. [
  917. self.create_span(
  918. {
  919. "description": "foo",
  920. "sentry_tags": {"status": "success"},
  921. "tags": {"foo": "five"},
  922. },
  923. measurements={"foo": {"value": 5}},
  924. start_ts=self.ten_mins_ago,
  925. ),
  926. ],
  927. is_eap=self.is_eap,
  928. )
  929. response = self.do_request(
  930. {
  931. "field": ["description", "tags[foo,number]", "tags[foo,string]", "tags[foo]"],
  932. "query": "",
  933. "orderby": "description",
  934. "project": self.project.id,
  935. "dataset": self.dataset,
  936. }
  937. )
  938. assert response.status_code == 200, response.content
  939. assert len(response.data["data"]) == 1
  940. data = response.data["data"]
  941. assert data[0]["tags[foo,number]"] == 5
  942. assert data[0]["tags[foo,string]"] == "five"
  943. assert data[0]["tags[foo]"] == "five"
  944. def test_numeric_attr_with_spaces(self):
  945. self.store_spans(
  946. [
  947. self.create_span(
  948. {
  949. "description": "foo",
  950. "sentry_tags": {"status": "success"},
  951. "tags": {"foo": "five"},
  952. },
  953. measurements={"foo": {"value": 5}},
  954. start_ts=self.ten_mins_ago,
  955. ),
  956. ],
  957. is_eap=self.is_eap,
  958. )
  959. response = self.do_request(
  960. {
  961. "field": ["description", "tags[foo, number]", "tags[foo, string]", "tags[foo]"],
  962. "query": "",
  963. "orderby": "description",
  964. "project": self.project.id,
  965. "dataset": self.dataset,
  966. }
  967. )
  968. assert response.status_code == 200, response.content
  969. assert len(response.data["data"]) == 1
  970. data = response.data["data"]
  971. assert data[0]["tags[foo, number]"] == 5
  972. assert data[0]["tags[foo, string]"] == "five"
  973. assert data[0]["tags[foo]"] == "five"
  974. def test_numeric_attr_filtering(self):
  975. self.store_spans(
  976. [
  977. self.create_span(
  978. {
  979. "description": "foo",
  980. "sentry_tags": {"status": "success"},
  981. "tags": {"foo": "five"},
  982. },
  983. measurements={"foo": {"value": 5}},
  984. start_ts=self.ten_mins_ago,
  985. ),
  986. self.create_span(
  987. {"description": "bar", "sentry_tags": {"status": "success", "foo": "five"}},
  988. measurements={"foo": {"value": 8}},
  989. start_ts=self.ten_mins_ago,
  990. ),
  991. ],
  992. is_eap=self.is_eap,
  993. )
  994. response = self.do_request(
  995. {
  996. "field": ["description", "tags[foo,number]"],
  997. "query": "tags[foo,number]:5",
  998. "orderby": "description",
  999. "project": self.project.id,
  1000. "dataset": self.dataset,
  1001. }
  1002. )
  1003. assert response.status_code == 200, response.content
  1004. assert len(response.data["data"]) == 1
  1005. data = response.data["data"]
  1006. assert data[0]["tags[foo,number]"] == 5
  1007. assert data[0]["description"] == "foo"
  1008. def test_long_attr_name(self):
  1009. response = self.do_request(
  1010. {
  1011. "field": ["description", "z" * 201],
  1012. "query": "",
  1013. "orderby": "description",
  1014. "project": self.project.id,
  1015. "dataset": self.dataset,
  1016. }
  1017. )
  1018. assert response.status_code == 400, response.content
  1019. assert "Is Too Long" in response.data["detail"].title()
  1020. def test_numeric_attr_orderby(self):
  1021. self.store_spans(
  1022. [
  1023. self.create_span(
  1024. {
  1025. "description": "baz",
  1026. "sentry_tags": {"status": "success"},
  1027. "tags": {"foo": "five"},
  1028. },
  1029. measurements={"foo": {"value": 71}},
  1030. start_ts=self.ten_mins_ago,
  1031. ),
  1032. self.create_span(
  1033. {
  1034. "description": "foo",
  1035. "sentry_tags": {"status": "success"},
  1036. "tags": {"foo": "five"},
  1037. },
  1038. measurements={"foo": {"value": 5}},
  1039. start_ts=self.ten_mins_ago,
  1040. ),
  1041. self.create_span(
  1042. {
  1043. "description": "bar",
  1044. "sentry_tags": {"status": "success"},
  1045. "tags": {"foo": "five"},
  1046. },
  1047. measurements={"foo": {"value": 8}},
  1048. start_ts=self.ten_mins_ago,
  1049. ),
  1050. ],
  1051. is_eap=self.is_eap,
  1052. )
  1053. response = self.do_request(
  1054. {
  1055. "field": ["description", "tags[foo,number]"],
  1056. "query": "",
  1057. "orderby": ["tags[foo,number]"],
  1058. "project": self.project.id,
  1059. "dataset": self.dataset,
  1060. }
  1061. )
  1062. assert response.status_code == 200, response.content
  1063. assert len(response.data["data"]) == 3
  1064. data = response.data["data"]
  1065. assert data[0]["tags[foo,number]"] == 5
  1066. assert data[0]["description"] == "foo"
  1067. assert data[1]["tags[foo,number]"] == 8
  1068. assert data[1]["description"] == "bar"
  1069. assert data[2]["tags[foo,number]"] == 71
  1070. assert data[2]["description"] == "baz"
  1071. def test_aggregate_numeric_attr(self):
  1072. self.store_spans(
  1073. [
  1074. self.create_span(
  1075. {
  1076. "description": "foo",
  1077. "sentry_tags": {"status": "success"},
  1078. "tags": {"bar": "bar1"},
  1079. },
  1080. start_ts=self.ten_mins_ago,
  1081. ),
  1082. self.create_span(
  1083. {
  1084. "description": "foo",
  1085. "sentry_tags": {"status": "success"},
  1086. "tags": {"bar": "bar2"},
  1087. },
  1088. measurements={"foo": {"value": 5}},
  1089. start_ts=self.ten_mins_ago,
  1090. ),
  1091. ],
  1092. is_eap=self.is_eap,
  1093. )
  1094. response = self.do_request(
  1095. {
  1096. "field": [
  1097. "description",
  1098. "count_unique(bar)",
  1099. "count_unique(tags[bar])",
  1100. "count_unique(tags[bar,string])",
  1101. "count()",
  1102. "count(span.duration)",
  1103. "count(tags[foo, number])",
  1104. "sum(tags[foo,number])",
  1105. "avg(tags[foo,number])",
  1106. "p50(tags[foo,number])",
  1107. "p75(tags[foo,number])",
  1108. "p95(tags[foo,number])",
  1109. "p99(tags[foo,number])",
  1110. "p100(tags[foo,number])",
  1111. "min(tags[foo,number])",
  1112. "max(tags[foo,number])",
  1113. ],
  1114. "query": "",
  1115. "orderby": "description",
  1116. "project": self.project.id,
  1117. "dataset": self.dataset,
  1118. }
  1119. )
  1120. assert response.status_code == 200, response.content
  1121. assert len(response.data["data"]) == 1
  1122. data = response.data["data"]
  1123. assert data[0] == {
  1124. "description": "foo",
  1125. "count_unique(bar)": 2,
  1126. "count_unique(tags[bar])": 2,
  1127. "count_unique(tags[bar,string])": 2,
  1128. "count()": 2,
  1129. "count(span.duration)": 2,
  1130. "count(tags[foo, number])": 1,
  1131. "sum(tags[foo,number])": 5.0,
  1132. "avg(tags[foo,number])": 5.0,
  1133. "p50(tags[foo,number])": 5.0,
  1134. "p75(tags[foo,number])": 5.0,
  1135. "p95(tags[foo,number])": 5.0,
  1136. "p99(tags[foo,number])": 5.0,
  1137. "p100(tags[foo,number])": 5.0,
  1138. "min(tags[foo,number])": 5.0,
  1139. "max(tags[foo,number])": 5.0,
  1140. }
  1141. def test_margin_of_error(self):
  1142. total_samples = 10
  1143. in_group = 5
  1144. spans = []
  1145. for _ in range(in_group):
  1146. spans.append(
  1147. self.create_span(
  1148. {
  1149. "description": "foo",
  1150. "sentry_tags": {"status": "success"},
  1151. "measurements": {"client_sample_rate": {"value": 0.00001}},
  1152. },
  1153. start_ts=self.ten_mins_ago,
  1154. )
  1155. )
  1156. for _ in range(total_samples - in_group):
  1157. spans.append(
  1158. self.create_span(
  1159. {
  1160. "description": "bar",
  1161. "sentry_tags": {"status": "success"},
  1162. "measurements": {"client_sample_rate": {"value": 0.00001}},
  1163. },
  1164. )
  1165. )
  1166. self.store_spans(
  1167. spans,
  1168. is_eap=self.is_eap,
  1169. )
  1170. response = self.do_request(
  1171. {
  1172. "field": [
  1173. "margin_of_error()",
  1174. "lower_count_limit()",
  1175. "upper_count_limit()",
  1176. "count()",
  1177. ],
  1178. "query": "description:foo",
  1179. "project": self.project.id,
  1180. "dataset": self.dataset,
  1181. }
  1182. )
  1183. assert response.status_code == 200, response.content
  1184. assert len(response.data["data"]) == 1
  1185. data = response.data["data"][0]
  1186. margin_of_error = data["margin_of_error()"]
  1187. lower_limit = data["lower_count_limit()"]
  1188. upper_limit = data["upper_count_limit()"]
  1189. extrapolated = data["count()"]
  1190. assert margin_of_error == pytest.approx(0.306, rel=1e-1)
  1191. # How to read this; these results mean that the extrapolated count is
  1192. # 500k, with a lower estimated bound of ~200k, and an upper bound of 800k
  1193. assert lower_limit == pytest.approx(190_000, abs=5000)
  1194. assert extrapolated == pytest.approx(500_000, abs=5000)
  1195. assert upper_limit == pytest.approx(810_000, abs=5000)
  1196. def test_skip_aggregate_conditions_option(self):
  1197. span_1 = self.create_span(
  1198. {"description": "foo", "sentry_tags": {"status": "success"}},
  1199. start_ts=self.ten_mins_ago,
  1200. )
  1201. span_2 = self.create_span(
  1202. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  1203. start_ts=self.ten_mins_ago,
  1204. )
  1205. self.store_spans(
  1206. [span_1, span_2],
  1207. is_eap=self.is_eap,
  1208. )
  1209. response = self.do_request(
  1210. {
  1211. "field": ["description"],
  1212. "query": "description:foo count():>1",
  1213. "orderby": "description",
  1214. "project": self.project.id,
  1215. "dataset": self.dataset,
  1216. "allowAggregateConditions": "0",
  1217. }
  1218. )
  1219. assert response.status_code == 200, response.content
  1220. data = response.data["data"]
  1221. meta = response.data["meta"]
  1222. assert len(data) == 1
  1223. assert data == [
  1224. {
  1225. "description": "foo",
  1226. "project.name": self.project.slug,
  1227. "id": span_1["span_id"],
  1228. },
  1229. ]
  1230. assert meta["dataset"] == self.dataset
  1231. class OrganizationEventsEAPRPCSpanEndpointTest(OrganizationEventsEAPSpanEndpointTest):
  1232. """These tests aren't fully passing yet, currently inheriting xfail from the eap tests"""
  1233. is_eap = True
  1234. use_rpc = True
  1235. def test_extrapolation(self):
  1236. """Extrapolation only changes the number when there's a sample rate"""
  1237. spans = []
  1238. spans.append(
  1239. self.create_span(
  1240. {
  1241. "description": "foo",
  1242. "sentry_tags": {"status": "success"},
  1243. "measurements": {"client_sample_rate": {"value": 0.1}},
  1244. },
  1245. start_ts=self.ten_mins_ago,
  1246. )
  1247. )
  1248. self.store_spans(spans, is_eap=self.is_eap)
  1249. response = self.do_request(
  1250. {
  1251. "field": ["count()"],
  1252. "query": "",
  1253. "project": self.project.id,
  1254. "dataset": self.dataset,
  1255. }
  1256. )
  1257. assert response.status_code == 200, response.content
  1258. data = response.data["data"]
  1259. assert len(data) == 1
  1260. assert data[0]["count()"] == 10
  1261. def test_span_duration(self):
  1262. spans = [
  1263. self.create_span(
  1264. {"description": "bar", "sentry_tags": {"status": "invalid_argument"}},
  1265. start_ts=self.ten_mins_ago,
  1266. ),
  1267. self.create_span(
  1268. {"description": "foo", "sentry_tags": {"status": "success"}},
  1269. start_ts=self.ten_mins_ago,
  1270. ),
  1271. ]
  1272. self.store_spans(spans, is_eap=self.is_eap)
  1273. response = self.do_request(
  1274. {
  1275. "field": ["span.duration", "description"],
  1276. "query": "",
  1277. "orderby": "description",
  1278. "project": self.project.id,
  1279. "dataset": self.dataset,
  1280. }
  1281. )
  1282. assert response.status_code == 200, response.content
  1283. data = response.data["data"]
  1284. meta = response.data["meta"]
  1285. assert len(data) == 2
  1286. assert data == [
  1287. {
  1288. "span.duration": 1000.0,
  1289. "description": "bar",
  1290. "project.name": self.project.slug,
  1291. "id": spans[0]["span_id"],
  1292. },
  1293. {
  1294. "span.duration": 1000.0,
  1295. "description": "foo",
  1296. "project.name": self.project.slug,
  1297. "id": spans[1]["span_id"],
  1298. },
  1299. ]
  1300. assert meta["dataset"] == self.dataset
  1301. @pytest.mark.xfail(reason="weighted functions will not be moved to the RPC")
  1302. def test_aggregate_numeric_attr_weighted(self):
  1303. super().test_aggregate_numeric_attr_weighted()
  1304. def test_aggregate_numeric_attr(self):
  1305. self.store_spans(
  1306. [
  1307. self.create_span(
  1308. {
  1309. "description": "foo",
  1310. "sentry_tags": {"status": "success"},
  1311. "tags": {"bar": "bar1"},
  1312. },
  1313. start_ts=self.ten_mins_ago,
  1314. ),
  1315. self.create_span(
  1316. {
  1317. "description": "foo",
  1318. "sentry_tags": {"status": "success"},
  1319. "tags": {"bar": "bar2"},
  1320. },
  1321. measurements={"foo": {"value": 5}},
  1322. start_ts=self.ten_mins_ago,
  1323. ),
  1324. ],
  1325. is_eap=self.is_eap,
  1326. )
  1327. response = self.do_request(
  1328. {
  1329. "field": [
  1330. "description",
  1331. "count_unique(bar)",
  1332. "count_unique(tags[bar])",
  1333. "count_unique(tags[bar,string])",
  1334. "count()",
  1335. "count(span.duration)",
  1336. "count(tags[foo, number])",
  1337. "sum(tags[foo,number])",
  1338. "avg(tags[foo,number])",
  1339. "p50(tags[foo,number])",
  1340. "p75(tags[foo,number])",
  1341. "p95(tags[foo,number])",
  1342. "p99(tags[foo,number])",
  1343. "p100(tags[foo,number])",
  1344. "min(tags[foo,number])",
  1345. "max(tags[foo,number])",
  1346. ],
  1347. "query": "",
  1348. "orderby": "description",
  1349. "project": self.project.id,
  1350. "dataset": self.dataset,
  1351. }
  1352. )
  1353. assert response.status_code == 200, response.content
  1354. assert len(response.data["data"]) == 1
  1355. data = response.data["data"]
  1356. assert data[0] == {
  1357. "description": "foo",
  1358. "count_unique(bar)": 2,
  1359. "count_unique(tags[bar])": 2,
  1360. "count_unique(tags[bar,string])": 2,
  1361. "count()": 2,
  1362. "count(span.duration)": 2,
  1363. "count(tags[foo, number])": 1,
  1364. "sum(tags[foo,number])": 5.0,
  1365. "avg(tags[foo,number])": 5.0,
  1366. "p50(tags[foo,number])": 5.0,
  1367. "p75(tags[foo,number])": 5.0,
  1368. "p95(tags[foo,number])": 5.0,
  1369. "p99(tags[foo,number])": 5.0,
  1370. "p100(tags[foo,number])": 5.0,
  1371. "min(tags[foo,number])": 5.0,
  1372. "max(tags[foo,number])": 5.0,
  1373. }
  1374. @pytest.mark.xfail(reason="margin will not be moved to the RPC")
  1375. def test_margin_of_error(self):
  1376. super().test_margin_of_error()
  1377. @pytest.mark.xfail(reason="rpc not handling attr_str vs attr_num with same alias")
  1378. def test_numeric_attr_without_space(self):
  1379. super().test_numeric_attr_without_space()
  1380. @pytest.mark.xfail(reason="rpc not handling attr_str vs attr_num with same alias")
  1381. def test_numeric_attr_with_spaces(self):
  1382. super().test_numeric_attr_with_spaces()
  1383. @pytest.mark.xfail(reason="module not migrated over")
  1384. def test_module_alias(self):
  1385. super().test_module_alias()
  1386. @pytest.mark.xfail(reason="wip: not implemented yet")
  1387. def test_inp_span(self):
  1388. super().test_inp_span()
  1389. @pytest.mark.xfail(reason="wip: not implemented yet")
  1390. def test_network_span(self):
  1391. super().test_network_span()
  1392. @pytest.mark.xfail(reason="wip: not implemented yet")
  1393. def test_other_category_span(self):
  1394. super().test_other_category_span()
  1395. @pytest.mark.xfail(reason="wip: not implemented yet")
  1396. def test_queue_span(self):
  1397. super().test_queue_span()
  1398. @pytest.mark.xfail(reason="wip: not implemented yet")
  1399. def test_sentry_tags_syntax(self):
  1400. super().test_sentry_tags_syntax()
  1401. @pytest.mark.xfail(reason="wip: not implemented yet")
  1402. def test_span_op_casing(self):
  1403. super().test_span_op_casing()
  1404. def test_tag_wildcards(self):
  1405. self.store_spans(
  1406. [
  1407. self.create_span(
  1408. {"description": "foo", "tags": {"foo": "bar"}},
  1409. start_ts=self.ten_mins_ago,
  1410. ),
  1411. self.create_span(
  1412. {"description": "qux", "tags": {"foo": "qux"}},
  1413. start_ts=self.ten_mins_ago,
  1414. ),
  1415. ],
  1416. is_eap=self.is_eap,
  1417. )
  1418. for query in [
  1419. "foo:b*",
  1420. "foo:*r",
  1421. "foo:*a*",
  1422. "foo:b*r",
  1423. ]:
  1424. response = self.do_request(
  1425. {
  1426. "field": ["foo", "count()"],
  1427. "query": query,
  1428. "project": self.project.id,
  1429. "dataset": self.dataset,
  1430. }
  1431. )
  1432. assert response.status_code == 200, response.content
  1433. assert response.data["data"] == [{"foo": "bar", "count()": 1}]
  1434. @pytest.mark.xfail(reason="rate not implemented yet")
  1435. def test_spm(self):
  1436. super().test_spm()
  1437. @pytest.mark.xfail(reason="units not implemented yet")
  1438. def test_simple_measurements(self):
  1439. super().test_simple_measurements()