123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475 |
- import json
- from typing import List, Dict, Union
- from django.utils import dateparse
- from django.urls import reverse
- from events.test_data.django_error_factory import message
- from events.test_data.csp import mdn_sample_csp
- from glitchtip.test_utils.test_case import GlitchTipTestCase
- from events.models import Event, LogLevel
- TEST_DATA_DIR = "events/test_data"
- is_exception = lambda v: v.get("type") == "exception"
- class SentryAPICompatTestCase(GlitchTipTestCase):
- """
- These tests use raw json data taken from the OSS Sentry SDK.
- It compares GlitchTip's processing of the events with OSS Sentry's raw event
- JSON link and API Endpoints JSON.
- These tests do not include references to the proprietary Sentry API at this time.
- Compatibility with the proprietary Sentry API will be considered for interoperability reasons
- as long as testing can be done while respecting's Sentry's license by not referencing/viewing
- proprietary code
- """
- def setUp(self):
- self.create_user_and_project()
- key = self.project.projectkey_set.first().public_key
- self.event_store_url = (
- reverse("event_store", args=[self.project.id]) + "?sentry_key=" + key.hex
- )
- self.csp_store_url = (
- reverse("csp_store", args=[self.project.id]) + "?sentry_key=" + key.hex
- )
- # Upgrade functions handle intentional differences between GlitchTip and Sentry OSS
- def upgrade_title(self, value: str):
- """ Sentry OSS uses ... while GlitchTip uses unicode …"""
- if value[-1] == "…":
- return value[:-4]
- return value.strip("...")
- def upgrade_metadata(self, value: dict):
- value["title"] = self.upgrade_title(value.get("title"))
- return value
- def upgrade_datetime(self, value: str):
- """
- DRF DateTimeField sets precision while Sentry OSS does not
- DRF DateTimeField 2020-07-23T02:54:37.823000Z
- Sentry OSS: 2020-07-23T02:54:37.823Z
- """
- date_value = dateparse.parse_datetime(value)
- if date_value:
- value = date_value.isoformat()
- if value.endswith("+00:00"):
- value = value[:-6] + "Z"
- return value
- def upgrade_data(self, data: Union[str, dict, list]):
- """ A recursive replace function """
- if isinstance(data, dict):
- return {k: self.upgrade_data(v) for k, v in data.items()}
- elif isinstance(data, list):
- return [self.upgrade_data(i) for i in data]
- elif isinstance(data, str):
- if data.endswith("Z"):
- return self.upgrade_datetime(data)
- return data
- return data
- def assertCompareData(self, data1: Dict, data2: Dict, fields: List[str]):
- """ Compare data of two dict objects. Compare only provided fields list """
- for field in fields:
- field_value1 = data1.get(field)
- field_value2 = data2.get(field)
- if field == "title" and isinstance(field_value1, str):
- field_value1 = self.upgrade_title(field_value1)
- field_value2 = self.upgrade_title(field_value2)
- if (
- field == "metadata"
- and isinstance(field_value1, dict)
- and field_value1.get("title")
- ):
- field_value1 = self.upgrade_metadata(field_value1)
- field_value2 = self.upgrade_metadata(field_value2)
- self.assertEqual(
- field_value1, field_value2, f"Failed for field '{field}'",
- )
- def get_json_test_data(self, name: str):
- """ Get incoming event, sentry json, sentry api event """
- event = self.get_json_data(f"{TEST_DATA_DIR}/incoming_events/{name}.json")
- sentry_json = self.get_json_data(f"{TEST_DATA_DIR}/oss_sentry_json/{name}.json")
- # Force captured test data to match test generated data
- sentry_json["project"] = self.project.id
- api_sentry_event = self.get_json_data(
- f"{TEST_DATA_DIR}/oss_sentry_events/{name}.json"
- )
- return event, sentry_json, api_sentry_event
- def get_json_data(self, path: str):
- with open(path) as json_file:
- return json.load(json_file)
- def get_project_events_detail(self, event_id):
- return reverse(
- "project-events-detail",
- kwargs={
- "project_pk": f"{self.project.organization.slug}/{self.project.slug}",
- "pk": event_id,
- },
- )
- def test_template_error(self):
- sdk_error, sentry_json, sentry_data = self.get_json_test_data(
- "django_template_error"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- self.assertEqual(res.status_code, 200)
- event = Event.objects.get(pk=res.data["id"])
- url = self.get_project_events_detail(event.pk)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- self.assertCompareData(res.data, sentry_data, ["culprit", "title", "metadata"])
- res_frames = res.data["entries"][0]["data"]["values"][0]["stacktrace"]["frames"]
- frames = sentry_data["entries"][0]["data"]["values"][0]["stacktrace"]["frames"]
- for i in range(6):
- # absPath don't always match - needs fixed
- self.assertCompareData(res_frames[i], frames[i], ["absPath"])
- for res_frame, frame in zip(res_frames, frames):
- self.assertCompareData(
- res_frame,
- frame,
- ["lineNo", "function", "filename", "module", "context"],
- )
- if frame.get("vars"):
- self.assertCompareData(
- res_frame["vars"], frame["vars"], ["exc", "request"]
- )
- if frame["vars"].get("get_response"):
- # Memory address is different, truncate it
- self.assertEqual(
- res_frame["vars"]["get_response"][:-16],
- frame["vars"]["get_response"][:-16],
- )
- self.assertCompareData(
- res.data["entries"][0]["data"],
- sentry_data["entries"][0]["data"],
- ["env", "headers", "url", "method", "inferredContentType"],
- )
- url = reverse("issue-detail", kwargs={"pk": event.issue.pk})
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- data = self.get_json_data("events/test_data/django_template_error_issue.json")
- self.assertCompareData(res.data, data, ["title", "metadata"])
- def test_js_sdk_with_unix_timestamp(self):
- sdk_error, sentry_json, sentry_data = self.get_json_test_data(
- "js_event_with_unix_timestamp"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- event = Event.objects.first()
- self.assertIsNotNone(event.timestamp)
- self.assertNotEqual(event.timestamp, sdk_error["timestamp"])
- event_json = event.event_json()
- self.assertCompareData(event_json, sentry_json, ["datetime", "breadcrumbs"])
- url = self.get_project_events_detail(event.pk)
- res = self.client.get(url)
- res_data = res.json()
- self.assertCompareData(res_data, sentry_data, ["datetime"])
- self.assertEqual(res_data["entries"][1].get("type"), "breadcrumbs")
- self.assertEqual(
- res_data["entries"][1], self.upgrade_data(sentry_data["entries"][1]),
- )
- def test_dotnet_error(self):
- # Don't mimic this test, use self.get_jest_test_data instead
- sdk_error = self.get_json_data(
- "events/test_data/incoming_events/dotnet_error.json"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- self.assertEqual(res.status_code, 200)
- self.assertEqual(Event.objects.count(), 1)
- event_id = res.data["id"]
- sentry_data = self.get_json_data(
- "events/test_data/oss_sentry_events/dotnet_error.json"
- )
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- self.assertCompareData(
- res.data,
- sentry_data,
- ["eventID", "title", "culprit", "platform", "type", "metadata"],
- )
- res_exception = next(filter(is_exception, res.data["entries"]), None)
- sentry_exception = next(filter(is_exception, sentry_data["entries"]), None)
- self.assertEqual(
- res_exception["data"].get("hasSystemFrames"),
- sentry_exception["data"].get("hasSystemFrames"),
- )
- def test_csp_event(self):
- # Don't mimic this test, use self.get_jest_test_data instead
- data = mdn_sample_csp
- res = self.client.post(self.csp_store_url, data, format="json")
- self.assertEqual(res.status_code, 200)
- event_id = res.data["id"]
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- data = self.get_json_data("events/test_data/csp_event.json")
- self.assertCompareData(res.data, data, ["title", "culprit", "type", "metadata"])
- self.assertEqual(res.data["entries"][0], data["entries"][0])
- self.assertEqual(res.data["entries"][1], data["entries"][1])
- def test_csp_event_glitchtip_key(self):
- """ Check compatibility for using glitchtip_key or sentry_key interchangably """
- key = self.project.projectkey_set.first().public_key
- csp_store_url = (
- reverse("csp_store", args=[self.project.id]) + "?glitchtip_key=" + key.hex
- )
- data = mdn_sample_csp
- res = self.client.post(csp_store_url, data, format="json")
- self.assertEqual(res.status_code, 200)
- event_id = res.data["id"]
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- def test_message_event(self):
- """ A generic message made with the Sentry SDK. Generally has less data than exceptions. """
- # Don't mimic this test, use self.get_jest_test_data instead
- res = self.client.post(self.event_store_url, message, format="json")
- self.assertEqual(res.status_code, 200)
- event_id = res.data["id"]
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- data = self.get_json_data("events/test_data/django_message_event.json")
- self.assertCompareData(
- res.data,
- data,
- ["title", "culprit", "type", "metadata", "platform", "packages"],
- )
- def test_python_logging(self):
- """ Test Sentry SDK logging integration based event """
- # Don't mimic this test, use self.get_jest_test_data instead
- sdk_error = self.get_json_data(
- "events/test_data/incoming_events/python_logging.json"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- self.assertEqual(res.status_code, 200)
- self.assertEqual(Event.objects.count(), 1)
- event_id = res.data["id"]
- sentry_data = self.get_json_data(
- "events/test_data/oss_sentry_events/python_logging.json"
- )
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- self.assertCompareData(
- res.data,
- sentry_data,
- ["title", "culprit", "type", "metadata", "platform", "packages"],
- )
- def test_go_file_not_found(self):
- # Don't mimic this test, use self.get_jest_test_data instead
- sdk_error = self.get_json_data(
- "events/test_data/incoming_events/go_file_not_found.json"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- self.assertEqual(res.status_code, 200)
- self.assertEqual(Event.objects.count(), 1)
- event_id = res.data["id"]
- sentry_data = self.get_json_data(
- "events/test_data/oss_sentry_events/go_file_not_found.json"
- )
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- self.assertCompareData(
- res.data, sentry_data, ["title", "culprit", "type", "metadata", "platform"],
- )
- def test_very_small_event(self):
- """
- Shows a very minimalist event example. Good for seeing what data is null
- """
- # Don't mimic this test, use self.get_jest_test_data instead
- sdk_error = self.get_json_data(
- "events/test_data/incoming_events/very_small_event.json"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- event_id = res.data["id"]
- sentry_data = self.get_json_data(
- "events/test_data/oss_sentry_events/very_small_event.json"
- )
- url = self.get_project_events_detail(event_id)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- self.assertCompareData(
- res.data,
- sentry_data,
- ["title", "culprit", "type", "metadata", "platform", "entries"],
- )
- def test_python_zero_division(self):
- sdk_error, sentry_json, sentry_data = self.get_json_test_data(
- "python_zero_division"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- event = Event.objects.get(pk=res.data["id"])
- event_json = event.event_json()
- self.assertCompareData(
- event_json,
- sentry_json,
- [
- "event_id",
- "project",
- "release",
- "dist",
- "platform",
- "level",
- "exception",
- "modules",
- "time_spent",
- "sdk",
- "type",
- "title",
- "culprit",
- "breadcrumbs",
- ],
- )
- self.assertCompareData(
- event_json["request"],
- sentry_json["request"],
- [
- "url",
- # "headers",
- "method",
- "env",
- "query_string",
- "inferred_content_type",
- ],
- )
- self.assertEqual(
- event_json["datetime"][:22],
- sentry_json["datetime"][:22],
- "Compare if datetime is almost the same",
- )
- url = self.get_project_events_detail(event.pk)
- res = self.client.get(url)
- self.assertEqual(res.status_code, 200)
- self.assertCompareData(
- res.data,
- sentry_data,
- ["title", "culprit", "type", "metadata", "platform", "packages"],
- )
- issue = event.issue
- issue.refresh_from_db()
- self.assertEqual(issue.level, LogLevel.ERROR)
- def test_dotnet_zero_division(self):
- sdk_error, sentry_json, sentry_data = self.get_json_test_data(
- "dotnet_divide_zero"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- event = Event.objects.get(pk=res.data["id"])
- url = self.get_project_events_detail(event.pk)
- res = self.client.get(url)
- self.assertCompareData(event.event_json(), sentry_json, ["environment"])
- self.assertCompareData(
- res.data,
- sentry_data,
- ["eventID", "title", "culprit", "platform", "type", "metadata",],
- )
- res_exception = next(filter(is_exception, res.data["entries"]), None)
- sentry_exception = next(filter(is_exception, sentry_data["entries"]), None)
- self.assertEqual(
- res_exception["data"]["values"][0]["stacktrace"]["frames"][4]["context"],
- sentry_exception["data"]["values"][0]["stacktrace"]["frames"][4]["context"],
- )
- tags = res.data.get("tags")
- browser_tag = next(filter(lambda tag: tag["key"] == "browser", tags), None)
- self.assertEqual(browser_tag["value"], "Firefox 76.0")
- environment_tag = next(
- filter(lambda tag: tag["key"] == "environment", tags), None
- )
- self.assertEqual(environment_tag["value"], "Development")
- event_json = event.event_json()
- browser_tag = next(
- filter(lambda tag: tag[0] == "browser", event_json.get("tags")), None
- )
- self.assertEqual(browser_tag[1], "Firefox 76.0")
- def test_sentry_cli_send_event_no_level(self):
- sdk_error, sentry_json, sentry_data = self.get_json_test_data(
- "sentry_cli_send_event_no_level"
- )
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- event = Event.objects.get(pk=res.data["id"])
- event_json = event.event_json()
- self.assertCompareData(event_json, sentry_json, ["title", "message"])
- self.assertEqual(event_json["project"], event.issue.project_id)
- url = self.get_project_events_detail(event.pk)
- res = self.client.get(url)
- self.assertCompareData(
- res.data,
- sentry_data,
- [
- "userReport",
- "title",
- "culprit",
- "type",
- "metadata",
- "message",
- "platform",
- "previousEventID",
- ],
- )
- self.assertEqual(res.data["projectID"], event.issue.project_id)
- def test_js_error_with_context(self):
- self.project.scrub_ip_addresses = False
- self.project.save()
- sdk_error, sentry_json, sentry_data = self.get_json_test_data(
- "js_error_with_context"
- )
- res = self.client.post(
- self.event_store_url, sdk_error, format="json", REMOTE_ADDR="142.255.29.14"
- )
- event = Event.objects.get(pk=res.data["id"])
- event_json = event.event_json()
- self.assertCompareData(
- event_json, sentry_json, ["title", "message", "extra", "user"]
- )
- url = self.get_project_events_detail(event.pk)
- res = self.client.get(url)
- self.assertCompareData(res.json(), sentry_data, ["context", "user"])
- def test_elixir_stacktrace(self):
- """ The elixir SDK does things differently """
- sdk_error, sentry_json, sentry_data = self.get_json_test_data("elixir_error")
- res = self.client.post(self.event_store_url, sdk_error, format="json")
- event = Event.objects.get(pk=res.data["id"])
- event_json = event.event_json()
- self.assertCompareData(
- event_json["exception"]["values"][0],
- sentry_json["exception"]["values"][0],
- ["type", "values", "exception"],
- )
|