123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670 |
- from django.urls import reverse
- from sentry.discover.models import DiscoverSavedQuery
- from sentry.testutils import APITestCase, SnubaTestCase
- from sentry.testutils.helpers.datetime import before_now, iso_format
- class DiscoverSavedQueryBase(APITestCase, SnubaTestCase):
- def setUp(self):
- super().setUp()
- self.login_as(user=self.user)
- self.org = self.create_organization(owner=self.user)
- self.projects = [
- self.create_project(organization=self.org),
- self.create_project(organization=self.org),
- ]
- self.project_ids = [project.id for project in self.projects]
- self.project_ids_without_access = [self.create_project().id]
- query = {"fields": ["test"], "conditions": [], "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org, created_by=self.user, name="Test query", query=query, version=1
- )
- model.set_projects(self.project_ids)
- class DiscoverSavedQueriesTest(DiscoverSavedQueryBase):
- feature_name = "organizations:discover"
- def setUp(self):
- super().setUp()
- self.url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- def test_get(self):
- with self.feature(self.feature_name):
- response = self.client.get(self.url)
- assert response.status_code == 200, response.content
- assert len(response.data) == 1
- assert response.data[0]["name"] == "Test query"
- assert response.data[0]["projects"] == self.project_ids
- assert response.data[0]["fields"] == ["test"]
- assert response.data[0]["conditions"] == []
- assert response.data[0]["limit"] == 10
- assert response.data[0]["version"] == 1
- assert "createdBy" in response.data[0]
- assert response.data[0]["createdBy"]["username"] == self.user.username
- assert not response.data[0]["expired"]
- def test_get_version_filter(self):
- with self.feature(self.feature_name):
- response = self.client.get(self.url, format="json", data={"query": "version:1"})
- assert response.status_code == 200, response.content
- assert len(response.data) == 1
- assert response.data[0]["name"] == "Test query"
- with self.feature(self.feature_name):
- response = self.client.get(self.url, format="json", data={"query": "version:2"})
- assert response.status_code == 200, response.content
- assert len(response.data) == 0
- def test_get_name_filter(self):
- with self.feature(self.feature_name):
- response = self.client.get(self.url, format="json", data={"query": "Test"})
- assert response.status_code == 200, response.content
- assert len(response.data) == 1
- assert response.data[0]["name"] == "Test query"
- with self.feature(self.feature_name):
- # Also available as the name: filter.
- response = self.client.get(self.url, format="json", data={"query": "name:Test"})
- assert response.status_code == 200, response.content
- assert len(response.data) == 1
- assert response.data[0]["name"] == "Test query"
- with self.feature(self.feature_name):
- response = self.client.get(self.url, format="json", data={"query": "name:Nope"})
- assert response.status_code == 200, response.content
- assert len(response.data) == 0
- def test_get_all_paginated(self):
- for i in range(0, 10):
- query = {"fields": ["test"], "conditions": [], "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=self.user,
- name=f"My query {i}",
- query=query,
- version=1,
- )
- model.set_projects(self.project_ids)
- with self.feature(self.feature_name):
- response = self.client.get(self.url, data={"per_page": 1})
- assert response.status_code == 200, response.content
- assert len(response.data) == 1
- with self.feature(self.feature_name):
- # The all parameter ignores pagination and returns all values.
- response = self.client.get(self.url, data={"per_page": 1, "all": 1})
- assert response.status_code == 200, response.content
- assert len(response.data) == 11
- def test_get_sortby(self):
- query = {"fields": ["message"], "query": "", "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=self.user,
- name="My query",
- query=query,
- version=2,
- date_created=before_now(minutes=10),
- date_updated=before_now(minutes=10),
- )
- model.set_projects(self.project_ids)
- sort_options = {
- "dateCreated": True,
- "-dateCreated": False,
- "dateUpdated": True,
- "-dateUpdated": False,
- "name": True,
- "-name": False,
- }
- for sorting, forward_sort in sort_options.items():
- with self.feature(self.feature_name):
- response = self.client.get(self.url, data={"sortBy": sorting})
- assert response.status_code == 200
- values = [row[sorting.strip("-")] for row in response.data]
- if not forward_sort:
- values = list(reversed(values))
- assert list(sorted(values)) == values
- def test_get_sortby_most_popular(self):
- query = {"fields": ["message"], "query": "", "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=self.user,
- name="My query",
- query=query,
- version=2,
- visits=3,
- date_created=before_now(minutes=10),
- date_updated=before_now(minutes=10),
- last_visited=before_now(minutes=5),
- )
- model.set_projects(self.project_ids)
- for forward_sort in [True, False]:
- sorting = "mostPopular" if forward_sort else "-mostPopular"
- with self.feature(self.feature_name):
- response = self.client.get(self.url, data={"sortBy": sorting})
- assert response.status_code == 200
- values = [row["name"] for row in response.data]
- expected = ["My query", "Test query"]
- if not forward_sort:
- expected = list(reversed(expected))
- assert values == expected
- def test_get_sortby_recently_viewed(self):
- query = {"fields": ["message"], "query": "", "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=self.user,
- name="My query",
- query=query,
- version=2,
- visits=3,
- date_created=before_now(minutes=10),
- date_updated=before_now(minutes=10),
- last_visited=before_now(minutes=5),
- )
- model.set_projects(self.project_ids)
- for forward_sort in [True, False]:
- sorting = "recentlyViewed" if forward_sort else "-recentlyViewed"
- with self.feature(self.feature_name):
- response = self.client.get(self.url, data={"sortBy": sorting})
- assert response.status_code == 200
- values = [row["name"] for row in response.data]
- expected = ["Test query", "My query"]
- if not forward_sort:
- expected = list(reversed(expected))
- assert values == expected
- def test_get_sortby_myqueries(self):
- uhoh_user = self.create_user(username="uhoh")
- self.create_member(organization=self.org, user=uhoh_user)
- whoops_user = self.create_user(username="whoops")
- self.create_member(organization=self.org, user=whoops_user)
- query = {"fields": ["message"], "query": "", "limit": 10}
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=uhoh_user,
- name="a query for uhoh",
- query=query,
- version=2,
- date_created=before_now(minutes=10),
- date_updated=before_now(minutes=10),
- )
- model.set_projects(self.project_ids)
- model = DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=whoops_user,
- name="a query for whoops",
- query=query,
- version=2,
- date_created=before_now(minutes=10),
- date_updated=before_now(minutes=10),
- )
- model.set_projects(self.project_ids)
- with self.feature(self.feature_name):
- response = self.client.get(self.url, data={"sortBy": "myqueries"})
- assert response.status_code == 200, response.content
- values = [int(item["createdBy"]["id"]) for item in response.data]
- assert values == [self.user.id, uhoh_user.id, whoops_user.id]
- def test_get_expired_query(self):
- query = {
- "start": iso_format(before_now(days=90)),
- "end": iso_format(before_now(days=61)),
- }
- DiscoverSavedQuery.objects.create(
- organization=self.org,
- created_by=self.user,
- name="My expired query",
- query=query,
- version=2,
- date_created=before_now(days=90),
- date_updated=before_now(minutes=10),
- )
- with self.options({"system.event-retention-days": 60}), self.feature(self.feature_name):
- response = self.client.get(self.url, {"query": "name:My expired query"})
- assert response.status_code == 200, response.content
- assert response.data[0]["expired"]
- def test_post(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- },
- )
- assert response.status_code == 201, response.content
- assert response.data["name"] == "New query"
- assert response.data["projects"] == self.project_ids
- assert response.data["range"] == "24h"
- assert not hasattr(response.data, "start")
- assert not hasattr(response.data, "end")
- def test_post_invalid_projects(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "New query",
- "projects": self.project_ids_without_access,
- "fields": [],
- "range": "24h",
- "limit": 20,
- "conditions": [],
- "aggregations": [],
- "orderby": "-time",
- },
- )
- assert response.status_code == 403, response.content
- def test_post_all_projects(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "All projects",
- "projects": [-1],
- "conditions": [],
- "fields": ["title", "count()"],
- "range": "24h",
- "orderby": "time",
- },
- )
- assert response.status_code == 201, response.content
- assert response.data["projects"] == [-1]
- assert response.data["name"] == "All projects"
- def test_post_cannot_use_version_two_fields(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": ["id"],
- "range": "24h",
- "limit": 20,
- "environment": ["dev"],
- "yAxis": ["count(id)"],
- "aggregations": [],
- "orderby": "-time",
- },
- )
- assert response.status_code == 400, response.content
- assert (
- "You cannot use the environment, yAxis attribute(s) with the selected version"
- == response.data["non_field_errors"][0]
- )
- class DiscoverSavedQueriesVersion2Test(DiscoverSavedQueryBase):
- feature_name = "organizations:discover-query"
- def setUp(self):
- super().setUp()
- self.url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- def test_post_invalid_conditions(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": ["title", "count()"],
- "range": "24h",
- "version": 2,
- "conditions": [["field", "=", "value"]],
- },
- )
- assert response.status_code == 400, response.content
- assert (
- "You cannot use the conditions attribute(s) with the selected version"
- == response.data["non_field_errors"][0]
- )
- def test_post_require_selected_fields(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "New query",
- "projects": self.project_ids,
- "fields": [],
- "range": "24h",
- "version": 2,
- },
- )
- assert response.status_code == 400, response.content
- assert "You must include at least one field." == response.data["non_field_errors"][0]
- def test_post_success(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "new query",
- "projects": self.project_ids,
- "fields": ["title", "count()", "project"],
- "environment": ["dev"],
- "query": "event.type:error browser.name:Firefox",
- "range": "24h",
- "yAxis": ["count(id)"],
- "display": "releases",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- data = response.data
- assert data["fields"] == ["title", "count()", "project"]
- assert data["range"] == "24h"
- assert data["environment"] == ["dev"]
- assert data["query"] == "event.type:error browser.name:Firefox"
- assert data["yAxis"] == ["count(id)"]
- assert data["display"] == "releases"
- assert data["version"] == 2
- def test_post_all_projects(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "New query",
- "projects": [-1],
- "fields": ["title", "count()"],
- "range": "24h",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert response.data["projects"] == [-1]
- def test_save_with_project(self):
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": self.project_ids,
- "fields": ["title", "count()"],
- "range": "24h",
- "query": f"project:{self.projects[0].slug}",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert DiscoverSavedQuery.objects.filter(name="project query").exists()
- def test_save_with_project_and_my_projects(self):
- team = self.create_team(organization=self.org, members=[self.user])
- project = self.create_project(organization=self.org, teams=[team])
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": [],
- "fields": ["title", "count()"],
- "range": "24h",
- "query": f"project:{project.slug}",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert DiscoverSavedQuery.objects.filter(name="project query").exists()
- def test_save_with_org_projects(self):
- project = self.create_project(organization=self.org)
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": [project.id],
- "fields": ["title", "count()"],
- "range": "24h",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert DiscoverSavedQuery.objects.filter(name="project query").exists()
- def test_save_with_team_project(self):
- team = self.create_team(organization=self.org, members=[self.user])
- project = self.create_project(organization=self.org, teams=[team])
- self.create_project(organization=self.org, teams=[team])
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": [project.id],
- "fields": ["title", "count()"],
- "range": "24h",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert DiscoverSavedQuery.objects.filter(name="project query").exists()
- def test_save_without_team(self):
- team = self.create_team(organization=self.org, members=[])
- self.create_project(organization=self.org, teams=[team])
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "without team query",
- "projects": [],
- "fields": ["title", "count()"],
- "range": "24h",
- "version": 2,
- },
- )
- assert response.status_code == 400
- assert "No Projects found, join a Team" == response.data["detail"]
- def test_save_with_team_and_without_project(self):
- team = self.create_team(organization=self.org, members=[self.user])
- self.create_project(organization=self.org, teams=[team])
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "with team query",
- "projects": [],
- "fields": ["title", "count()"],
- "range": "24h",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert DiscoverSavedQuery.objects.filter(name="with team query").exists()
- def test_save_with_wrong_projects(self):
- other_org = self.create_organization(owner=self.user)
- project = self.create_project(organization=other_org)
- project2 = self.create_project(organization=self.org)
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": [project.id],
- "fields": ["title", "count()"],
- "range": "24h",
- "query": f"project:{project.slug}",
- "version": 2,
- },
- )
- assert response.status_code == 403, response.content
- assert not DiscoverSavedQuery.objects.filter(name="project query").exists()
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": [project.id, project2.id],
- "fields": ["title", "count()"],
- "range": "24h",
- "query": f"project:{project.slug} project:{project2.slug}",
- "version": 2,
- },
- )
- assert response.status_code == 403, response.content
- assert not DiscoverSavedQuery.objects.filter(name="project query").exists()
- # Mix of wrong + valid
- with self.feature(self.feature_name):
- url = reverse("sentry-api-0-discover-saved-queries", args=[self.org.slug])
- response = self.client.post(
- url,
- {
- "name": "project query",
- "projects": [-1],
- "fields": ["title", "count()"],
- "range": "24h",
- "query": f"project:{project.slug} project:{project2.slug}",
- "version": 2,
- },
- )
- assert response.status_code == 400, response.content
- assert not DiscoverSavedQuery.objects.filter(name="project query").exists()
- def test_save_with_equation(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "Equation query",
- "projects": [-1],
- "fields": [
- "title",
- "equation|count_if(measurements.lcp,greater,4000) / count()",
- "count()",
- "count_if(measurements.lcp,greater,4000)",
- ],
- "orderby": "equation[0]",
- "range": "24h",
- "query": "title:1",
- "version": 2,
- },
- )
- assert response.status_code == 201, response.content
- assert DiscoverSavedQuery.objects.filter(name="Equation query").exists()
- def test_save_with_invalid_equation(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "Equation query",
- "projects": [-1],
- "fields": [
- "title",
- "equation|count_if(measurements.lcp,greater,4000) / 0",
- "count()",
- "count_if(measurements.lcp,greater,4000)",
- ],
- "orderby": "equation[0]",
- "range": "24h",
- "query": "title:1",
- "version": 2,
- },
- )
- assert response.status_code == 400, response.content
- assert not DiscoverSavedQuery.objects.filter(name="Equation query").exists()
- def test_save_invalid_query(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "Bad query",
- "projects": [-1],
- "fields": ["title", "count()"],
- "range": "24h",
- "query": "spaceAfterColon: 1",
- "version": 2,
- },
- )
- assert response.status_code == 400, response.content
- assert not DiscoverSavedQuery.objects.filter(name="Bad query").exists()
- def test_save_invalid_query_orderby(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "Bad query",
- "projects": [-1],
- "fields": ["title", "count()"],
- "orderby": "fake()",
- "range": "24h",
- "query": "title:1",
- "version": 2,
- },
- )
- assert response.status_code == 400, response.content
- assert not DiscoverSavedQuery.objects.filter(name="Bad query").exists()
- def test_save_query_long_name(self):
- with self.feature(self.feature_name):
- response = self.client.post(
- self.url,
- {
- "name": "Bad query" * 200,
- "projects": [-1],
- "fields": ["title", "count()"],
- "range": "24h",
- "query": "spaceAfterColon:1",
- "version": 2,
- },
- )
- assert response.status_code == 400, response.content
- assert not DiscoverSavedQuery.objects.filter(name="Bad query" * 200).exists()
|