123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555 |
- import pytest
- from django.urls import NoReverseMatch, reverse
- from sentry.discover.models import (
- DiscoverSavedQuery,
- DiscoverSavedQueryProject,
- DiscoverSavedQueryTypes,
- )
- from sentry.testutils.cases import APITestCase, SnubaTestCase
- class DiscoverSavedQueryDetailTest(APITestCase, SnubaTestCase):
- feature_name = "organizations:discover"
- def setUp(self):
- super().setUp()
- self.login_as(user=self.user)
- self.org = self.create_organization(owner=self.user)
- self.org_without_access = self.create_organization()
- self.project_ids = [
- self.create_project(organization=self.org).id,
- self.create_project(organization=self.org).id,
- ]
- query = {"fields": ["test"], "conditions": [], "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org, created_by_id=self.user.id, name="Test query", query=query
- )
- model.set_projects(self.project_ids)
- self.query_id = model.id
- invalid = DiscoverSavedQuery.objects.create(
- organization=self.org_without_access, name="Query without access", query=query
- )
- invalid.set_projects(self.project_ids)
- self.query_id_without_access = invalid.id
- def setup_no_team_user(self):
- # disable Open Membership
- self.org.flags.allow_joinleave = False
- self.org.save()
- # user has no access to the first project
- user_no_team = self.create_user(is_superuser=False)
- self.create_member(user=user_no_team, organization=self.org, role="member", teams=[])
- self.login_as(user_no_team)
- def test_invalid_id(self):
- with pytest.raises(NoReverseMatch):
- reverse("sentry-api-0-discover-saved-query-detail", args=[self.org.slug, "not-an-id"])
- def test_get(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.get(url)
- assert response.status_code == 200, response.content
- assert response.data["id"] == str(self.query_id)
- assert set(response.data["projects"]) == set(self.project_ids)
- assert response.data["fields"] == ["test"]
- assert response.data["conditions"] == []
- assert response.data["limit"] == 10
- def test_get_discover_query_flag(self):
- with self.feature("organizations:discover-query"):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.get(url)
- assert response.status_code == 200, response.content
- assert response.data["id"] == str(self.query_id)
- assert set(response.data["projects"]) == set(self.project_ids)
- assert response.data["fields"] == ["test"]
- assert response.data["conditions"] == []
- assert response.data["limit"] == 10
- def test_get_version(self):
- query = {"fields": ["event_id"], "query": "event.type:error", "limit": 10, "version": 2}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org, created_by_id=self.user.id, name="v2 query", query=query
- )
- model.set_projects(self.project_ids)
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, model.id]
- )
- response = self.client.get(url)
- assert response.status_code == 200, response.content
- assert response.data["id"] == str(model.id)
- assert set(response.data["projects"]) == set(self.project_ids)
- assert response.data["fields"] == ["event_id"]
- assert response.data["query"] == "event.type:error"
- assert response.data["limit"] == 10
- assert response.data["version"] == 2
- def test_get_org_without_access(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org_without_access.slug, self.query_id],
- )
- response = self.client.get(url)
- assert response.status_code == 403, response.content
- def test_get_homepage_query(self):
- query = {"fields": ["event_id"], "query": "event.type:error", "limit": 10, "version": 2}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="v2 query",
- query=query,
- is_homepage=True,
- )
- model.set_projects(self.project_ids)
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, model.id]
- )
- response = self.client.get(url)
- assert response.status_code == 404, response.content
- def test_get_disallow_when_no_project_access(self):
- self.setup_no_team_user()
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.get(url)
- assert response.status_code == 403, response.data
- assert response.data == {"detail": "You do not have permission to perform this action."}
- def test_put(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.put(
- url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- },
- )
- assert response.status_code == 200, response.content
- assert response.data["id"] == str(self.query_id)
- assert set(response.data["projects"]) == set(self.project_ids)
- assert response.data["fields"] == []
- assert response.data["conditions"] == []
- assert response.data["limit"] == 20
- def test_put_dataset(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.put(
- url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- "queryDataset": "transaction-like",
- },
- )
- assert response.status_code == 200, response.content
- assert response.data["id"] == str(self.query_id)
- assert set(response.data["projects"]) == set(self.project_ids)
- assert response.data["fields"] == []
- assert response.data["conditions"] == []
- assert response.data["limit"] == 20
- assert response.data["queryDataset"] == "transaction-like"
- def test_put_dataset_with_discover_dataset_returns_validation_error(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- with self.feature({"organizations:deprecate-discover-widget-type": True}):
- response = self.client.put(
- url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- "queryDataset": "discover",
- },
- )
- assert response.status_code == 400, response.content
- assert (
- "Attribute value `discover` is deprecated. Please use `error-events` or `transaction-like`"
- in response.content.decode()
- )
- def test_dataset_set_to_discover_on_update(self):
- query = {"fields": ["event_id"], "query": "event.type:error", "limit": 10, "version": 2}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="query",
- query=query,
- dataset=DiscoverSavedQueryTypes.TRANSACTION_LIKE,
- )
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, model.id]
- )
- response = self.client.put(
- url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- },
- )
- assert response.status_code == 200, response.content
- assert response.data["id"] == str(model.id)
- assert response.data["queryDataset"] == "error-events"
- assert response.data["datasetSource"] == "unknown"
- def test_put_with_interval(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.put(
- url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": ["transaction", "count()"],
- "range": "24h",
- "interval": "10m",
- "version": 2,
- "orderby": "-count",
- },
- )
- assert response.status_code == 200, response.content
- assert response.data["fields"] == ["transaction", "count()"]
- assert response.data["interval"] == "10m"
- def test_put_query_without_access(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org.slug, self.query_id_without_access],
- )
- response = self.client.put(
- url, {"name": "New query", "projects": self.project_ids, "range": "24h"}
- )
- assert response.status_code == 404
- def test_put_query_with_team(self):
- team = self.create_team(organization=self.org, members=[self.user])
- project = self.create_project(organization=self.org, teams=[team])
- query = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="Test query",
- query={"fields": ["test"], "conditions": [], "limit": 10},
- )
- query.set_projects([project.id])
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org.slug, query.id],
- )
- response = self.client.put(url, {"name": "New query", "projects": [], "range": "24h"})
- assert response.status_code == 200
- def test_put_query_without_team(self):
- team = self.create_team(organization=self.org, members=[])
- project = self.create_project(organization=self.org, teams=[team])
- query = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="Test query",
- query={"fields": ["test"], "conditions": [], "limit": 10},
- )
- query.set_projects([project.id])
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org.slug, query.id],
- )
- response = self.client.put(url, {"name": "New query", "projects": [], "range": "24h"})
- assert response.status_code == 400
- assert "No Projects found, join a Team" == response.data["detail"]
- def test_put_homepage_query(self):
- query = {"fields": ["event_id"], "query": "event.type:error", "limit": 10, "version": 2}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="v2 query",
- query=query,
- is_homepage=True,
- )
- model.set_projects(self.project_ids)
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org.slug, model.id],
- )
- response = self.client.put(
- url, {"name": "New query", "projects": [], "range": "24h", "fields": []}
- )
- assert response.status_code == 404, response.content
- def test_put_org_without_access(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org_without_access.slug, self.query_id],
- )
- response = self.client.put(
- url, {"name": "New query", "projects": self.project_ids, "range": "24h"}
- )
- assert response.status_code == 403, response.content
- def test_put_disallow_when_no_project_access(self):
- self.setup_no_team_user()
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.put(
- url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- },
- )
- assert response.status_code == 403, response.data
- assert response.data == {"detail": "You do not have permission to perform this action."}
- def test_delete(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.delete(url)
- assert response.status_code == 204
- assert self.client.get(url).status_code == 404
- def test_delete_removes_projects(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- self.client.delete(url)
- projects = list(
- DiscoverSavedQueryProject.objects.filter(discover_saved_query=self.query_id)
- )
- assert projects == []
- def test_delete_query_without_access(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org.slug, self.query_id_without_access],
- )
- response = self.client.delete(url)
- assert response.status_code == 404
- def test_delete_org_without_access(self):
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org_without_access.slug, self.query_id],
- )
- response = self.client.delete(url)
- assert response.status_code == 403, response.content
- def test_delete_homepage_query(self):
- query = {"fields": ["event_id"], "query": "event.type:error", "limit": 10, "version": 2}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="v2 query",
- query=query,
- is_homepage=True,
- )
- model.set_projects(self.project_ids)
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail",
- args=[self.org.slug, model.id],
- )
- response = self.client.delete(url)
- assert response.status_code == 404, response.content
- def test_delete_disallow_when_no_project_access(self):
- self.setup_no_team_user()
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, self.query_id]
- )
- response = self.client.delete(url)
- assert response.status_code == 403, response.data
- assert response.data == {"detail": "You do not have permission to perform this action."}
- def test_disallow_delete_all_projects_savedquery_when_no_open_membership(self):
- self.setup_no_team_user()
- query = {"fields": ["event_id"], "query": "event.type:error", "limit": 10, "version": 2}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by_id=self.user.id,
- name="v2 query",
- query=query,
- )
- assert not model.projects.exists()
- with self.feature(self.feature_name):
- url = reverse(
- "sentry-api-0-discover-saved-query-detail", args=[self.org.slug, model.id]
- )
- response = self.client.delete(url)
- assert response.status_code == 403, response.data
- assert response.data == {"detail": "You do not have permission to perform this action."}
- class OrganizationDiscoverQueryVisitTest(APITestCase, SnubaTestCase):
- def setUp(self):
- super().setUp()
- self.login_as(user=self.user)
- self.org = self.create_organization(owner=self.user)
- self.org_without_access = self.create_organization()
- self.project_ids = [
- self.create_project(organization=self.org).id,
- self.create_project(organization=self.org).id,
- ]
- q = {"fields": ["test"], "conditions": [], "limit": 10}
- self.query = DiscoverSavedQuery.objects.create(
- organization=self.org, created_by_id=self.user.id, name="Test query", query=q
- )
- self.query.set_projects(self.project_ids)
- def url(self, query_id):
- return reverse(
- "sentry-api-0-discover-saved-query-visit",
- kwargs={"organization_id_or_slug": self.org.slug, "query_id": query_id},
- )
- def test_visit_query(self):
- last_visited = self.query.last_visited
- assert last_visited is not None
- assert self.query.visits == 1
- with self.feature("organizations:discover-query"):
- response = self.client.post(self.url(self.query.id))
- assert response.status_code == 204
- query = DiscoverSavedQuery.objects.get(id=self.query.id)
- assert query.visits == 2
- assert query.last_visited is not None
- assert query.last_visited > last_visited
- def test_visit_query_no_access(self):
- last_visited = self.query.last_visited
- assert self.query.visits == 1
- with self.feature({"organizations:discover-query": False}):
- response = self.client.post(self.url(self.query.id))
- assert response.status_code == 404
- query = DiscoverSavedQuery.objects.get(id=self.query.id)
- assert query.visits == 1
- assert query.last_visited == last_visited
|