From 7283ea07fc229c89b896380c2fc0bc0cc0c7c27c Mon Sep 17 00:00:00 2001 From: Haven Barnes Date: Mon, 16 Dec 2024 13:07:32 -0800 Subject: [PATCH 1/3] chore(flags): stop checking clickhouse for feature flag status endpoint --- posthog/models/feature_flag/flag_status.py | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/posthog/models/feature_flag/flag_status.py b/posthog/models/feature_flag/flag_status.py index fa7ad52929304..df5a90674ac07 100644 --- a/posthog/models/feature_flag/flag_status.py +++ b/posthog/models/feature_flag/flag_status.py @@ -23,6 +23,7 @@ class FeatureFlagStatus(StrEnum): # - ACTIVE: The feature flag is actively evaluated and the evaluations continue to vary. # - STALE: The feature flag has been fully rolled out to users. Its evaluations can not vary. # - INACTIVE: The feature flag is not being actively evaluated. STALE takes precedence over INACTIVE. +# NOTE: This status is not currently used, but may be used in the future to automatically archive flags. # - DELETED: The feature flag has been soft deleted. # - UNKNOWN: The feature flag is not found in the database. class FeatureFlagStatusChecker: @@ -49,10 +50,6 @@ def get_status(self) -> tuple[FeatureFlagStatus, FeatureFlagStatusReason]: if is_flag_fully_rolled_out: return FeatureFlagStatus.STALE, fully_rolled_out_explanation - # Final, and most expensive check: see if the flag has been evaluated recently. - if self.is_flag_unevaluated_recently(flag): - return FeatureFlagStatus.INACTIVE, "Flag has not been evaluated recently" - return FeatureFlagStatus.ACTIVE, "Flag is not fully rolled out and may still be active" def is_flag_fully_rolled_out(self, flag: FeatureFlag) -> tuple[bool, FeatureFlagStatusReason]: From 0e3870e71ca362362ab91fc350cf81386a264d5a Mon Sep 17 00:00:00 2001 From: Haven Date: Mon, 16 Dec 2024 13:25:47 -0800 Subject: [PATCH 2/3] Update posthog/models/feature_flag/flag_status.py Co-authored-by: Dylan Martin --- posthog/models/feature_flag/flag_status.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/posthog/models/feature_flag/flag_status.py b/posthog/models/feature_flag/flag_status.py index df5a90674ac07..18caaa5814a5a 100644 --- a/posthog/models/feature_flag/flag_status.py +++ b/posthog/models/feature_flag/flag_status.py @@ -23,7 +23,7 @@ class FeatureFlagStatus(StrEnum): # - ACTIVE: The feature flag is actively evaluated and the evaluations continue to vary. # - STALE: The feature flag has been fully rolled out to users. Its evaluations can not vary. # - INACTIVE: The feature flag is not being actively evaluated. STALE takes precedence over INACTIVE. -# NOTE: This status is not currently used, but may be used in the future to automatically archive flags. +# NOTE: The "inactive" status is not currently used, but may be used in the future to automatically archive flags. # - DELETED: The feature flag has been soft deleted. # - UNKNOWN: The feature flag is not found in the database. class FeatureFlagStatusChecker: From fc713607070758aaa65ec409315e9a9e41fe6c64 Mon Sep 17 00:00:00 2001 From: Haven Barnes Date: Mon, 16 Dec 2024 13:30:59 -0800 Subject: [PATCH 3/3] fix tests --- posthog/api/test/test_feature_flag.py | 70 +++++---------------------- 1 file changed, 12 insertions(+), 58 deletions(-) diff --git a/posthog/api/test/test_feature_flag.py b/posthog/api/test/test_feature_flag.py index efc67f79e3f40..4ae2d364d1f03 100644 --- a/posthog/api/test/test_feature_flag.py +++ b/posthog/api/test/test_feature_flag.py @@ -2,7 +2,6 @@ import json from typing import Optional from unittest.mock import call, patch -from dateutil.relativedelta import relativedelta from django.core.cache import cache from django.db import connection @@ -41,7 +40,6 @@ ClickhouseTestMixin, FuzzyInt, QueryMatchingTest, - _create_event, _create_person, flush_persons_and_events, snapshot_clickhouse_queries, @@ -6130,18 +6128,6 @@ def assert_expected_response( if expected_reason is not None: self.assertEqual(response_data.get("reason"), expected_reason) - def create_feature_flag_called_event( - self, feature_flag_key: str, response: Optional[bool] = True, datetime: Optional[datetime.datetime] = None - ): - timestamp = datetime or now() - relativedelta(hours=12) - _create_event( - event="$feature_flag_called", - distinct_id="person1", - properties={"$feature_flag": feature_flag_key, "$feature_flag_response": response}, - team=self.team, - timestamp=timestamp, - ) - def test_flag_status_reasons(self): FeatureFlag.objects.all().delete() @@ -6165,7 +6151,7 @@ def test_flag_status_reasons(self): team=self.team, active=False, ) - self.create_feature_flag_called_event(disabled_flag.key) + self.assert_expected_response(disabled_flag.id, FeatureFlagStatus.ACTIVE) # Request status for flag that has super group rolled out to <100% @@ -6176,7 +6162,7 @@ def test_flag_status_reasons(self): active=True, filters={"super_groups": [{"rollout_percentage": 50, "properties": []}]}, ) - self.create_feature_flag_called_event(fifty_percent_super_group_flag.key) + self.assert_expected_response(fifty_percent_super_group_flag.id, FeatureFlagStatus.ACTIVE) # Request status for flag that has super group rolled out to 100% and specific properties @@ -6201,7 +6187,7 @@ def test_flag_status_reasons(self): ] }, ) - self.create_feature_flag_called_event(fully_rolled_out_super_group_flag_with_properties.key) + self.assert_expected_response(fully_rolled_out_super_group_flag_with_properties.id, FeatureFlagStatus.ACTIVE) # Request status for flag that has super group rolled out to 100% and has no specific properties @@ -6231,7 +6217,7 @@ def test_flag_status_reasons(self): active=True, filters={"holdout_groups": [{"rollout_percentage": 50, "properties": []}]}, ) - self.create_feature_flag_called_event(fifty_percent_holdout_group_flag.key) + self.assert_expected_response(fifty_percent_holdout_group_flag.id, FeatureFlagStatus.ACTIVE) # Request status for flag that has holdout group rolled out to 100% and specific properties @@ -6256,7 +6242,7 @@ def test_flag_status_reasons(self): ] }, ) - self.create_feature_flag_called_event(fully_rolled_out_holdout_group_flag_with_properties.key) + self.assert_expected_response(fully_rolled_out_holdout_group_flag_with_properties.id, FeatureFlagStatus.ACTIVE) # Request status for flag that has holdout group rolled out to 100% and has no specific properties @@ -6293,7 +6279,7 @@ def test_flag_status_reasons(self): } }, ) - self.create_feature_flag_called_event(multivariate_flag_no_rolled_out_variants.key) + self.assert_expected_response(multivariate_flag_no_rolled_out_variants.id, FeatureFlagStatus.ACTIVE) # Request status for multivariate flag with no variants set to 100% @@ -6337,7 +6323,7 @@ def test_flag_status_reasons(self): ], }, ) - self.create_feature_flag_called_event(multivariate_flag_rolled_out_variant_no_rolled_out_release.key) + self.assert_expected_response( multivariate_flag_rolled_out_variant_no_rolled_out_release.id, FeatureFlagStatus.ACTIVE, @@ -6361,7 +6347,7 @@ def test_flag_status_reasons(self): ], }, ) - self.create_feature_flag_called_event(multivariate_flag_rolled_out_release_condition_half_variant.key) + self.assert_expected_response( multivariate_flag_rolled_out_release_condition_half_variant.id, FeatureFlagStatus.ACTIVE, @@ -6396,7 +6382,7 @@ def test_flag_status_reasons(self): ], }, ) - self.create_feature_flag_called_event(multivariate_flag_rolled_out_variant_rolled_out_filtered_release.key) + self.assert_expected_response( multivariate_flag_rolled_out_variant_rolled_out_filtered_release.id, FeatureFlagStatus.ACTIVE, @@ -6431,7 +6417,7 @@ def test_flag_status_reasons(self): ], }, ) - self.create_feature_flag_called_event(multivariate_flag_filtered_rolled_out_release_with_override.key) + self.assert_expected_response( multivariate_flag_filtered_rolled_out_release_with_override.id, FeatureFlagStatus.ACTIVE, @@ -6509,7 +6495,7 @@ def test_flag_status_reasons(self): ], }, ) - self.create_feature_flag_called_event(boolean_flag_no_rolled_out_release_conditions.key) + self.assert_expected_response( boolean_flag_no_rolled_out_release_conditions.id, FeatureFlagStatus.ACTIVE, @@ -6570,39 +6556,7 @@ def test_flag_status_reasons(self): ], }, ) - self.create_feature_flag_called_event(boolean_flag_no_rolled_out_release_condition_recently_evaluated.key) - self.assert_expected_response( - boolean_flag_no_rolled_out_release_condition_recently_evaluated.id, FeatureFlagStatus.ACTIVE - ) - # Request status for a boolean flag with no rolled out release conditions, and has - # been called, but not recently - boolean_flag_rolled_out_release_condition_not_recently_evaluated = FeatureFlag.objects.create( - name="Boolean flag with a release condition set to 100%", - key="boolean-not-recently-evaluated-flag", - team=self.team, - active=True, - filters={ - "groups": [ - { - "properties": [ - { - "key": "name", - "type": "person", - "value": ["Smith"], - "operator": "contains", - } - ], - "rollout_percentage": 50, - }, - ], - }, - ) - self.create_feature_flag_called_event( - boolean_flag_rolled_out_release_condition_not_recently_evaluated.key, True, now() - relativedelta(days=31) - ) self.assert_expected_response( - boolean_flag_rolled_out_release_condition_not_recently_evaluated.id, - FeatureFlagStatus.INACTIVE, - "Flag has not been evaluated recently", + boolean_flag_no_rolled_out_release_condition_recently_evaluated.id, FeatureFlagStatus.ACTIVE )