test_sentry_api_compat.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. import json
  2. from typing import Dict, List, Union
  3. from django.urls import reverse
  4. from django.utils import dateparse
  5. from events.models import Event, LogLevel
  6. from events.test_data.csp import mdn_sample_csp
  7. from events.test_data.django_error_factory import message
  8. from glitchtip.test_utils.test_case import GlitchTipTestCase
  9. TEST_DATA_DIR = "events/test_data"
  10. is_exception = lambda v: v.get("type") == "exception"
  11. class SentryAPICompatTestCase(GlitchTipTestCase):
  12. """
  13. These tests use raw json data taken from the OSS Sentry SDK.
  14. It compares GlitchTip's processing of the events with OSS Sentry's raw event
  15. JSON link and API Endpoints JSON.
  16. These tests do not include references to the proprietary Sentry API at this time.
  17. Compatibility with the proprietary Sentry API will be considered for interoperability reasons
  18. as long as testing can be done while respecting's Sentry's license by not referencing/viewing
  19. proprietary code
  20. """
  21. def setUp(self):
  22. self.create_user_and_project()
  23. key = self.project.projectkey_set.first().public_key
  24. self.event_store_url = (
  25. reverse("event_store", args=[self.project.id]) + "?sentry_key=" + key.hex
  26. )
  27. self.csp_store_url = (
  28. reverse("csp_store", args=[self.project.id]) + "?sentry_key=" + key.hex
  29. )
  30. # Upgrade functions handle intentional differences between GlitchTip and Sentry OSS
  31. def upgrade_title(self, value: str):
  32. """Sentry OSS uses ... while GlitchTip uses unicode …"""
  33. if value[-1] == "…":
  34. return value[:-4]
  35. return value.strip("...")
  36. def upgrade_metadata(self, value: dict):
  37. value["title"] = self.upgrade_title(value.get("title"))
  38. return value
  39. def upgrade_datetime(self, value: str):
  40. """
  41. DRF DateTimeField sets precision while Sentry OSS does not
  42. DRF DateTimeField 2020-07-23T02:54:37.823000Z
  43. Sentry OSS: 2020-07-23T02:54:37.823Z
  44. """
  45. date_value = dateparse.parse_datetime(value)
  46. if date_value:
  47. value = date_value.isoformat()
  48. if value.endswith("+00:00"):
  49. value = value[:-6] + "Z"
  50. return value
  51. def upgrade_data(self, data: Union[str, dict, list]):
  52. """A recursive replace function"""
  53. if isinstance(data, dict):
  54. return {k: self.upgrade_data(v) for k, v in data.items()}
  55. elif isinstance(data, list):
  56. return [self.upgrade_data(i) for i in data]
  57. elif isinstance(data, str):
  58. if data.endswith("Z"):
  59. return self.upgrade_datetime(data)
  60. return data
  61. return data
  62. def assertCompareData(self, data1: Dict, data2: Dict, fields: List[str]):
  63. """Compare data of two dict objects. Compare only provided fields list"""
  64. for field in fields:
  65. field_value1 = data1.get(field)
  66. field_value2 = data2.get(field)
  67. if field == "title" and isinstance(field_value1, str):
  68. field_value1 = self.upgrade_title(field_value1)
  69. field_value2 = self.upgrade_title(field_value2)
  70. if (
  71. field == "metadata"
  72. and isinstance(field_value1, dict)
  73. and field_value1.get("title")
  74. ):
  75. field_value1 = self.upgrade_metadata(field_value1)
  76. field_value2 = self.upgrade_metadata(field_value2)
  77. self.assertEqual(
  78. field_value1,
  79. field_value2,
  80. f"Failed for field '{field}'",
  81. )
  82. def get_json_test_data(self, name: str):
  83. """Get incoming event, sentry json, sentry api event"""
  84. event = self.get_json_data(f"{TEST_DATA_DIR}/incoming_events/{name}.json")
  85. sentry_json = self.get_json_data(f"{TEST_DATA_DIR}/oss_sentry_json/{name}.json")
  86. # Force captured test data to match test generated data
  87. sentry_json["project"] = self.project.id
  88. api_sentry_event = self.get_json_data(
  89. f"{TEST_DATA_DIR}/oss_sentry_events/{name}.json"
  90. )
  91. return event, sentry_json, api_sentry_event
  92. def get_json_data(self, path: str):
  93. with open(path) as json_file:
  94. return json.load(json_file)
  95. def get_project_events_detail(self, event_id):
  96. return reverse(
  97. "project-events-detail",
  98. kwargs={
  99. "project_pk": f"{self.project.organization.slug}/{self.project.slug}",
  100. "pk": event_id,
  101. },
  102. )
  103. def test_template_error(self):
  104. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  105. "django_template_error"
  106. )
  107. res = self.client.post(self.event_store_url, sdk_error, format="json")
  108. self.assertEqual(res.status_code, 200)
  109. event = Event.objects.get(pk=res.data["id"])
  110. url = self.get_project_events_detail(event.pk)
  111. res = self.client.get(url)
  112. self.assertEqual(res.status_code, 200)
  113. self.assertCompareData(res.data, sentry_data, ["culprit", "title", "metadata"])
  114. res_frames = res.data["entries"][0]["data"]["values"][0]["stacktrace"]["frames"]
  115. frames = sentry_data["entries"][0]["data"]["values"][0]["stacktrace"]["frames"]
  116. for i in range(6):
  117. # absPath don't always match - needs fixed
  118. self.assertCompareData(res_frames[i], frames[i], ["absPath"])
  119. for res_frame, frame in zip(res_frames, frames):
  120. self.assertCompareData(
  121. res_frame,
  122. frame,
  123. ["lineNo", "function", "filename", "module", "context"],
  124. )
  125. if frame.get("vars"):
  126. self.assertCompareData(
  127. res_frame["vars"], frame["vars"], ["exc", "request"]
  128. )
  129. if frame["vars"].get("get_response"):
  130. # Memory address is different, truncate it
  131. self.assertEqual(
  132. res_frame["vars"]["get_response"][:-16],
  133. frame["vars"]["get_response"][:-16],
  134. )
  135. self.assertCompareData(
  136. res.data["entries"][0]["data"],
  137. sentry_data["entries"][0]["data"],
  138. ["env", "headers", "url", "method", "inferredContentType"],
  139. )
  140. url = reverse("issue-detail", kwargs={"pk": event.issue.pk})
  141. res = self.client.get(url)
  142. self.assertEqual(res.status_code, 200)
  143. data = self.get_json_data("events/test_data/django_template_error_issue.json")
  144. self.assertCompareData(res.data, data, ["title", "metadata"])
  145. def test_js_sdk_with_unix_timestamp(self):
  146. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  147. "js_event_with_unix_timestamp"
  148. )
  149. res = self.client.post(self.event_store_url, sdk_error, format="json")
  150. event = Event.objects.first()
  151. self.assertIsNotNone(event.timestamp)
  152. self.assertNotEqual(event.timestamp, sdk_error["timestamp"])
  153. event_json = event.event_json()
  154. self.assertCompareData(event_json, sentry_json, ["datetime", "breadcrumbs"])
  155. url = self.get_project_events_detail(event.pk)
  156. res = self.client.get(url)
  157. res_data = res.json()
  158. self.assertEqual(res_data["entries"][1].get("type"), "breadcrumbs")
  159. self.assertEqual(
  160. res_data["entries"][1],
  161. self.upgrade_data(sentry_data["entries"][1]),
  162. )
  163. def test_dotnet_error(self):
  164. # Don't mimic this test, use self.get_json_test_data instead
  165. sdk_error = self.get_json_data(
  166. "events/test_data/incoming_events/dotnet_error.json"
  167. )
  168. res = self.client.post(self.event_store_url, sdk_error, format="json")
  169. self.assertEqual(res.status_code, 200)
  170. self.assertEqual(Event.objects.count(), 1)
  171. event_id = res.data["id"]
  172. sentry_data = self.get_json_data(
  173. "events/test_data/oss_sentry_events/dotnet_error.json"
  174. )
  175. url = self.get_project_events_detail(event_id)
  176. res = self.client.get(url)
  177. self.assertEqual(res.status_code, 200)
  178. self.assertCompareData(
  179. res.data,
  180. sentry_data,
  181. ["eventID", "title", "culprit", "platform", "type", "metadata"],
  182. )
  183. res_exception = next(filter(is_exception, res.data["entries"]), None)
  184. sentry_exception = next(filter(is_exception, sentry_data["entries"]), None)
  185. self.assertEqual(
  186. res_exception["data"].get("hasSystemFrames"),
  187. sentry_exception["data"].get("hasSystemFrames"),
  188. )
  189. def test_php_message_event(self):
  190. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  191. "php_message_event"
  192. )
  193. res = self.client.post(self.event_store_url, sdk_error, format="json")
  194. event = Event.objects.first()
  195. self.assertEqual(res.status_code, 200)
  196. event_id = res.data["id"]
  197. event_json = event.event_json()
  198. self.assertCompareData(event_json, sentry_json, ["logentry"])
  199. sentry_data = self.get_json_data(
  200. "events/test_data/oss_sentry_events/php_message_event.json"
  201. )
  202. url = self.get_project_events_detail(event_id)
  203. res = self.client.get(url)
  204. self.assertEqual(res.status_code, 200)
  205. self.assertCompareData(
  206. res.data,
  207. sentry_data,
  208. [
  209. "message",
  210. "title",
  211. ],
  212. )
  213. self.assertEqual(res.data["entries"][0], sentry_data["entries"][0])
  214. def test_django_message_params(self):
  215. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  216. "django_message_params"
  217. )
  218. res = self.client.post(self.event_store_url, sdk_error, format="json")
  219. event = Event.objects.first()
  220. self.assertEqual(res.status_code, 200)
  221. event_id = res.data["id"]
  222. event_json = event.event_json()
  223. self.assertCompareData(event_json, sentry_json, ["logentry"])
  224. sentry_data = self.get_json_data(
  225. "events/test_data/oss_sentry_events/django_message_params.json"
  226. )
  227. url = self.get_project_events_detail(event_id)
  228. res = self.client.get(url)
  229. self.assertEqual(res.status_code, 200)
  230. self.assertCompareData(
  231. res.data,
  232. sentry_data,
  233. [
  234. "message",
  235. "title",
  236. ],
  237. )
  238. self.assertEqual(res.data["entries"][0], sentry_data["entries"][0])
  239. def test_csp_event(self):
  240. # Don't mimic this test, use self.get_jest_test_data instead
  241. data = mdn_sample_csp
  242. res = self.client.post(self.csp_store_url, data, format="json")
  243. self.assertEqual(res.status_code, 200)
  244. event_id = res.data["id"]
  245. url = self.get_project_events_detail(event_id)
  246. res = self.client.get(url)
  247. self.assertEqual(res.status_code, 200)
  248. data = self.get_json_data("events/test_data/csp_event.json")
  249. self.assertCompareData(res.data, data, ["title", "culprit", "type", "metadata"])
  250. self.assertEqual(res.data["entries"][0], data["entries"][0])
  251. self.assertEqual(res.data["entries"][1], data["entries"][1])
  252. def test_csp_event_glitchtip_key(self):
  253. """Check compatibility for using glitchtip_key or sentry_key interchangably"""
  254. key = self.project.projectkey_set.first().public_key
  255. csp_store_url = (
  256. reverse("csp_store", args=[self.project.id]) + "?glitchtip_key=" + key.hex
  257. )
  258. data = mdn_sample_csp
  259. res = self.client.post(csp_store_url, data, format="json")
  260. self.assertEqual(res.status_code, 200)
  261. event_id = res.data["id"]
  262. url = self.get_project_events_detail(event_id)
  263. res = self.client.get(url)
  264. self.assertEqual(res.status_code, 200)
  265. def test_message_event(self):
  266. """A generic message made with the Sentry SDK. Generally has less data than exceptions."""
  267. # Don't mimic this test, use self.get_json_test_data instead
  268. res = self.client.post(self.event_store_url, message, format="json")
  269. self.assertEqual(res.status_code, 200)
  270. event_id = res.data["id"]
  271. url = self.get_project_events_detail(event_id)
  272. res = self.client.get(url)
  273. self.assertEqual(res.status_code, 200)
  274. data = self.get_json_data("events/test_data/django_message_event.json")
  275. self.assertCompareData(
  276. res.data,
  277. data,
  278. ["title", "culprit", "type", "metadata", "platform", "packages"],
  279. )
  280. def test_python_logging(self):
  281. """Test Sentry SDK logging integration based event"""
  282. sdk_error, sentry_json, sentry_data = self.get_json_test_data("python_logging")
  283. res = self.client.post(self.event_store_url, sdk_error, format="json")
  284. event = Event.objects.get(pk=res.data["id"])
  285. event_json = event.event_json()
  286. res = self.client.get(self.get_project_events_detail(event.pk))
  287. self.assertEqual(res.status_code, 200)
  288. self.assertCompareData(event_json, sentry_json, ["logentry", "title"])
  289. self.assertCompareData(
  290. res.data,
  291. sentry_data,
  292. [
  293. "title",
  294. "logentry",
  295. "culprit",
  296. "type",
  297. "metadata",
  298. "platform",
  299. "packages",
  300. ],
  301. )
  302. def test_go_file_not_found(self):
  303. # Don't mimic this test, use self.get_jest_test_data instead
  304. sdk_error = self.get_json_data(
  305. "events/test_data/incoming_events/go_file_not_found.json"
  306. )
  307. res = self.client.post(self.event_store_url, sdk_error, format="json")
  308. self.assertEqual(res.status_code, 200)
  309. self.assertEqual(Event.objects.count(), 1)
  310. event_id = res.data["id"]
  311. sentry_data = self.get_json_data(
  312. "events/test_data/oss_sentry_events/go_file_not_found.json"
  313. )
  314. url = self.get_project_events_detail(event_id)
  315. res = self.client.get(url)
  316. self.assertEqual(res.status_code, 200)
  317. self.assertCompareData(
  318. res.data,
  319. sentry_data,
  320. ["title", "culprit", "type", "metadata", "platform"],
  321. )
  322. def test_very_small_event(self):
  323. """
  324. Shows a very minimalist event example. Good for seeing what data is null
  325. """
  326. # Don't mimic this test, use self.get_jest_test_data instead
  327. sdk_error = self.get_json_data(
  328. "events/test_data/incoming_events/very_small_event.json"
  329. )
  330. res = self.client.post(self.event_store_url, sdk_error, format="json")
  331. event_id = res.data["id"]
  332. sentry_data = self.get_json_data(
  333. "events/test_data/oss_sentry_events/very_small_event.json"
  334. )
  335. url = self.get_project_events_detail(event_id)
  336. res = self.client.get(url)
  337. self.assertEqual(res.status_code, 200)
  338. self.assertCompareData(
  339. res.data,
  340. sentry_data,
  341. ["title", "culprit", "type", "metadata", "platform", "entries"],
  342. )
  343. def test_python_zero_division(self):
  344. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  345. "python_zero_division"
  346. )
  347. res = self.client.post(self.event_store_url, sdk_error, format="json")
  348. event = Event.objects.get(pk=res.data["id"])
  349. event_json = event.event_json()
  350. self.assertCompareData(
  351. event_json,
  352. sentry_json,
  353. [
  354. "event_id",
  355. "project",
  356. "release",
  357. "dist",
  358. "platform",
  359. "level",
  360. "exception",
  361. "modules",
  362. "time_spent",
  363. "sdk",
  364. "type",
  365. "title",
  366. "culprit",
  367. "breadcrumbs",
  368. ],
  369. )
  370. self.assertCompareData(
  371. event_json["request"],
  372. sentry_json["request"],
  373. [
  374. "url",
  375. # "headers",
  376. "method",
  377. "env",
  378. "query_string",
  379. "inferred_content_type",
  380. ],
  381. )
  382. self.assertEqual(
  383. event_json["datetime"][:22],
  384. sentry_json["datetime"][:22],
  385. "Compare if datetime is almost the same",
  386. )
  387. url = self.get_project_events_detail(event.pk)
  388. res = self.client.get(url)
  389. self.assertEqual(res.status_code, 200)
  390. self.assertCompareData(
  391. res.data,
  392. sentry_data,
  393. ["title", "culprit", "type", "metadata", "platform", "packages"],
  394. )
  395. issue = event.issue
  396. issue.refresh_from_db()
  397. self.assertEqual(issue.level, LogLevel.ERROR)
  398. def test_dotnet_zero_division(self):
  399. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  400. "dotnet_divide_zero"
  401. )
  402. res = self.client.post(self.event_store_url, sdk_error, format="json")
  403. event = Event.objects.get(pk=res.data["id"])
  404. url = self.get_project_events_detail(event.pk)
  405. res = self.client.get(url)
  406. self.assertCompareData(event.event_json(), sentry_json, ["environment"])
  407. self.assertCompareData(
  408. res.data,
  409. sentry_data,
  410. [
  411. "eventID",
  412. "title",
  413. "culprit",
  414. "platform",
  415. "type",
  416. "metadata",
  417. ],
  418. )
  419. res_exception = next(filter(is_exception, res.data["entries"]), None)
  420. sentry_exception = next(filter(is_exception, sentry_data["entries"]), None)
  421. self.assertEqual(
  422. res_exception["data"]["values"][0]["stacktrace"]["frames"][4]["context"],
  423. sentry_exception["data"]["values"][0]["stacktrace"]["frames"][4]["context"],
  424. )
  425. tags = res.data.get("tags")
  426. browser_tag = next(filter(lambda tag: tag["key"] == "browser", tags), None)
  427. self.assertEqual(browser_tag["value"], "Firefox 76.0")
  428. environment_tag = next(
  429. filter(lambda tag: tag["key"] == "environment", tags), None
  430. )
  431. self.assertEqual(environment_tag["value"], "Development")
  432. event_json = event.event_json()
  433. browser_tag = next(
  434. filter(lambda tag: tag[0] == "browser", event_json.get("tags")), None
  435. )
  436. self.assertEqual(browser_tag[1], "Firefox 76.0")
  437. def test_sentry_cli_send_event_no_level(self):
  438. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  439. "sentry_cli_send_event_no_level"
  440. )
  441. res = self.client.post(self.event_store_url, sdk_error, format="json")
  442. event = Event.objects.get(pk=res.data["id"])
  443. event_json = event.event_json()
  444. self.assertCompareData(event_json, sentry_json, ["title", "message"])
  445. self.assertEqual(event_json["project"], event.issue.project_id)
  446. url = self.get_project_events_detail(event.pk)
  447. res = self.client.get(url)
  448. self.assertCompareData(
  449. res.data,
  450. sentry_data,
  451. [
  452. "userReport",
  453. "title",
  454. "culprit",
  455. "type",
  456. "metadata",
  457. "message",
  458. "platform",
  459. "previousEventID",
  460. ],
  461. )
  462. self.assertEqual(res.data["projectID"], event.issue.project_id)
  463. def test_js_error_with_context(self):
  464. self.project.scrub_ip_addresses = False
  465. self.project.save()
  466. sdk_error, sentry_json, sentry_data = self.get_json_test_data(
  467. "js_error_with_context"
  468. )
  469. res = self.client.post(
  470. self.event_store_url, sdk_error, format="json", REMOTE_ADDR="142.255.29.14"
  471. )
  472. event = Event.objects.get(pk=res.data["id"])
  473. event_json = event.event_json()
  474. self.assertCompareData(
  475. event_json, sentry_json, ["title", "message", "extra", "user"]
  476. )
  477. url = self.get_project_events_detail(event.pk)
  478. res = self.client.get(url)
  479. self.assertCompareData(res.json(), sentry_data, ["context", "user"])
  480. def test_elixir_stacktrace(self):
  481. """The elixir SDK does things differently"""
  482. sdk_error, sentry_json, sentry_data = self.get_json_test_data("elixir_error")
  483. res = self.client.post(self.event_store_url, sdk_error, format="json")
  484. event = Event.objects.get(pk=res.data["id"])
  485. event_json = event.event_json()
  486. self.assertCompareData(
  487. event_json["exception"]["values"][0],
  488. sentry_json["exception"]["values"][0],
  489. ["type", "values", "exception"],
  490. )
  491. def test_small_js_error(self):
  492. """A small example to test stacktraces"""
  493. sdk_error, sentry_json, sentry_data = self.get_json_test_data("small_js_error")
  494. res = self.client.post(self.event_store_url, sdk_error, format="json")
  495. event = Event.objects.get(pk=res.data["id"])
  496. event_json = event.event_json()
  497. self.assertCompareData(
  498. event_json["exception"]["values"][0],
  499. sentry_json["exception"]["values"][0],
  500. ["type", "values", "exception", "abs_path"],
  501. )