import re from timeit import default_timer as timer from django.shortcuts import reverse from django.utils import timezone from freezegun import freeze_time from model_bakery import baker from glitchtip.test_utils.test_case import GlitchTipTestCase from issues.models import EventStatus, Issue from ..tasks import update_search_index_all_issues class EventTestCase(GlitchTipTestCase): def setUp(self): self.create_user_and_project() self.url = reverse( "project-events-list", kwargs={ "project_pk": f"{self.project.organization.slug}/{self.project.slug}" }, ) def test_project_events_list(self): event = baker.make("events.Event", issue__project=self.project) baker.make("events.Event", issue__project=self.project, _quantity=3) not_my_event = baker.make("events.Event") with self.assertNumQueries(2): res = self.client.get(self.url) self.assertContains(res, event.pk.hex) self.assertNotContains(res, not_my_event.pk.hex) def test_events_latest(self): """ Should show more recent event with previousEventID of previous/first event """ event = baker.make("events.Event", issue__project=self.project) event2 = baker.make("events.Event", issue=event.issue) url = f"/api/0/issues/{event.issue.id}/events/latest/" res = self.client.get(url) self.assertContains(res, event2.pk.hex) self.assertEqual(res.data["previousEventID"], event.pk.hex) self.assertEqual(res.data["nextEventID"], None) def test_next_prev_event(self): """Get next and previous event IDs that belong to same issue""" issue1 = baker.make("issues.Issue", project=self.project) issue2 = baker.make("issues.Issue", project=self.project) baker.make("events.Event") issue1_event1 = baker.make("events.Event", issue=issue1) issue2_event1 = baker.make("events.Event", issue=issue2) issue1_event2 = baker.make("events.Event", issue=issue1) url = reverse("issue-events-latest", args=[issue1.id]) res = self.client.get(url) self.assertContains(res, issue1_event2.pk.hex) self.assertEqual(res.data["previousEventID"], issue1_event1.pk.hex) def test_entries_emtpy(self): """A minimal or incomplete data set should result in an empty entries array""" data = { "sdk": { "name": "sentry", "version": "5", "packages": [], "integrations": [], }, "type": "error", "title": "", "culprit": "", "request": { "url": "http://localhost", "headers": [], "inferred_content_type": None, }, "contexts": None, "metadata": {"value": "Non-Error exception"}, "packages": None, "platform": "javascript", "exception": { "values": [ { "type": "Error", "value": "Non-Error exception", "mechanism": { "data": {"function": ""}, "type": "instrument", "handled": True, }, } ] }, } event = baker.make("events.Event", issue__project=self.project, data=data) res = self.client.get(self.url) self.assertTrue("entries" in res.data[0]) def test_event_json(self): event = baker.make("events.Event", issue__project=self.project) team = baker.make("teams.Team", organization=self.organization) team.members.add(self.org_user) self.project.team_set.add(team) url = reverse( "event_json", kwargs={ "org": self.organization.slug, "issue": event.issue.id, "event": event.event_id_hex, }, ) res = self.client.get(url) self.assertContains(res, event.event_id_hex) url = reverse( "event_json", kwargs={ "org": "nope", "issue": event.issue.id, "event": event.event_id_hex, }, ) res = self.client.get(url) self.assertEqual(res.status_code, 404) def test_two_teams_event_detail(self): """ Addresses https://gitlab.com/glitchtip/glitchtip-backend/-/issues/215 Ensure a user can be in more than one team on the same project """ team = baker.make("teams.Team", organization=self.organization) team2 = baker.make("teams.Team", organization=self.organization) team.members.add(self.org_user) team2.members.add(self.org_user) self.project.team_set.add(team) self.project.team_set.add(team2) issue = baker.make("issues.Issue", project=self.project) event = baker.make("events.Event", issue=issue) url = reverse( "issue-events-detail", kwargs={ "issue_pk": event.issue.id, "pk": event.event_id_hex, }, ) res = self.client.get(url) self.assertContains(res, event.event_id_hex) class IssuesAPITestCase(GlitchTipTestCase): def setUp(self): self.create_user_and_project() self.url = reverse("issue-list") def test_issues_single_page_list(self): issue = baker.make("issues.Issue", project=self.project) not_my_issue = baker.make("issues.Issue") with self.assertNumQueries(2): res = self.client.get(self.url) self.assertContains(res, issue.title) self.assertNotContains(res, not_my_issue.title) self.assertEqual(res.get("X-Hits"), "1") def test_issues_multi_page_list(self): first_issue = baker.make("issues.Issue", project=self.project, title="first_issue") baker.make("issues.Issue", project=self.project, _quantity=50) last_issue = baker.make("issues.Issue", project=self.project, title="last_issue") with self.assertNumQueries(3): res = self.client.get(self.url) self.assertEqual(res.headers.get("X-Hits"), "52") self.assertEqual(res.data[0]["id"], str(last_issue.id)) self.assertNotContains(res, str(first_issue.title)) pattern = r'(?<=\<).+?(?=\>)' links = re.findall(pattern, res.headers.get("Link")) res = self.client.get(links[1]) self.assertEqual(res.headers.get("X-Hits"), "52") self.assertEqual(res.data[-1]["id"], str(first_issue.id)) self.assertNotContains(res, str(last_issue.title)) def test_no_duplicate_issues(self): """ Addresses https://gitlab.com/glitchtip/glitchtip-backend/-/issues/109 Ensure issues can be filtered by org membership but not duplicated """ baker.make("issues.Issue", project=self.project) team2 = baker.make("teams.Team", organization=self.organization) team2.members.add(self.org_user) self.project.team_set.add(team2) res = self.client.get(self.url) self.assertEqual(len(res.data), 1) team2.delete() self.team.delete() res = self.client.get(self.url) self.assertEqual(len(res.data), 1) self.org_user.delete() res = self.client.get(self.url) self.assertEqual(len(res.data), 0) def test_issue_retrieve(self): issue = baker.make("issues.Issue", project=self.project) not_my_issue = baker.make("issues.Issue") url = reverse("issue-detail", args=[issue.id]) res = self.client.get(url) self.assertContains(res, issue.title) url = reverse("issue-detail", args=[not_my_issue.id]) res = self.client.get(url) self.assertEqual(res.status_code, 404) def test_issue_last_seen(self): with freeze_time(timezone.datetime(2020, 3, 1)): issue = baker.make("issues.Issue", project=self.project) events = baker.make("events.Event", issue=issue, _quantity=2) res = self.client.get(self.url) self.assertEqual( res.data[0]["lastSeen"][:19], events[1].created.isoformat()[:19] ) def test_issue_delete(self): issue = baker.make("issues.Issue", project=self.project) not_my_issue = baker.make("issues.Issue") url = reverse("issue-detail", args=[issue.id]) res = self.client.delete(url) self.assertEqual(res.status_code, 204) url = reverse("issue-detail", args=[not_my_issue.id]) res = self.client.delete(url) self.assertEqual(res.status_code, 404) def test_issue_update(self): issue = baker.make(Issue, project=self.project) self.assertEqual(issue.status, EventStatus.UNRESOLVED) url = reverse("issue-detail", kwargs={"pk": issue.pk}) data = {"status": "resolved"} res = self.client.put(url, data) self.assertEqual(res.status_code, 200) issue.refresh_from_db() self.assertEqual(issue.status, EventStatus.RESOLVED) def test_bulk_update(self): """Bulk update only supports Issue status""" issues = baker.make(Issue, project=self.project, _quantity=2) url = f"{self.url}?id={issues[0].id}&id={issues[1].id}" status_to_set = EventStatus.RESOLVED data = {"status": status_to_set.label} res = self.client.put(url, data) self.assertContains(res, status_to_set.label) issues = Issue.objects.all() self.assertEqual(issues[0].status, status_to_set) self.assertEqual(issues[1].status, status_to_set) def test_bulk_delete_via_ids(self): """Bulk delete Issues with ids""" issues = baker.make(Issue, project=self.project, _quantity=2) url = f"{self.url}?id={issues[0].id}&id={issues[1].id}" res = self.client.delete(url) issues = Issue.objects.all().count() self.assertEqual(issues, 0) def test_bulk_delete_via_search(self): """Bulk delete Issues via search string""" project2 = baker.make("projects.Project", organization=self.organization) project2.team_set.add(self.team) issue1 = baker.make(Issue, project=self.project) issue2 = baker.make(Issue, project=project2) url = f"{self.url}?query=is:unresolved&project={self.project.id}" res = self.client.delete(url) self.assertEqual(Issue.objects.filter(id=issue1.id).exists(), False) self.assertEqual(Issue.objects.filter(id=issue2.id).exists(), True) def test_bulk_update_query(self): """Bulk update only supports Issue status""" project2 = baker.make("projects.Project", organization=self.organization) project2.team_set.add(self.team) issue1 = baker.make(Issue, project=self.project) issue2 = baker.make(Issue, project=project2) url = f"{self.url}?query=is:unresolved&project={self.project.id}" status_to_set = EventStatus.RESOLVED data = {"status": status_to_set.label} res = self.client.put(url, data) self.assertContains(res, status_to_set.label) issue1.refresh_from_db() issue2.refresh_from_db() self.assertEqual(issue1.status, status_to_set) self.assertEqual(issue2.status, EventStatus.UNRESOLVED) def test_filter_project(self): baker.make(Issue, project=self.project) project = baker.make("projects.Project", organization=self.organization) project.team_set.add(self.team) issue = baker.make(Issue, project=project) res = self.client.get(self.url, {"project": project.id}) self.assertEqual(len(res.data), 1) self.assertContains(res, issue.id) res = self.client.get(self.url, {"project": "nothing"}) self.assertEqual(res.status_code, 400) def test_filter_environment(self): environment1_name = "prod" environment2_name = "staging" issue1 = baker.make( Issue, project=self.project, event_set__tags={"environment": "??"}, ) baker.make( Issue, project=self.project, event_set__tags={"foos": environment1_name}, ) baker.make( "events.Event", issue=issue1, tags={"environment": environment1_name} ) issue2 = baker.make( Issue, project=self.project, event_set__tags={"environment": environment2_name}, ) baker.make( "events.Event", issue=issue2, tags={"environment": environment2_name} ) baker.make(Issue, project=self.project) baker.make(Issue, project=self.project, event_set__tags={"environment": "dev"}) baker.make( Issue, project=self.project, event_set__tags={"lol": environment2_name} ) update_search_index_all_issues() res = self.client.get( self.url, {"environment": [environment1_name, environment2_name]}, ) self.assertEqual(len(res.data), 2) self.assertContains(res, issue1.id) self.assertContains(res, issue2.id) def test_issue_list_filter(self): project1 = self.project project2 = baker.make("projects.Project", organization=self.organization) project2.team_set.add(self.team) project3 = baker.make("projects.Project", organization=self.organization) project3.team_set.add(self.team) issue1 = baker.make("issues.Issue", project=project1) issue2 = baker.make("issues.Issue", project=project2) issue3 = baker.make("issues.Issue", project=project3) res = self.client.get( self.url + f"?project={project1.id}&project={project2.id}" ) self.assertContains(res, issue1.title) self.assertContains(res, issue2.title) self.assertNotContains(res, issue3.title) def test_issue_list_sort(self): issue1 = baker.make("issues.Issue", project=self.project) issue2 = baker.make("issues.Issue", project=self.project) issue3 = baker.make("issues.Issue", project=self.project) baker.make("events.Event", issue=issue2, _quantity=2) baker.make("events.Event", issue=issue1) update_search_index_all_issues() res = self.client.get(self.url) self.assertEqual(res.data[0]["id"], str(issue1.id)) res = self.client.get(self.url + "?sort=-count") self.assertEqual(res.data[0]["id"], str(issue2.id)) res = self.client.get(self.url + "?sort=priority") self.assertEqual(res.data[0]["id"], str(issue3.id)) res = self.client.get(self.url + "?sort=-priority") self.assertEqual(res.data[0]["id"], str(issue2.id)) def test_filter_is_status(self): """Match sentry's usage of "is" for status filtering""" resolved_issue = baker.make( Issue, status=EventStatus.RESOLVED, project=self.project ) unresolved_issue = baker.make( Issue, status=EventStatus.UNRESOLVED, project=self.project, tags={"platform": "Linux"}, ) res = self.client.get(self.url, {"query": "is:unresolved has:platform"}) self.assertEqual(len(res.data), 1) self.assertContains(res, unresolved_issue.title) self.assertNotContains(res, resolved_issue.title) def test_issue_serializer_type(self): """ Ensure type field is show in serializer https://gitlab.com/glitchtip/glitchtip-backend/-/issues/9 """ issue = baker.make("issues.Issue", project=self.project) url = reverse("issue-detail", args=[issue.id]) res = self.client.get(url) self.assertContains(res, issue.get_type_display()) def test_event_release(self): release = baker.make("releases.Release", organization=self.organization) event = baker.make("events.Event", issue__project=self.project, release=release) url = reverse( "project-events-list", kwargs={ "project_pk": f"{self.project.organization.slug}/{self.project.slug}", }, ) res = self.client.get(url) # Not in list view self.assertNotContains(res, release.version) url = reverse( "project-events-detail", kwargs={ "project_pk": f"{self.project.organization.slug}/{self.project.slug}", "pk": event.pk, }, ) res = self.client.get(url) self.assertContains(res, release.version) def test_issue_tags(self): issue = baker.make("issues.Issue", project=self.project) baker.make("events.Event", issue=issue, tags={"foo": "bar"}, _quantity=2) baker.make("events.Event", issue=issue, tags={"foo": "bar", "animal": "cat"}) baker.make( "events.Event", issue=issue, tags={"animal": "dog", "foo": "cat"}, _quantity=4, ) url = reverse("issue-detail", args=[issue.id]) res = self.client.get(url + "tags/") # Order is random if res.data[0]["name"] == "animal": animal = res.data[0] foo = res.data[1] else: animal = res.data[1] foo = res.data[0] self.assertEqual(animal["totalValues"], 5) self.assertEqual(animal["topValues"][0]["value"], "dog") self.assertEqual(animal["topValues"][0]["count"], 4) self.assertEqual(animal["uniqueValues"], 2) self.assertEqual(foo["totalValues"], 7) self.assertEqual(foo["topValues"][0]["value"], "cat") self.assertEqual(foo["topValues"][0]["count"], 4) self.assertEqual(foo["uniqueValues"], 2) def test_issue_tags_filter(self): issue = baker.make("issues.Issue", project=self.project) baker.make("events.Event", issue=issue, tags={"foo": "bar", "lol": "bar"}) url = reverse("issue-detail", args=[issue.id]) res = self.client.get(url + "tags/?key=foo") self.assertEqual(len(res.data), 1) def test_issue_tags_performance(self): issue = baker.make("issues.Issue", project=self.project) baker.make("events.Event", issue=issue, tags={"foo": "bar"}, _quantity=50) baker.make( "events.Event", issue=issue, tags={"foo": "bar", "animal": "cat"}, _quantity=100, ) baker.make( "events.Event", issue=issue, tags={"type": "a", "animal": "cat"}, _quantity=100, ) baker.make( "events.Event", issue=issue, tags={"haha": "a", "arg": "cat", "b": "b"}, _quantity=100, ) baker.make( "events.Event", issue=issue, tags={"type": "b", "foo": "bar"}, _quantity=200 ) url = reverse("issue-detail", args=[issue.id]) with self.assertNumQueries(5): # Includes many auth related queries start = timer() res = self.client.get(url + "tags/") end = timer() # print(end - start) def test_issue_comment_count(self): issue = baker.make("issues.Issue", project=self.project) baker.make("issues.Comment", issue=issue, _quantity=2) with self.assertNumQueries(4): res = self.client.get(self.url + f"{issue.pk}/") self.assertEqual(res.data["numComments"], 2) def test_issue_tag_detail(self): issue = baker.make("issues.Issue", project=self.project) baker.make( "events.Event", issue=issue, tags={"foo": "bar", "a": "b"}, _quantity=2 ) baker.make("events.Event", issue=issue, tags={"foo": "foobar"}) baker.make("events.Event", issue=issue, tags={"type": "a"}) url = reverse("issue-detail", args=[issue.id]) res = self.client.get(url + "tags/foo/") self.assertContains(res, "foobar") self.assertEqual(res.data["totalValues"], 3) self.assertEqual(res.data["uniqueValues"], 2) res = self.client.get(url + "tags/ahh/") self.assertEqual(res.status_code, 404) def test_issue_greatest_level(self): """ The issue should be the greatest level seen in events This is a deviation from Sentry OSS """ issue = baker.make("issues.Issue", level=1) baker.make("events.Event", issue=issue, level=1) baker.make("events.Event", issue=issue, level=3) baker.make("events.Event", issue=issue, level=2) Issue.update_index(issue.pk) issue.refresh_from_db() self.assertEqual(issue.level, 3)