diff --git a/latest_migrations.manifest b/latest_migrations.manifest index 90046f4109b84..7c9afa3620a30 100644 --- a/latest_migrations.manifest +++ b/latest_migrations.manifest @@ -5,7 +5,7 @@ contenttypes: 0002_remove_content_type_name ee: 0016_rolemembership_organization_member otp_static: 0002_throttling otp_totp: 0002_auto_20190420_0723 -posthog: 0502_team_session_recording_url_blocklist_config +posthog: 0503_remove_errortrackingissuefingerprint_issue_and_more sessions: 0001_initial social_django: 0010_uid_db_index two_factor: 0007_auto_20201201_1019 diff --git a/posthog/api/__init__.py b/posthog/api/__init__.py index 06c97742c3a27..cdcad75f521b4 100644 --- a/posthog/api/__init__.py +++ b/posthog/api/__init__.py @@ -496,12 +496,12 @@ def register_grandfathered_environment_nested_viewset( ["project_id"], ) -projects_router.register( - r"error_tracking", - error_tracking.ErrorTrackingGroupViewSet, - "project_error_tracking", - ["team_id"], -) +# projects_router.register( +# r"error_tracking", +# error_tracking.ErrorTrackingGroupViewSet, +# "project_error_tracking", +# ["team_id"], +# ) projects_router.register( r"comments", diff --git a/posthog/api/error_tracking.py b/posthog/api/error_tracking.py index e3b9852085e6f..339cb61e59437 100644 --- a/posthog/api/error_tracking.py +++ b/posthog/api/error_tracking.py @@ -1,19 +1,5 @@ -import json import structlog -from rest_framework import serializers, viewsets, status -from rest_framework.response import Response -from rest_framework.exceptions import ValidationError - -from django.db.models import QuerySet -from django.conf import settings -from django.utils.http import urlsafe_base64_decode - -from posthog.api.forbid_destroy_model import ForbidDestroyModel -from posthog.models.error_tracking import ErrorTrackingGroup -from posthog.api.routing import TeamAndOrgViewSetMixin -from posthog.api.utils import action -from posthog.storage import object_storage FIFTY_MEGABYTES = 50 * 1024 * 1024 @@ -24,48 +10,48 @@ class ObjectStorageUnavailable(Exception): pass -class ErrorTrackingGroupSerializer(serializers.ModelSerializer): - class Meta: - model = ErrorTrackingGroup - fields = ["assignee", "status"] - - -class ErrorTrackingGroupViewSet(TeamAndOrgViewSetMixin, ForbidDestroyModel, viewsets.ModelViewSet): - scope_object = "INTERNAL" - queryset = ErrorTrackingGroup.objects.all() - serializer_class = ErrorTrackingGroupSerializer - - def safely_get_object(self, queryset) -> QuerySet: - stringified_fingerprint = self.kwargs["pk"] - fingerprint = json.loads(urlsafe_base64_decode(stringified_fingerprint)) - group, _ = queryset.get_or_create(fingerprint=fingerprint, team=self.team) - return group - - @action(methods=["POST"], detail=True) - def merge(self, request, **kwargs): - group: ErrorTrackingGroup = self.get_object() - merging_fingerprints: list[list[str]] = request.data.get("merging_fingerprints", []) - group.merge(merging_fingerprints) - return Response({"success": True}) - - @action(methods=["POST"], detail=False) - def upload_source_maps(self, request, **kwargs): - try: - if settings.OBJECT_STORAGE_ENABLED: - file = request.FILES["source_map"] - if file.size > FIFTY_MEGABYTES: - raise ValidationError(code="file_too_large", detail="Source maps must be less than 50MB") - - upload_path = ( - f"{settings.OBJECT_STORAGE_ERROR_TRACKING_SOURCE_MAPS_FOLDER}/team-{self.team_id}/{file.name}" - ) - - object_storage.write(upload_path, file) - return Response({"ok": True}, status=status.HTTP_201_CREATED) - else: - raise ObjectStorageUnavailable() - except ObjectStorageUnavailable: - raise ValidationError( - code="object_storage_required", - detail="Object storage must be available to allow source map uploads.", - ) +# class ErrorTrackingGroupSerializer(serializers.ModelSerializer): +# class Meta: +# model = ErrorTrackingGroup +# fields = ["assignee", "status"] + + +# class ErrorTrackingGroupViewSet(TeamAndOrgViewSetMixin, ForbidDestroyModel, viewsets.ModelViewSet): +# scope_object = "INTERNAL" +# queryset = ErrorTrackingGroup.objects.all() +# serializer_class = ErrorTrackingGroupSerializer + +# def safely_get_object(self, queryset) -> QuerySet: +# stringified_fingerprint = self.kwargs["pk"] +# fingerprint = json.loads(urlsafe_base64_decode(stringified_fingerprint)) +# group, _ = queryset.get_or_create(fingerprint=fingerprint, team=self.team) +# return group + +# @action(methods=["POST"], detail=True) +# def merge(self, request, **kwargs): +# group: ErrorTrackingGroup = self.get_object() +# merging_fingerprints: list[list[str]] = request.data.get("merging_fingerprints", []) +# group.merge(merging_fingerprints) +# return Response({"success": True}) + +# @action(methods=["POST"], detail=False) +# def upload_source_maps(self, request, **kwargs): +# try: +# if settings.OBJECT_STORAGE_ENABLED: +# file = request.FILES["source_map"] +# if file.size > FIFTY_MEGABYTES: +# raise ValidationError(code="file_too_large", detail="Source maps must be less than 50MB") + +# upload_path = ( +# f"{settings.OBJECT_STORAGE_ERROR_TRACKING_SOURCE_MAPS_FOLDER}/team-{self.team_id}/{file.name}" +# ) + +# object_storage.write(upload_path, file) +# return Response({"ok": True}, status=status.HTTP_201_CREATED) +# else: +# raise ObjectStorageUnavailable() +# except ObjectStorageUnavailable: +# raise ValidationError( +# code="object_storage_required", +# detail="Object storage must be available to allow source map uploads.", +# ) diff --git a/posthog/api/test/test_error_tracking.py b/posthog/api/test/test_error_tracking.py index fdd4569488327..dd53f10f27c44 100644 --- a/posthog/api/test/test_error_tracking.py +++ b/posthog/api/test/test_error_tracking.py @@ -1,21 +1,5 @@ import os -import json -from boto3 import resource -from rest_framework import status - -from django.utils.http import urlsafe_base64_encode -from django.core.files.uploadedfile import SimpleUploadedFile -from django.test import override_settings - -from posthog.test.base import APIBaseTest -from posthog.models import Team, ErrorTrackingGroup -from botocore.config import Config -from posthog.settings import ( - OBJECT_STORAGE_ENDPOINT, - OBJECT_STORAGE_ACCESS_KEY_ID, - OBJECT_STORAGE_SECRET_ACCESS_KEY, - OBJECT_STORAGE_BUCKET, -) + TEST_BUCKET = "test_storage_bucket-TestErrorTracking" @@ -25,108 +9,108 @@ def get_path_to(fixture_file: str) -> str: return os.path.join(file_dir, "fixtures", fixture_file) -class TestErrorTracking(APIBaseTest): - def teardown_method(self, method) -> None: - s3 = resource( - "s3", - endpoint_url=OBJECT_STORAGE_ENDPOINT, - aws_access_key_id=OBJECT_STORAGE_ACCESS_KEY_ID, - aws_secret_access_key=OBJECT_STORAGE_SECRET_ACCESS_KEY, - config=Config(signature_version="s3v4"), - region_name="us-east-1", - ) - bucket = s3.Bucket(OBJECT_STORAGE_BUCKET) - bucket.objects.filter(Prefix=TEST_BUCKET).delete() - - def send_request(self, fingerprint, data, endpoint=""): - base64_fingerprint = urlsafe_base64_encode(json.dumps(fingerprint).encode("utf-8")) - request_method = self.client.patch if endpoint == "" else self.client.post - request_method( - f"/api/projects/{self.team.id}/error_tracking/{base64_fingerprint}/{endpoint}", - data=data, - ) - - def test_reuses_existing_group_for_team(self): - fingerprint = ["CustomFingerprint"] - ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=self.team) - - self.assertEqual(ErrorTrackingGroup.objects.count(), 1) - self.send_request(fingerprint, {"assignee": self.user.id}) - self.assertEqual(ErrorTrackingGroup.objects.count(), 1) - - def test_creates_group_if_not_already_existing_for_team(self): - fingerprint = ["CustomFingerprint"] - other_team = Team.objects.create(organization=self.organization) - ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=other_team) - - self.assertEqual(ErrorTrackingGroup.objects.count(), 1) - self.send_request(fingerprint, {"assignee": self.user.id}) - self.assertEqual(ErrorTrackingGroup.objects.count(), 2) - - def test_can_only_update_allowed_fields(self): - fingerprint = ["CustomFingerprint"] - other_team = Team.objects.create(organization=self.organization) - group = ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=other_team) - - self.send_request(fingerprint, {"fingerprint": ["NewFingerprint"], "assignee": self.user.id}) - group.refresh_from_db() - self.assertEqual(group.fingerprint, ["CustomFingerprint"]) - - def test_merging_of_an_existing_group(self): - fingerprint = ["CustomFingerprint"] - merging_fingerprints = [["NewFingerprint"]] - group = ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=self.team) - - self.send_request(fingerprint, {"merging_fingerprints": merging_fingerprints}, endpoint="merge") - - group.refresh_from_db() - self.assertEqual(group.merged_fingerprints, merging_fingerprints) - - def test_merging_when_no_group_exists(self): - fingerprint = ["CustomFingerprint"] - merging_fingerprints = [["NewFingerprint"]] - - self.assertEqual(ErrorTrackingGroup.objects.count(), 0) - self.send_request(fingerprint, {"merging_fingerprints": merging_fingerprints}, endpoint="merge") - self.assertEqual(ErrorTrackingGroup.objects.count(), 1) - groups = ErrorTrackingGroup.objects.only("merged_fingerprints") - self.assertEqual(groups[0].merged_fingerprints, merging_fingerprints) - - def test_can_upload_a_source_map(self) -> None: - with self.settings(OBJECT_STORAGE_ENABLED=True, OBJECT_STORAGE_ERROR_TRACKING_SOURCE_MAPS_FOLDER=TEST_BUCKET): - with open(get_path_to("source.js.map"), "rb") as image: - response = self.client.post( - f"/api/projects/{self.team.id}/error_tracking/upload_source_maps", - {"source_map": image}, - format="multipart", - ) - self.assertEqual(response.status_code, status.HTTP_201_CREATED, response.json()) - - def test_rejects_too_large_file_type(self) -> None: - fifty_megabytes_plus_a_little = b"1" * (50 * 1024 * 1024 + 1) - fake_big_file = SimpleUploadedFile( - name="large_source.js.map", - content=fifty_megabytes_plus_a_little, - content_type="text/plain", - ) - response = self.client.post( - f"/api/projects/{self.team.id}/error_tracking/upload_source_maps", - {"source_map": fake_big_file}, - format="multipart", - ) - self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST, response.json()) - self.assertEqual(response.json()["detail"], "Source maps must be less than 50MB") - - def test_rejects_upload_when_object_storage_is_unavailable(self) -> None: - with override_settings(OBJECT_STORAGE_ENABLED=False): - fake_big_file = SimpleUploadedFile(name="large_source.js.map", content=b"", content_type="text/plain") - response = self.client.post( - f"/api/projects/{self.team.id}/error_tracking/upload_source_maps", - {"source_map": fake_big_file}, - format="multipart", - ) - self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST, response.json()) - self.assertEqual( - response.json()["detail"], - "Object storage must be available to allow source map uploads.", - ) +# class TestErrorTracking(APIBaseTest): +# def teardown_method(self, method) -> None: +# s3 = resource( +# "s3", +# endpoint_url=OBJECT_STORAGE_ENDPOINT, +# aws_access_key_id=OBJECT_STORAGE_ACCESS_KEY_ID, +# aws_secret_access_key=OBJECT_STORAGE_SECRET_ACCESS_KEY, +# config=Config(signature_version="s3v4"), +# region_name="us-east-1", +# ) +# bucket = s3.Bucket(OBJECT_STORAGE_BUCKET) +# bucket.objects.filter(Prefix=TEST_BUCKET).delete() + +# def send_request(self, fingerprint, data, endpoint=""): +# base64_fingerprint = urlsafe_base64_encode(json.dumps(fingerprint).encode("utf-8")) +# request_method = self.client.patch if endpoint == "" else self.client.post +# request_method( +# f"/api/projects/{self.team.id}/error_tracking/{base64_fingerprint}/{endpoint}", +# data=data, +# ) + +# def test_reuses_existing_group_for_team(self): +# fingerprint = ["CustomFingerprint"] +# ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=self.team) + +# self.assertEqual(ErrorTrackingGroup.objects.count(), 1) +# self.send_request(fingerprint, {"assignee": self.user.id}) +# self.assertEqual(ErrorTrackingGroup.objects.count(), 1) + +# def test_creates_group_if_not_already_existing_for_team(self): +# fingerprint = ["CustomFingerprint"] +# other_team = Team.objects.create(organization=self.organization) +# ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=other_team) + +# self.assertEqual(ErrorTrackingGroup.objects.count(), 1) +# self.send_request(fingerprint, {"assignee": self.user.id}) +# self.assertEqual(ErrorTrackingGroup.objects.count(), 2) + +# def test_can_only_update_allowed_fields(self): +# fingerprint = ["CustomFingerprint"] +# other_team = Team.objects.create(organization=self.organization) +# group = ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=other_team) + +# self.send_request(fingerprint, {"fingerprint": ["NewFingerprint"], "assignee": self.user.id}) +# group.refresh_from_db() +# self.assertEqual(group.fingerprint, ["CustomFingerprint"]) + +# def test_merging_of_an_existing_group(self): +# fingerprint = ["CustomFingerprint"] +# merging_fingerprints = [["NewFingerprint"]] +# group = ErrorTrackingGroup.objects.create(fingerprint=fingerprint, team=self.team) + +# self.send_request(fingerprint, {"merging_fingerprints": merging_fingerprints}, endpoint="merge") + +# group.refresh_from_db() +# self.assertEqual(group.merged_fingerprints, merging_fingerprints) + +# def test_merging_when_no_group_exists(self): +# fingerprint = ["CustomFingerprint"] +# merging_fingerprints = [["NewFingerprint"]] + +# self.assertEqual(ErrorTrackingGroup.objects.count(), 0) +# self.send_request(fingerprint, {"merging_fingerprints": merging_fingerprints}, endpoint="merge") +# self.assertEqual(ErrorTrackingGroup.objects.count(), 1) +# groups = ErrorTrackingGroup.objects.only("merged_fingerprints") +# self.assertEqual(groups[0].merged_fingerprints, merging_fingerprints) + +# def test_can_upload_a_source_map(self) -> None: +# with self.settings(OBJECT_STORAGE_ENABLED=True, OBJECT_STORAGE_ERROR_TRACKING_SOURCE_MAPS_FOLDER=TEST_BUCKET): +# with open(get_path_to("source.js.map"), "rb") as image: +# response = self.client.post( +# f"/api/projects/{self.team.id}/error_tracking/upload_source_maps", +# {"source_map": image}, +# format="multipart", +# ) +# self.assertEqual(response.status_code, status.HTTP_201_CREATED, response.json()) + +# def test_rejects_too_large_file_type(self) -> None: +# fifty_megabytes_plus_a_little = b"1" * (50 * 1024 * 1024 + 1) +# fake_big_file = SimpleUploadedFile( +# name="large_source.js.map", +# content=fifty_megabytes_plus_a_little, +# content_type="text/plain", +# ) +# response = self.client.post( +# f"/api/projects/{self.team.id}/error_tracking/upload_source_maps", +# {"source_map": fake_big_file}, +# format="multipart", +# ) +# self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST, response.json()) +# self.assertEqual(response.json()["detail"], "Source maps must be less than 50MB") + +# def test_rejects_upload_when_object_storage_is_unavailable(self) -> None: +# with override_settings(OBJECT_STORAGE_ENABLED=False): +# fake_big_file = SimpleUploadedFile(name="large_source.js.map", content=b"", content_type="text/plain") +# response = self.client.post( +# f"/api/projects/{self.team.id}/error_tracking/upload_source_maps", +# {"source_map": fake_big_file}, +# format="multipart", +# ) +# self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST, response.json()) +# self.assertEqual( +# response.json()["detail"], +# "Object storage must be available to allow source map uploads.", +# ) diff --git a/posthog/hogql_queries/error_tracking_query_runner.py b/posthog/hogql_queries/error_tracking_query_runner.py index f32b55a4a078d..b277f6a5c7c47 100644 --- a/posthog/hogql_queries/error_tracking_query_runner.py +++ b/posthog/hogql_queries/error_tracking_query_runner.py @@ -10,7 +10,7 @@ CachedErrorTrackingQueryResponse, ) from posthog.hogql.parser import parse_expr -from posthog.models.error_tracking import ErrorTrackingGroup + from posthog.models.filters.mixins.utils import cached_property @@ -240,7 +240,7 @@ def group_or_default(self, fingerprint): "fingerprint": fingerprint, "assignee": None, "merged_fingerprints": [], - "status": str(ErrorTrackingGroup.Status.ACTIVE), + # "status": str(ErrorTrackingGroup.Status.ACTIVE), }, ) @@ -269,18 +269,19 @@ def extracted_fingerprint_property(self): @cached_property def error_tracking_groups(self): - queryset = ErrorTrackingGroup.objects.filter(team=self.team) - # :TRICKY: Ideally we'd have no null characters in the fingerprint, but if something made it into the pipeline with null characters - # (because rest of the system supports it), try cleaning it up here. Make sure this cleaning is consistent with the rest of the system. - cleaned_fingerprint = [part.replace("\x00", "\ufffd") for part in self.query.fingerprint or []] - queryset = ( - queryset.filter(fingerprint=cleaned_fingerprint) - if self.query.fingerprint - else queryset.filter(status__in=[ErrorTrackingGroup.Status.ACTIVE]) - ) - queryset = queryset.filter(assignee=self.query.assignee) if self.query.assignee else queryset - groups = queryset.values("fingerprint", "merged_fingerprints", "status", "assignee") - return {str(item["fingerprint"]): item for item in groups} + return {} + # queryset = ErrorTrackingGroup.objects.filter(team=self.team) + # # :TRICKY: Ideally we'd have no null characters in the fingerprint, but if something made it into the pipeline with null characters + # # (because rest of the system supports it), try cleaning it up here. Make sure this cleaning is consistent with the rest of the system. + # cleaned_fingerprint = [part.replace("\x00", "\ufffd") for part in self.query.fingerprint or []] + # queryset = ( + # queryset.filter(fingerprint=cleaned_fingerprint) + # if self.query.fingerprint + # else queryset.filter(status__in=[ErrorTrackingGroup.Status.ACTIVE]) + # ) + # queryset = queryset.filter(assignee=self.query.assignee) if self.query.assignee else queryset + # groups = queryset.values("fingerprint", "merged_fingerprints", "status", "assignee") + # return {str(item["fingerprint"]): item for item in groups} def search_tokenizer(query: str) -> list[str]: diff --git a/posthog/hogql_queries/test/test_error_tracking_query_runner.py b/posthog/hogql_queries/test/test_error_tracking_query_runner.py index 0a17c061f3f50..de12f9a7187ba 100644 --- a/posthog/hogql_queries/test/test_error_tracking_query_runner.py +++ b/posthog/hogql_queries/test/test_error_tracking_query_runner.py @@ -1,27 +1,13 @@ -from unittest import TestCase from freezegun import freeze_time -from posthog.hogql_queries.error_tracking_query_runner import ErrorTrackingQueryRunner, search_tokenizer -from posthog.schema import ( - ErrorTrackingQuery, - DateRange, - FilterLogicalOperator, - PropertyGroupFilter, - PropertyGroupFilterValue, - PersonPropertyFilter, - PropertyOperator, -) +from posthog.hogql_queries.error_tracking_query_runner import ErrorTrackingQueryRunner from posthog.test.base import ( APIBaseTest, ClickhouseTestMixin, - snapshot_clickhouse_queries, _create_person, _create_event, flush_persons_and_events, ) -from posthog.models import ErrorTrackingGroup -from datetime import datetime -from zoneinfo import ZoneInfo SAMPLE_STACK_TRACE = [ { @@ -250,456 +236,457 @@ def setUp(self): def _calculate(self, runner: ErrorTrackingQueryRunner): return runner.calculate().model_dump() - @snapshot_clickhouse_queries - def test_column_names(self): - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=None, - dateRange=DateRange(), - filterTestAccounts=True, - ), - ) - - columns = self._calculate(runner)["columns"] - self.assertEqual( - columns, - [ - "occurrences", - "sessions", - "users", - "last_seen", - "first_seen", - "description", - "exception_type", - "fingerprint", - ], - ) - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=["SyntaxError"], - dateRange=DateRange(), - filterTestAccounts=True, - ), - ) - - columns = self._calculate(runner)["columns"] - self.assertEqual( - columns, - [ - "occurrences", - "sessions", - "users", - "last_seen", - "first_seen", - "description", - "exception_type", - ], - ) - - @snapshot_clickhouse_queries - def test_search_query(self): - with freeze_time("2022-01-10 12:11:00"): - _create_event( - distinct_id=self.distinct_id_one, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": ["DatabaseNotFoundX"], - "$exception_type": "DatabaseNotFoundX", - "$exception_message": "this is the same error message", - }, - ) - _create_event( - distinct_id=self.distinct_id_one, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": ["DatabaseNotFoundY"], - "$exception_type": "DatabaseNotFoundY", - "$exception_message": "this is the same error message", - }, - ) - _create_event( - distinct_id=self.distinct_id_two, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": ["xyz"], - "$exception_type": "xyz", - "$exception_message": "this is the same error message", - }, - ) - flush_persons_and_events() - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=None, - dateRange=DateRange(date_from="2022-01-10", date_to="2022-01-11"), - filterTestAccounts=True, - searchQuery="databasenot", - ), - ) - - results = sorted(self._calculate(runner)["results"], key=lambda x: x["fingerprint"]) - - self.assertEqual(len(results), 2) - self.assertEqual(results[0]["fingerprint"], ["DatabaseNotFoundX"]) - self.assertEqual(results[0]["occurrences"], 1) - self.assertEqual(results[0]["sessions"], 1) - self.assertEqual(results[0]["users"], 1) - - self.assertEqual(results[1]["fingerprint"], ["DatabaseNotFoundY"]) - self.assertEqual(results[1]["occurrences"], 1) - self.assertEqual(results[1]["sessions"], 1) - self.assertEqual(results[1]["users"], 1) - - def test_empty_search_query(self): - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=None, - dateRange=DateRange(), - filterTestAccounts=False, - searchQuery="probs not found", - ), - ) - - results = self._calculate(runner)["results"] - - self.assertEqual(len(results), 0) - - @snapshot_clickhouse_queries - def test_search_query_with_multiple_search_items(self): - with freeze_time("2022-01-10 12:11:00"): - _create_event( - distinct_id=self.distinct_id_one, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": ["DatabaseNotFoundX"], - "$exception_type": "DatabaseNotFoundX", - "$exception_message": "this is the same error message", - "$exception_list": [{"stack_trace": {"frames": SAMPLE_STACK_TRACE}}], - }, - ) - - _create_event( - distinct_id=self.distinct_id_two, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": ["DatabaseNotFoundY"], - "$exception_type": "DatabaseNotFoundY", - "$exception_message": "this is the same error message", - "$exception_list": [{"stack_trace": {"frames": SAMPLE_STACK_TRACE}}], - }, - ) - flush_persons_and_events() - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=None, - dateRange=DateRange(), - filterTestAccounts=True, - searchQuery="databasenotfoundX clickhouse/client/execute.py", - ), - ) - - results = self._calculate(runner)["results"] - - self.assertEqual(len(results), 1) - self.assertEqual(results[0]["fingerprint"], ["DatabaseNotFoundX"]) - self.assertEqual(results[0]["occurrences"], 1) - self.assertEqual(results[0]["sessions"], 1) - self.assertEqual(results[0]["users"], 1) - - @snapshot_clickhouse_queries - def test_search_query_with_null_characters(self): - fingerprint_with_null_bytes = [ - "SyntaxError", - "Cannot use 'in' operator to search for 'wireframes' in \x1f\x8b\x08\x00\x94\x0cýf\x00\x03ì½é\x96\"¹\x920ø*Lö¹SY\x1dA\x00Î\x9e÷Ô\x9df\r\x88\x00Ø", - ] - exception_type_with_null_bytes = "SyntaxError\x00" - exception_message_with_null_bytes = "this is the same error message\x00" - exception_stack_trace_with_null_bytes = { - "frames": [ - { - "filename": "file.py\x00", - "lineno": 1, - "colno": 1, - "function": "function\x00", - "extra": "Cannot use 'in' operator to search for 'wireframes' in \x1f\x8b\x08\x00\x94\x0cýf\x00\x03ì½é\x96\"¹\x920ø*Lö¹SY\x1dA\x00Î\x9e÷Ô\x9df\r\x88\x00Ø", - } - ] - } - with freeze_time("2021-01-10 12:11:00"): - _create_event( - distinct_id=self.distinct_id_one, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": fingerprint_with_null_bytes, - "$exception_type": exception_type_with_null_bytes, - "$exception_message": exception_message_with_null_bytes, - "$exception_list": [{"stack_trace": exception_stack_trace_with_null_bytes}], - }, - ) - flush_persons_and_events() - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - searchQuery="wireframe", - dateRange=DateRange(date_from="2021-01-10", date_to="2021-01-11"), - ), - ) - - results = self._calculate(runner)["results"] - self.assertEqual(len(results), 1) - self.assertEqual(results[0]["fingerprint"], fingerprint_with_null_bytes) - self.assertEqual(results[0]["occurrences"], 1) - - # TODO: Searching for null characters doesn't work, probs because of how clickhouse handles this. Should it work??? - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - searchQuery="f\x00\x03ì½é", - dateRange=DateRange(date_from="2021-01-10", date_to="2021-01-11"), - ), - ) - results = self._calculate(runner)["results"] - self.assertEqual(len(results), 0) - - @snapshot_clickhouse_queries - def test_fingerprints(self): - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=["SyntaxError"], - dateRange=DateRange(), - ), - ) - - results = self._calculate(runner)["results"] - # returns a single group with multiple errors - self.assertEqual(len(results), 1) - self.assertEqual(results[0]["fingerprint"], ["SyntaxError"]) - self.assertEqual(results[0]["occurrences"], 2) - - @snapshot_clickhouse_queries - def test_fingerprints_with_null_characters(self): - fingerprint_with_null_bytes = [ - "SyntaxError", - "Cannot use 'in' operator to search for 'wireframes' in \x1f\x8b\x08\x00\x94\x0cýf\x00\x03ì½é\x96\"\x00Ø", - ] - exception_type_with_null_bytes = "SyntaxError\x00" - exception_message_with_null_bytes = "this is the same error message\x00" - exception_stack_trace_with_null_bytes = { - "frames": [{"filename": "file.py\x00", "lineno": 1, "colno": 1, "function": "function\x00"}] - } - with freeze_time("2020-01-10 12:11:00"): - _create_event( - distinct_id=self.distinct_id_one, - event="$exception", - team=self.team, - properties={ - "$exception_fingerprint": fingerprint_with_null_bytes, - "$exception_type": exception_type_with_null_bytes, - "$exception_message": exception_message_with_null_bytes, - "$exception_list": [{"stack_trace": exception_stack_trace_with_null_bytes}], - }, - ) - flush_persons_and_events() - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - fingerprint=fingerprint_with_null_bytes, - dateRange=DateRange(), - ), - ) - - results = self._calculate(runner)["results"] - self.assertEqual(len(results), 1) - self.assertEqual(results[0]["fingerprint"], fingerprint_with_null_bytes) - self.assertEqual(results[0]["occurrences"], 1) - - def test_only_returns_exception_events(self): - with freeze_time("2020-01-10 12:11:00"): - _create_event( - distinct_id=self.distinct_id_one, - event="$pageview", - team=self.team, - properties={ - "$exception_fingerprint": ["SyntaxError"], - }, - ) - flush_persons_and_events() - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - dateRange=DateRange(), - ), - ) - - results = self._calculate(runner)["results"] - self.assertEqual(len(results), 3) - - @snapshot_clickhouse_queries - def test_hogql_filters(self): - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - dateRange=DateRange(), - filterGroup=PropertyGroupFilter( - type=FilterLogicalOperator.AND_, - values=[ - PropertyGroupFilterValue( - type=FilterLogicalOperator.OR_, - values=[ - PersonPropertyFilter( - key="email", value="email@posthog.com", operator=PropertyOperator.EXACT - ), - ], - ) - ], - ), - ), - ) - - results = self._calculate(runner)["results"] - # two errors exist for person with distinct_id_two - self.assertEqual(len(results), 2) - - def test_merges_and_defaults_groups(self): - ErrorTrackingGroup.objects.create( - team=self.team, - fingerprint=["SyntaxError"], - merged_fingerprints=[["custom_fingerprint"]], - assignee=self.user, - ) - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", fingerprint=None, dateRange=DateRange(), order="occurrences" - ), - ) - - results = self._calculate(runner)["results"] - self.assertEqual( - results, - [ - { - "assignee": self.user.id, - "description": "this is the same error message", - "exception_type": "SyntaxError", - "fingerprint": ["SyntaxError"], - "first_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), - "last_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), - "merged_fingerprints": [["custom_fingerprint"]], - # count is (2 x SyntaxError) + (1 x custom_fingerprint) - "occurrences": 3, - "sessions": 1, - "users": 2, - "volume": None, - "status": ErrorTrackingGroup.Status.ACTIVE, - }, - { - "assignee": None, - "description": None, - "exception_type": "TypeError", - "fingerprint": ["TypeError"], - "first_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), - "last_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), - "merged_fingerprints": [], - "occurrences": 1, - "sessions": 1, - "users": 1, - "volume": None, - "status": ErrorTrackingGroup.Status.ACTIVE, - }, - ], - ) - - @snapshot_clickhouse_queries - def test_assignee_groups(self): - ErrorTrackingGroup.objects.create( - team=self.team, - fingerprint=["SyntaxError"], - assignee=self.user, - ) - ErrorTrackingGroup.objects.create( - team=self.team, - fingerprint=["custom_fingerprint"], - assignee=self.user, - ) - ErrorTrackingGroup.objects.create( - team=self.team, - fingerprint=["TypeError"], - ) - - runner = ErrorTrackingQueryRunner( - team=self.team, - query=ErrorTrackingQuery( - kind="ErrorTrackingQuery", - dateRange=DateRange(), - assignee=self.user.pk, - ), - ) - - results = self._calculate(runner)["results"] - - self.assertEqual(sorted([x["fingerprint"] for x in results]), [["SyntaxError"], ["custom_fingerprint"]]) - - -class TestSearchTokenizer(TestCase): - test_cases = [ - ( - "This is a \"quoted string\" and this is 'another one' with some words", - ["This", "is", "a", "quoted string", "and", "this", "is", "another one", "with", "some", "words"], - ), - ( - "Empty quotes: \"\" and '' should be preserved", - ["Empty", "quotes:", "", "and", "", "should", "be", "preserved"], - ), - ("Nested \"quotes 'are' tricky\" to handle", ["Nested", "quotes 'are' tricky", "to", "handle"]), - ( - "Unmatched quotes: \"open quote and 'partial quote", - ["Unmatched", "quotes:", "open", "quote", "and", "partial", "quote"], - ), - ("Multiple spaces between words", ["Multiple", "spaces", "between", "words"]), - ( - "Special characters: @#$% should be treated as words", - ["Special", "characters:", "@#$%", "should", "be", "treated", "as", "words"], - ), - ( - "Single quotes at \"start\" and 'end' of string", - ["Single", "quotes", "at", "start", "and", "end", "of", "string"], - ), - ('"Entire string is quoted"', ["Entire string is quoted"]), - ('Escaped quotes: "He said "Hello" to me"', ["Escaped", "quotes:", "He said ", "Hello", "to", "me"]), - ] - - def test_tokenizer(self): - for case, output in self.test_cases: - with self.subTest(case=case): - tokens = search_tokenizer(case) - self.assertEqual(tokens, output) +# @snapshot_clickhouse_queries +# def test_column_names(self): +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=None, +# dateRange=DateRange(), +# filterTestAccounts=True, +# ), +# ) + +# columns = self._calculate(runner)["columns"] +# self.assertEqual( +# columns, +# [ +# "occurrences", +# "sessions", +# "users", +# "last_seen", +# "first_seen", +# "description", +# "exception_type", +# "fingerprint", +# ], +# ) + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=["SyntaxError"], +# dateRange=DateRange(), +# filterTestAccounts=True, +# ), +# ) + +# columns = self._calculate(runner)["columns"] +# self.assertEqual( +# columns, +# [ +# "occurrences", +# "sessions", +# "users", +# "last_seen", +# "first_seen", +# "description", +# "exception_type", +# ], +# ) + +# @snapshot_clickhouse_queries +# def test_search_query(self): +# with freeze_time("2022-01-10 12:11:00"): +# _create_event( +# distinct_id=self.distinct_id_one, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": ["DatabaseNotFoundX"], +# "$exception_type": "DatabaseNotFoundX", +# "$exception_message": "this is the same error message", +# }, +# ) +# _create_event( +# distinct_id=self.distinct_id_one, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": ["DatabaseNotFoundY"], +# "$exception_type": "DatabaseNotFoundY", +# "$exception_message": "this is the same error message", +# }, +# ) +# _create_event( +# distinct_id=self.distinct_id_two, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": ["xyz"], +# "$exception_type": "xyz", +# "$exception_message": "this is the same error message", +# }, +# ) +# flush_persons_and_events() + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=None, +# dateRange=DateRange(date_from="2022-01-10", date_to="2022-01-11"), +# filterTestAccounts=True, +# searchQuery="databasenot", +# ), +# ) + +# results = sorted(self._calculate(runner)["results"], key=lambda x: x["fingerprint"]) + +# self.assertEqual(len(results), 2) +# self.assertEqual(results[0]["fingerprint"], ["DatabaseNotFoundX"]) +# self.assertEqual(results[0]["occurrences"], 1) +# self.assertEqual(results[0]["sessions"], 1) +# self.assertEqual(results[0]["users"], 1) + +# self.assertEqual(results[1]["fingerprint"], ["DatabaseNotFoundY"]) +# self.assertEqual(results[1]["occurrences"], 1) +# self.assertEqual(results[1]["sessions"], 1) +# self.assertEqual(results[1]["users"], 1) + +# def test_empty_search_query(self): +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=None, +# dateRange=DateRange(), +# filterTestAccounts=False, +# searchQuery="probs not found", +# ), +# ) + +# results = self._calculate(runner)["results"] + +# self.assertEqual(len(results), 0) + +# @snapshot_clickhouse_queries +# def test_search_query_with_multiple_search_items(self): +# with freeze_time("2022-01-10 12:11:00"): +# _create_event( +# distinct_id=self.distinct_id_one, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": ["DatabaseNotFoundX"], +# "$exception_type": "DatabaseNotFoundX", +# "$exception_message": "this is the same error message", +# "$exception_list": [{"stack_trace": {"frames": SAMPLE_STACK_TRACE}}], +# }, +# ) + +# _create_event( +# distinct_id=self.distinct_id_two, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": ["DatabaseNotFoundY"], +# "$exception_type": "DatabaseNotFoundY", +# "$exception_message": "this is the same error message", +# "$exception_list": [{"stack_trace": {"frames": SAMPLE_STACK_TRACE}}], +# }, +# ) +# flush_persons_and_events() + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=None, +# dateRange=DateRange(), +# filterTestAccounts=True, +# searchQuery="databasenotfoundX clickhouse/client/execute.py", +# ), +# ) + +# results = self._calculate(runner)["results"] + +# self.assertEqual(len(results), 1) +# self.assertEqual(results[0]["fingerprint"], ["DatabaseNotFoundX"]) +# self.assertEqual(results[0]["occurrences"], 1) +# self.assertEqual(results[0]["sessions"], 1) +# self.assertEqual(results[0]["users"], 1) + +# @snapshot_clickhouse_queries +# def test_search_query_with_null_characters(self): +# fingerprint_with_null_bytes = [ +# "SyntaxError", +# "Cannot use 'in' operator to search for 'wireframes' in \x1f\x8b\x08\x00\x94\x0cýf\x00\x03ì½é\x96\"¹\x920ø*Lö¹SY\x1dA\x00Î\x9e÷Ô\x9df\r\x88\x00Ø", +# ] +# exception_type_with_null_bytes = "SyntaxError\x00" +# exception_message_with_null_bytes = "this is the same error message\x00" +# exception_stack_trace_with_null_bytes = { +# "frames": [ +# { +# "filename": "file.py\x00", +# "lineno": 1, +# "colno": 1, +# "function": "function\x00", +# "extra": "Cannot use 'in' operator to search for 'wireframes' in \x1f\x8b\x08\x00\x94\x0cýf\x00\x03ì½é\x96\"¹\x920ø*Lö¹SY\x1dA\x00Î\x9e÷Ô\x9df\r\x88\x00Ø", +# } +# ] +# } +# with freeze_time("2021-01-10 12:11:00"): +# _create_event( +# distinct_id=self.distinct_id_one, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": fingerprint_with_null_bytes, +# "$exception_type": exception_type_with_null_bytes, +# "$exception_message": exception_message_with_null_bytes, +# "$exception_list": [{"stack_trace": exception_stack_trace_with_null_bytes}], +# }, +# ) +# flush_persons_and_events() + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# searchQuery="wireframe", +# dateRange=DateRange(date_from="2021-01-10", date_to="2021-01-11"), +# ), +# ) + +# results = self._calculate(runner)["results"] +# self.assertEqual(len(results), 1) +# self.assertEqual(results[0]["fingerprint"], fingerprint_with_null_bytes) +# self.assertEqual(results[0]["occurrences"], 1) + +# # TODO: Searching for null characters doesn't work, probs because of how clickhouse handles this. Should it work??? +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# searchQuery="f\x00\x03ì½é", +# dateRange=DateRange(date_from="2021-01-10", date_to="2021-01-11"), +# ), +# ) +# results = self._calculate(runner)["results"] +# self.assertEqual(len(results), 0) + +# @snapshot_clickhouse_queries +# def test_fingerprints(self): +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=["SyntaxError"], +# dateRange=DateRange(), +# ), +# ) + +# results = self._calculate(runner)["results"] +# # returns a single group with multiple errors +# self.assertEqual(len(results), 1) +# self.assertEqual(results[0]["fingerprint"], ["SyntaxError"]) +# self.assertEqual(results[0]["occurrences"], 2) + +# @snapshot_clickhouse_queries +# def test_fingerprints_with_null_characters(self): +# fingerprint_with_null_bytes = [ +# "SyntaxError", +# "Cannot use 'in' operator to search for 'wireframes' in \x1f\x8b\x08\x00\x94\x0cýf\x00\x03ì½é\x96\"\x00Ø", +# ] +# exception_type_with_null_bytes = "SyntaxError\x00" +# exception_message_with_null_bytes = "this is the same error message\x00" +# exception_stack_trace_with_null_bytes = { +# "frames": [{"filename": "file.py\x00", "lineno": 1, "colno": 1, "function": "function\x00"}] +# } +# with freeze_time("2020-01-10 12:11:00"): +# _create_event( +# distinct_id=self.distinct_id_one, +# event="$exception", +# team=self.team, +# properties={ +# "$exception_fingerprint": fingerprint_with_null_bytes, +# "$exception_type": exception_type_with_null_bytes, +# "$exception_message": exception_message_with_null_bytes, +# "$exception_list": [{"stack_trace": exception_stack_trace_with_null_bytes}], +# }, +# ) +# flush_persons_and_events() + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# fingerprint=fingerprint_with_null_bytes, +# dateRange=DateRange(), +# ), +# ) + +# results = self._calculate(runner)["results"] +# self.assertEqual(len(results), 1) +# self.assertEqual(results[0]["fingerprint"], fingerprint_with_null_bytes) +# self.assertEqual(results[0]["occurrences"], 1) + +# def test_only_returns_exception_events(self): +# with freeze_time("2020-01-10 12:11:00"): +# _create_event( +# distinct_id=self.distinct_id_one, +# event="$pageview", +# team=self.team, +# properties={ +# "$exception_fingerprint": ["SyntaxError"], +# }, +# ) +# flush_persons_and_events() + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# dateRange=DateRange(), +# ), +# ) + +# results = self._calculate(runner)["results"] +# self.assertEqual(len(results), 3) + +# @snapshot_clickhouse_queries +# def test_hogql_filters(self): +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# dateRange=DateRange(), +# filterGroup=PropertyGroupFilter( +# type=FilterLogicalOperator.AND_, +# values=[ +# PropertyGroupFilterValue( +# type=FilterLogicalOperator.OR_, +# values=[ +# PersonPropertyFilter( +# key="email", value="email@posthog.com", operator=PropertyOperator.EXACT +# ), +# ], +# ) +# ], +# ), +# ), +# ) + +# results = self._calculate(runner)["results"] +# # two errors exist for person with distinct_id_two +# self.assertEqual(len(results), 2) + +# def test_merges_and_defaults_groups(self): +# ErrorTrackingGroup.objects.create( +# team=self.team, +# fingerprint=["SyntaxError"], +# merged_fingerprints=[["custom_fingerprint"]], +# assignee=self.user, +# ) + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", fingerprint=None, dateRange=DateRange(), order="occurrences" +# ), +# ) + +# results = self._calculate(runner)["results"] +# self.assertEqual( +# results, +# [ +# { +# "assignee": self.user.id, +# "description": "this is the same error message", +# "exception_type": "SyntaxError", +# "fingerprint": ["SyntaxError"], +# "first_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), +# "last_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), +# "merged_fingerprints": [["custom_fingerprint"]], +# # count is (2 x SyntaxError) + (1 x custom_fingerprint) +# "occurrences": 3, +# "sessions": 1, +# "users": 2, +# "volume": None, +# "status": ErrorTrackingGroup.Status.ACTIVE, +# }, +# { +# "assignee": None, +# "description": None, +# "exception_type": "TypeError", +# "fingerprint": ["TypeError"], +# "first_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), +# "last_seen": datetime(2020, 1, 10, 12, 11, tzinfo=ZoneInfo("UTC")), +# "merged_fingerprints": [], +# "occurrences": 1, +# "sessions": 1, +# "users": 1, +# "volume": None, +# "status": ErrorTrackingGroup.Status.ACTIVE, +# }, +# ], +# ) + +# @snapshot_clickhouse_queries +# def test_assignee_groups(self): +# ErrorTrackingGroup.objects.create( +# team=self.team, +# fingerprint=["SyntaxError"], +# assignee=self.user, +# ) +# ErrorTrackingGroup.objects.create( +# team=self.team, +# fingerprint=["custom_fingerprint"], +# assignee=self.user, +# ) +# ErrorTrackingGroup.objects.create( +# team=self.team, +# fingerprint=["TypeError"], +# ) + +# runner = ErrorTrackingQueryRunner( +# team=self.team, +# query=ErrorTrackingQuery( +# kind="ErrorTrackingQuery", +# dateRange=DateRange(), +# assignee=self.user.pk, +# ), +# ) + +# results = self._calculate(runner)["results"] + +# self.assertEqual(sorted([x["fingerprint"] for x in results]), [["SyntaxError"], ["custom_fingerprint"]]) + + +# class TestSearchTokenizer(TestCase): +# test_cases = [ +# ( +# "This is a \"quoted string\" and this is 'another one' with some words", +# ["This", "is", "a", "quoted string", "and", "this", "is", "another one", "with", "some", "words"], +# ), +# ( +# "Empty quotes: \"\" and '' should be preserved", +# ["Empty", "quotes:", "", "and", "", "should", "be", "preserved"], +# ), +# ("Nested \"quotes 'are' tricky\" to handle", ["Nested", "quotes 'are' tricky", "to", "handle"]), +# ( +# "Unmatched quotes: \"open quote and 'partial quote", +# ["Unmatched", "quotes:", "open", "quote", "and", "partial", "quote"], +# ), +# ("Multiple spaces between words", ["Multiple", "spaces", "between", "words"]), +# ( +# "Special characters: @#$% should be treated as words", +# ["Special", "characters:", "@#$%", "should", "be", "treated", "as", "words"], +# ), +# ( +# "Single quotes at \"start\" and 'end' of string", +# ["Single", "quotes", "at", "start", "and", "end", "of", "string"], +# ), +# ('"Entire string is quoted"', ["Entire string is quoted"]), +# ('Escaped quotes: "He said "Hello" to me"', ["Escaped", "quotes:", "He said ", "Hello", "to", "me"]), +# ] + +# def test_tokenizer(self): +# for case, output in self.test_cases: +# with self.subTest(case=case): +# tokens = search_tokenizer(case) +# self.assertEqual(tokens, output) diff --git a/posthog/migrations/0503_remove_errortrackingissuefingerprint_issue_and_more.py b/posthog/migrations/0503_remove_errortrackingissuefingerprint_issue_and_more.py new file mode 100644 index 0000000000000..f92b50e1b7bea --- /dev/null +++ b/posthog/migrations/0503_remove_errortrackingissuefingerprint_issue_and_more.py @@ -0,0 +1,26 @@ +# Generated by Django 4.2.15 on 2024-10-30 17:37 + +from django.db import migrations + + +class Migration(migrations.Migration): + dependencies = [ + ("posthog", "0502_team_session_recording_url_blocklist_config"), + ] + + operations = [ + migrations.RemoveField( + model_name="errortrackingissuefingerprint", + name="issue", + ), + migrations.RemoveField( + model_name="errortrackingissuefingerprint", + name="team", + ), + migrations.DeleteModel( + name="ErrorTrackingGroup", + ), + migrations.DeleteModel( + name="ErrorTrackingIssueFingerprint", + ), + ] diff --git a/posthog/models/__init__.py b/posthog/models/__init__.py index 5a57fbe5c2c51..de52a34208ae7 100644 --- a/posthog/models/__init__.py +++ b/posthog/models/__init__.py @@ -29,7 +29,7 @@ from .element import Element from .element_group import ElementGroup from .entity import Entity -from .error_tracking import ErrorTrackingGroup +from .error_tracking import ErrorTrackingStackFrame, ErrorTrackingSymbolSet from .event.event import Event from .event_buffer import EventBuffer from .event_definition import EventDefinition @@ -99,7 +99,8 @@ "Element", "ElementGroup", "Entity", - "ErrorTrackingGroup", + "ErrorTrackingStackFrame", + "ErrorTrackingSymbolSet", "Event", "EventBuffer", "EventDefinition", @@ -117,6 +118,7 @@ "InsightViewed", "InstanceSetting", "Integration", + "InviteExpiredException", "MessagingRecord", "Notebook", "MigrationStatus", diff --git a/posthog/models/error_tracking/error_tracking.py b/posthog/models/error_tracking/error_tracking.py index 6cca164d2afe0..0edecf20c886e 100644 --- a/posthog/models/error_tracking/error_tracking.py +++ b/posthog/models/error_tracking/error_tracking.py @@ -1,74 +1,71 @@ from django.db import models -from django.contrib.postgres.fields import ArrayField from posthog.models.utils import UUIDModel -from django.db import transaction -from django.db.models import Q, QuerySet -class ErrorTrackingGroup(UUIDModel): - class Status(models.TextChoices): - ARCHIVED = "archived", "Archived" - ACTIVE = "active", "Active" - RESOLVED = "resolved", "Resolved" - PENDING_RELEASE = "pending_release", "Pending release" - - team = models.ForeignKey("Team", on_delete=models.CASCADE) - created_at = models.DateTimeField(auto_now_add=True, blank=True) - fingerprint: ArrayField = ArrayField(models.TextField(null=False, blank=False), null=False, blank=False) - merged_fingerprints: ArrayField = ArrayField( - ArrayField(models.TextField(null=False, blank=False), null=False, blank=False), - null=False, - blank=False, - default=list, - ) - status = models.CharField(max_length=40, choices=Status.choices, default=Status.ACTIVE, null=False) - assignee = models.ForeignKey( - "User", - on_delete=models.SET_NULL, - null=True, - blank=True, - ) - - @classmethod - def filter_fingerprints(cls, queryset, fingerprints: list[list]) -> QuerySet: - query = Q(fingerprint__in=fingerprints) - - for fp in fingerprints: - query |= Q(merged_fingerprints__contains=fp) - - return queryset.filter(query) - - @transaction.atomic - def merge(self, fingerprints: list[list[str]]) -> None: - if not fingerprints: - return - - # sets don't like lists so we're converting fingerprints to tuples - def convert_fingerprints_to_tuples(fps: list[list[str]]): - return [tuple(f) for f in fps] - - merged_fingerprints = set(convert_fingerprints_to_tuples(self.merged_fingerprints)) - merged_fingerprints.update(convert_fingerprints_to_tuples(fingerprints)) - - merging_groups = ErrorTrackingGroup.objects.filter(team=self.team, fingerprint__in=fingerprints) - for group in merging_groups: - merged_fingerprints |= set(convert_fingerprints_to_tuples(group.merged_fingerprints)) - - merging_groups.delete() - # converting back to list of lists before saving - self.merged_fingerprints = [list(f) for f in merged_fingerprints] - self.save() - - -class ErrorTrackingIssueFingerprint(models.Model): - team = models.ForeignKey("Team", on_delete=models.CASCADE, db_index=False) - issue = models.ForeignKey(ErrorTrackingGroup, on_delete=models.CASCADE) - fingerprint = models.TextField(null=False, blank=False) - # current version of the id, used to sync with ClickHouse and collapse rows correctly for overrides ClickHouse table - version = models.BigIntegerField(blank=True, default=0) - - class Meta: - constraints = [models.UniqueConstraint(fields=["team", "fingerprint"], name="unique fingerprint for team")] +# class ErrorTrackingGroup(UUIDModel): +# class Status(models.TextChoices): +# ARCHIVED = "archived", "Archived" +# ACTIVE = "active", "Active" +# RESOLVED = "resolved", "Resolved" +# PENDING_RELEASE = "pending_release", "Pending release" + +# team = models.ForeignKey("Team", on_delete=models.CASCADE) +# created_at = models.DateTimeField(auto_now_add=True, blank=True) +# fingerprint: ArrayField = ArrayField(models.TextField(null=False, blank=False), null=False, blank=False) +# merged_fingerprints: ArrayField = ArrayField( +# ArrayField(models.TextField(null=False, blank=False), null=False, blank=False), +# null=False, +# blank=False, +# default=list, +# ) +# status = models.CharField(max_length=40, choices=Status.choices, default=Status.ACTIVE, null=False) +# assignee = models.ForeignKey( +# "User", +# on_delete=models.SET_NULL, +# null=True, +# blank=True, +# ) + +# @classmethod +# def filter_fingerprints(cls, queryset, fingerprints: list[list]) -> QuerySet: +# query = Q(fingerprint__in=fingerprints) + +# for fp in fingerprints: +# query |= Q(merged_fingerprints__contains=fp) + +# return queryset.filter(query) + +# @transaction.atomic +# def merge(self, fingerprints: list[list[str]]) -> None: +# if not fingerprints: +# return + +# # sets don't like lists so we're converting fingerprints to tuples +# def convert_fingerprints_to_tuples(fps: list[list[str]]): +# return [tuple(f) for f in fps] + +# merged_fingerprints = set(convert_fingerprints_to_tuples(self.merged_fingerprints)) +# merged_fingerprints.update(convert_fingerprints_to_tuples(fingerprints)) + +# merging_groups = ErrorTrackingGroup.objects.filter(team=self.team, fingerprint__in=fingerprints) +# for group in merging_groups: +# merged_fingerprints |= set(convert_fingerprints_to_tuples(group.merged_fingerprints)) + +# merging_groups.delete() +# # converting back to list of lists before saving +# self.merged_fingerprints = [list(f) for f in merged_fingerprints] +# self.save() + + +# class ErrorTrackingIssueFingerprint(models.Model): +# team = models.ForeignKey("Team", on_delete=models.CASCADE, db_index=False) +# issue = models.ForeignKey(ErrorTrackingGroup, on_delete=models.CASCADE) +# fingerprint = models.TextField(null=False, blank=False) +# # current version of the id, used to sync with ClickHouse and collapse rows correctly for overrides ClickHouse table +# version = models.BigIntegerField(blank=True, default=0) + +# class Meta: +# constraints = [models.UniqueConstraint(fields=["team", "fingerprint"], name="unique fingerprint for team")] class ErrorTrackingSymbolSet(UUIDModel): diff --git a/posthog/models/error_tracking/test/test_error_tracking.py b/posthog/models/error_tracking/test/test_error_tracking.py index c7c4136f45949..c123c1f23427e 100644 --- a/posthog/models/error_tracking/test/test_error_tracking.py +++ b/posthog/models/error_tracking/test/test_error_tracking.py @@ -1,64 +1,38 @@ -from posthog.models.error_tracking import ErrorTrackingGroup -from posthog.test.base import BaseTest - - -class TestErrorTracking(BaseTest): - def test_defaults(self): - group = ErrorTrackingGroup.objects.create(status="active", team=self.team, fingerprint=["a_fingerprint"]) - - assert group.fingerprint == ["a_fingerprint"] - assert group.merged_fingerprints == [] - assert group.assignee is None - - def test_filtering(self): - ErrorTrackingGroup.objects.bulk_create( - [ - ErrorTrackingGroup(team=self.team, fingerprint=["first_error"]), - ErrorTrackingGroup( - team=self.team, fingerprint=["second_error"], merged_fingerprints=[["previously_merged"]] - ), - ErrorTrackingGroup(team=self.team, fingerprint=["third_error"]), - ] - ) - - matching_groups = ErrorTrackingGroup.objects.filter(fingerprint__in=[["first_error"], ["second_error"]]) - assert matching_groups.count() == 2 - - matching_groups = ErrorTrackingGroup.objects.filter(merged_fingerprints__contains=["previously_merged"]) - assert matching_groups.count() == 1 - - matching_groups = ErrorTrackingGroup.filter_fingerprints( - queryset=ErrorTrackingGroup.objects, fingerprints=[["first_error"], ["previously_merged"]] - ) - assert matching_groups.count() == 2 - - def test_merge(self): - primary_group = ErrorTrackingGroup.objects.create( - status="active", - team=self.team, - fingerprint=["a_fingerprint"], - merged_fingerprints=[["already_merged_fingerprint"]], - ) - merge_group_1 = ErrorTrackingGroup.objects.create( - status="active", team=self.team, fingerprint=["new_fingerprint"] - ) - merge_group_2 = ErrorTrackingGroup.objects.create( - status="active", - team=self.team, - fingerprint=["another_fingerprint"], - merged_fingerprints=[["merged_fingerprint"]], - ) - - merging_fingerprints = [merge_group_1.fingerprint, merge_group_2.fingerprint, ["no_group_fingerprint"]] - primary_group.merge(merging_fingerprints) - - assert sorted(primary_group.merged_fingerprints) == [ - ["already_merged_fingerprint"], - ["another_fingerprint"], - ["merged_fingerprint"], - ["new_fingerprint"], - ["no_group_fingerprint"], - ] - - # deletes the old groups - assert ErrorTrackingGroup.objects.count() == 1 +# class TestErrorTracking(BaseTest): +# def test_defaults(self): +# group = ErrorTrackingGroup.objects.create(status="active", team=self.team, fingerprint=["a_fingerprint"]) + +# assert group.fingerprint == ["a_fingerprint"] +# assert group.merged_fingerprints == [] +# assert group.assignee is None + +# def test_merge(self): +# primary_group = ErrorTrackingGroup.objects.create( +# status="active", +# team=self.team, +# fingerprint=["a_fingerprint"], +# merged_fingerprints=[["already_merged_fingerprint"]], +# ) +# merge_group_1 = ErrorTrackingGroup.objects.create( +# status="active", team=self.team, fingerprint=["new_fingerprint"] +# ) +# merge_group_2 = ErrorTrackingGroup.objects.create( +# status="active", +# team=self.team, +# fingerprint=["another_fingerprint"], +# merged_fingerprints=[["merged_fingerprint"]], +# ) + +# merging_fingerprints = [merge_group_1.fingerprint, merge_group_2.fingerprint, ["no_group_fingerprint"]] +# primary_group.merge(merging_fingerprints) + +# assert sorted(primary_group.merged_fingerprints) == [ +# ["already_merged_fingerprint"], +# ["another_fingerprint"], +# ["merged_fingerprint"], +# ["new_fingerprint"], +# ["no_group_fingerprint"], +# ] + +# # deletes the old groups +# assert ErrorTrackingGroup.objects.count() == 1