Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(flags): Set up initial backend for feature management #26628

Closed
wants to merge 131 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
131 commits
Select commit Hold shift + click to select a range
0846ff6
feat(flags): Set up initial UI-reorganization for feature management
havenbarnes Nov 22, 2024
b345652
tweak
havenbarnes Nov 22, 2024
0e42571
properly handle New Feature / New Flag -> Cancel navigation
havenbarnes Nov 23, 2024
7bd1ee1
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Nov 23, 2024
32a11e2
handle breadcrumbs properly based on ff
havenbarnes Nov 23, 2024
3c45f0e
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Nov 25, 2024
e2300bc
Merge branch 'master' into feature-management-ui-setup
havenbarnes Nov 25, 2024
0a2294a
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Nov 26, 2024
b2c36b8
cleanup
havenbarnes Nov 26, 2024
745bad1
tweak
havenbarnes Nov 26, 2024
b20b91c
Merge branch 'feature-management-ui-setup' of https://github.com/Post…
havenbarnes Nov 26, 2024
dbba27e
tweak
havenbarnes Nov 26, 2024
4419ec0
tweak
havenbarnes Nov 26, 2024
c07d6d4
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 2, 2024
e98fc0d
Re-worked with master-detail pattern
havenbarnes Dec 3, 2024
3d35964
tweak
havenbarnes Dec 3, 2024
75181d7
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 3, 2024
0bef6a7
tweak
havenbarnes Dec 3, 2024
00a837b
stash
havenbarnes Dec 3, 2024
8f1e249
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 3, 2024
19719f4
stash
havenbarnes Dec 3, 2024
38a761f
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 3, 2024
9008eb5
Initial endpoints, still needs tests and FE integration
havenbarnes Dec 4, 2024
31fa157
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 4, 2024
c0d5fdf
Merge branch 'feature-management-ui-setup' of https://github.com/Post…
havenbarnes Dec 4, 2024
199b552
Add tests for feature viewset
havenbarnes Dec 4, 2024
00b3f8b
tweak
havenbarnes Dec 4, 2024
a51dd17
Update query snapshots
github-actions[bot] Dec 4, 2024
38a52c9
tweak
havenbarnes Dec 4, 2024
8902068
Update query snapshots
github-actions[bot] Dec 4, 2024
5c077e8
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 4, 2024
f70dfb8
Update query snapshots
github-actions[bot] Dec 4, 2024
a3a7a3c
Update query snapshots
github-actions[bot] Dec 4, 2024
6c3e7f8
address comment
havenbarnes Dec 5, 2024
050d44f
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 5, 2024
aad2381
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 5, 2024
d1985e5
Merge branch 'feature-management-ui-setup' of https://github.com/Post…
havenbarnes Dec 5, 2024
ca7ead8
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 6, 2024
fe74f58
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 9, 2024
3cd9fc5
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 9, 2024
a32bb35
tweak
havenbarnes Dec 9, 2024
3cb58ee
adjust feature management schema / migration
havenbarnes Dec 9, 2024
4c9e59e
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 9, 2024
2ed5491
rm unnecessary diffs
havenbarnes Dec 9, 2024
36b654d
rm unnecessary diffs
havenbarnes Dec 9, 2024
9887ea4
rm unnecessary diffs
havenbarnes Dec 9, 2024
6ec3092
Update query snapshots
github-actions[bot] Dec 9, 2024
4e4ce63
Update query snapshots
github-actions[bot] Dec 9, 2024
3e0f000
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 9, 2024
18a45ff
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 9, 2024
4adbf16
rm unnecessary diffs
havenbarnes Dec 9, 2024
c7fdde6
Update query snapshots
github-actions[bot] Dec 9, 2024
6a5bf26
Update query snapshots
github-actions[bot] Dec 9, 2024
6ca7b96
tweak
havenbarnes Dec 9, 2024
3db21f0
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 9, 2024
49d23d8
tweak
havenbarnes Dec 9, 2024
a1141c2
fix typecheck and tests
havenbarnes Dec 9, 2024
084a4f7
Merge branch 'master' into feature-management-backend-setup
havenbarnes Dec 10, 2024
cace516
fix typecheck and tests
havenbarnes Dec 10, 2024
0f7deeb
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 10, 2024
701eed7
Update query snapshots
github-actions[bot] Dec 10, 2024
202b419
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 17, 2024
c80640e
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 17, 2024
f242d4f
rework for v0 model
havenbarnes Dec 18, 2024
19d886e
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 18, 2024
5243896
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 18, 2024
64bc533
Update query snapshots
github-actions[bot] Dec 18, 2024
ca68d24
Update query snapshots
github-actions[bot] Dec 18, 2024
e7b07ef
tweak
havenbarnes Dec 18, 2024
ba8a5e6
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 18, 2024
ad6d336
remove unnecessary diff
havenbarnes Dec 18, 2024
8c8026c
Update query snapshots
github-actions[bot] Dec 18, 2024
a9792c3
tweak
havenbarnes Dec 18, 2024
cb0b4bb
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 18, 2024
2e721a6
Update query snapshots
github-actions[bot] Dec 18, 2024
8c70d49
tweak
havenbarnes Dec 18, 2024
5061056
tweak
havenbarnes Dec 18, 2024
20999b2
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 18, 2024
cfa0c4d
tweak
havenbarnes Dec 18, 2024
227295e
tweak
havenbarnes Dec 18, 2024
6fe434c
fix migration
havenbarnes Dec 18, 2024
374b9c5
tweak
havenbarnes Dec 18, 2024
9ac6bc6
Merge branch 'master' into feature-management-backend-setup
havenbarnes Dec 18, 2024
6964217
fix test
havenbarnes Dec 18, 2024
01959f1
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 18, 2024
8e6bb7f
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 18, 2024
3f1f88b
Update query snapshots
github-actions[bot] Dec 18, 2024
8e8b562
Update UI snapshots for `chromium` (1)
github-actions[bot] Dec 18, 2024
77a5a0c
Merge branch 'master' into feature-management-backend-setup
havenbarnes Dec 18, 2024
ee2635b
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 19, 2024
c10609f
tweak model fields
havenbarnes Dec 19, 2024
3c5fd20
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 19, 2024
6a675fe
tweak
havenbarnes Dec 19, 2024
a3fa2a7
Update UI snapshots for `chromium` (1)
github-actions[bot] Dec 19, 2024
cf0ff01
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
50ec2a1
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
9ad3981
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
349a0b5
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
87a626d
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
cf8f698
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
62f7e0d
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
42196f1
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
fcd9cc4
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
09970f6
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
275701d
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
b630896
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
251a049
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
a02d674
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
9de1f3e
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
6a66bbf
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
54e887f
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
43ab118
Update UI snapshots for `chromium` (2)
github-actions[bot] Dec 19, 2024
ce04bd8
stash
havenbarnes Dec 20, 2024
375bf83
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 20, 2024
9e988ee
fix linter
havenbarnes Dec 20, 2024
6d34dc2
Merge branch 'feature-management-backend-setup' of https://github.com…
havenbarnes Dec 20, 2024
5c2ba52
tweak
havenbarnes Dec 20, 2024
ec86c2f
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 20, 2024
d35989f
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Dec 30, 2024
db606bd
update
havenbarnes Dec 30, 2024
4a7d2bd
update
havenbarnes Dec 30, 2024
e4a67e1
tweak
havenbarnes Jan 3, 2025
796bfd5
tweak
havenbarnes Jan 3, 2025
adf602b
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Jan 3, 2025
3ee964b
Merge branch 'master' into feature-management-backend-setup
havenbarnes Jan 3, 2025
1ab5de5
delete dupe migration
havenbarnes Jan 3, 2025
5073dc6
Merge branch 'master' of https://github.com/PostHog/posthog into feat…
havenbarnes Jan 8, 2025
89a728e
update migrate script
havenbarnes Jan 8, 2025
dcfc9b1
remove field from serializer
havenbarnes Jan 8, 2025
f5fc480
remove field from serializer
havenbarnes Jan 8, 2025
46e68b1
Update query snapshots
github-actions[bot] Jan 8, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions frontend/src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3900,6 +3900,7 @@ export type APIScopeObject =
| 'event_definition'
| 'experiment'
| 'export'
| 'feature'
| 'feature_flag'
| 'group'
| 'insight'
Expand Down
7 changes: 7 additions & 0 deletions posthog/api/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
error_tracking,
event_definition,
exports,
feature,
feature_flag,
hog_function,
hog_function_template,
Expand Down Expand Up @@ -158,6 +159,12 @@ def register_grandfathered_environment_nested_viewset(
"project_activity_log",
["project_id"],
)
projects_router.register(
r"features",
feature.FeatureViewSet,
"project_feature_management",
["project_id"],
)
project_feature_flags_router = projects_router.register(
r"feature_flags",
feature_flag.FeatureFlagViewSet,
Expand Down
9 changes: 1 addition & 8 deletions posthog/api/early_access_feature.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,14 +30,7 @@ class MinimalEarlyAccessFeatureSerializer(serializers.ModelSerializer):

class Meta:
model = EarlyAccessFeature
fields = [
"id",
"name",
"description",
"stage",
"documentationUrl",
"flagKey",
]
fields = ["id", "name", "description", "stage", "documentationUrl", "flagKey"]
read_only_fields = fields


Expand Down
81 changes: 81 additions & 0 deletions posthog/api/feature.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
from rest_framework import viewsets, serializers
from rest_framework.response import Response
from rest_framework.decorators import action
from django.db.models import QuerySet
from posthog.models import Feature, FeatureAlertConfiguration
from posthog.api.alert import AlertSerializer
from posthog.api.early_access_feature import EarlyAccessFeatureSerializer
from posthog.api.routing import TeamAndOrgViewSetMixin
from posthog.api.forbid_destroy_model import ForbidDestroyModel
from posthog.rbac.access_control_api_mixin import AccessControlViewSetMixin


class FeatureAlertConfigurationSerializer(serializers.ModelSerializer):
alert_configuration = AlertSerializer()

class Meta:
model = FeatureAlertConfiguration
fields = ["alert_configuration", "feature_insight_type"]


class FeatureSerializer(serializers.ModelSerializer):
class Meta:
model = Feature
fields = [
"id",
"name",
"description",
"primary_early_access_feature_id",
"created_at",
"created_by",
"archived",
"deleted",
]

def create(self, validated_data):
request = self.context["request"]

validated_data["team_id"] = self.context["team_id"]
validated_data["created_by"] = request.user
validated_data["primary_early_access_feature_id"] = request.data.get("primary_early_access_feature_id")
return super().create(validated_data)

def get_primary_early_access_feature(self, feature: Feature):
return EarlyAccessFeatureSerializer(feature.primary_early_access_feature, context=self.context).data


class FeatureViewSet(TeamAndOrgViewSetMixin, AccessControlViewSetMixin, ForbidDestroyModel, viewsets.ModelViewSet):
scope_object = "feature"
queryset = Feature.objects.all()
serializer_class = FeatureSerializer

def safely_get_queryset(self, queryset) -> QuerySet:
# Base queryset with team filtering
queryset = Feature.objects.filter(team_id=self.team_id)

if self.action == "primary_early_access_feature":
queryset = queryset.select_related("primary_early_access_feature")
elif self.action == "alerts":
queryset = queryset.prefetch_related("alerts")

return queryset

@action(detail=True, methods=["get"])
def primary_early_access_feature(self, request, pk=None, **kwargs):
"""
Get primary feature flag associated with a specific feature.
"""
feature = self.get_object()
primary_early_access_feature = FeatureSerializer().get_primary_early_access_feature(feature)
return Response(primary_early_access_feature)

@action(detail=True, methods=["get"])
def alerts(self, request, pk=None, **kwargs):
"""
Get all alerts associated with a specific feature. These alerts are used to track
success and failure metrics for the feature.
"""
feature = self.get_object()
alerts_for_feature = FeatureAlertConfiguration.objects.filter(feature=feature).all()
alerts = FeatureAlertConfigurationSerializer(alerts_for_feature, many=True).data
return Response(alerts)
1 change: 1 addition & 0 deletions posthog/api/test/__snapshots__/test_api_docs.ambr
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@
'/home/runner/work/posthog/posthog/posthog/api/exports.py: Warning [ExportedAssetViewSet > ExportedAssetSerializer]: unable to resolve type hint for function "filename". Consider using a type hint or @extend_schema_field. Defaulting to string.',
'/home/runner/work/posthog/posthog/posthog/api/exports.py: Warning [ExportedAssetViewSet > ExportedAssetSerializer]: unable to resolve type hint for function "has_content". Consider using a type hint or @extend_schema_field. Defaulting to string.',
'/home/runner/work/posthog/posthog/posthog/api/exports.py: Warning [ExportedAssetViewSet]: could not derive type of path parameter "project_id" because model "posthog.models.exported_asset.ExportedAsset" contained no such field. Consider annotating parameter with @extend_schema. Defaulting to "string".',
'/home/runner/work/posthog/posthog/posthog/api/feature.py: Warning [FeatureViewSet]: could not derive type of path parameter "project_id" because model "posthog.models.feature.Feature" contained no such field. Consider annotating parameter with @extend_schema. Defaulting to "string".',
'/home/runner/work/posthog/posthog/posthog/api/feature_flag.py: Warning [FeatureFlagViewSet]: could not derive type of path parameter "project_id" because model "posthog.models.feature_flag.feature_flag.FeatureFlag" contained no such field. Consider annotating parameter with @extend_schema. Defaulting to "string".',
'/home/runner/work/posthog/posthog/posthog/api/insight.py: Warning [EnterpriseInsightsViewSet > InsightSerializer]: unable to resolve type hint for function "get_cache_target_age". Consider using a type hint or @extend_schema_field. Defaulting to string.',
'/home/runner/work/posthog/posthog/posthog/api/insight.py: Warning [EnterpriseInsightsViewSet > InsightSerializer]: unable to resolve type hint for function "get_columns". Consider using a type hint or @extend_schema_field. Defaulting to string.',
Expand Down
114 changes: 114 additions & 0 deletions posthog/api/test/test_feature.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
from rest_framework import status

from posthog.models import AlertConfiguration, EarlyAccessFeature, Feature, FeatureAlertConfiguration, Insight
from posthog.test.base import APIBaseTest


class TestFeatureAPI(APIBaseTest):
def setUp(self):
super().setUp()
self.early_access_feature = EarlyAccessFeature.objects.create(
team=self.team,
name="Test EAF",
description="Test Description",
)
self.feature = Feature.objects.create(
team=self.team,
name="Test Feature",
description="Test Description",
primary_early_access_feature=self.early_access_feature,
)

def test_list_features(self):
response = self.client.get(f"/api/projects/{self.team.id}/features/")
self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(len(response.json()["results"]), 1)
self.assertEqual(response.json()["results"][0]["name"], "Test Feature")
self.assertEqual(
response.json()["results"][0]["primary_early_access_feature_id"], f"{self.early_access_feature.id}"
)

# Ensure we are not returning joined data
with self.assertRaises(KeyError):
response.json()["results"][0]["alerts"]
with self.assertRaises(KeyError):
response.json()["results"][0]["primary_early_access_feature"]

def test_retrieve_feature(self):
eaf = EarlyAccessFeature.objects.create(
team=self.team,
name="Test EAF",
description="Test Description",
)
self.feature.primary_early_access_feature = eaf

response = self.client.get(f"/api/projects/{self.team.id}/features/{self.feature.id}/")
self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(response.json()["name"], "Test Feature")
self.assertEqual(response.json()["description"], "Test Description")

# Ensure we are not returning joined data
with self.assertRaises(KeyError):
response.json()["results"][0]["alerts"]
with self.assertRaises(KeyError):
response.json()["results"][0]["primary_early_access_feature"]

def test_create_feature(self):
response = self.client.post(
f"/api/projects/{self.team.id}/features/",
{
"name": "New Feature",
"team_id": self.team.id,
"description": "New Description",
"primary_early_access_feature_id": self.early_access_feature.id,
},
)
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
self.assertEqual(response.json()["name"], "New Feature")
self.assertEqual(Feature.objects.count(), 2)

def test_update_feature(self):
response = self.client.patch(
f"/api/projects/{self.team.id}/features/{self.feature.id}/",
{
"name": "Updated Feature",
},
)
self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(response.json()["name"], "Updated Feature")

def test_delete_not_allowed(self):
response = self.client.delete(f"/api/projects/{self.team.id}/features/{self.feature.id}/")
self.assertEqual(response.status_code, status.HTTP_405_METHOD_NOT_ALLOWED)

def test_get_primary_early_access_feature(self):
eaf = EarlyAccessFeature.objects.create(
team=self.team,
name="Test Flag",
)
self.feature.primary_early_access_feature = eaf
self.feature.save()

response = self.client.get(
f"/api/projects/{self.team.id}/features/{self.feature.id}/primary_early_access_feature/"
)
self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(response.json()["name"], "Test Flag")

def test_get_alerts(self):
insight = Insight.objects.create(
team=self.team,
filters={"events": [{"id": "$pageview", "type": "events", "name": "pageview"}]},
)
alert = AlertConfiguration.objects.create(team=self.team, name="Test Alert", insight=insight)
FeatureAlertConfiguration.objects.create(
team=self.team,
feature=self.feature,
alert_configuration=alert,
feature_insight_type="success_metric",
)

response = self.client.get(f"/api/projects/{self.team.id}/features/{self.feature.id}/alerts/")
self.assertEqual(response.status_code, status.HTTP_200_OK)
self.assertEqual(len(response.json()), 1)
self.assertEqual(response.json()[0]["feature_insight_type"], "success_metric")
87 changes: 87 additions & 0 deletions posthog/migrations/0540_feature_mgmt_tables.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,87 @@
# Generated by Django 3.2.18 on 2023-06-22 11:08

from django.db import migrations, models
from django.conf import settings

import posthog.models.utils


class Migration(migrations.Migration):
dependencies = [
("posthog", "0539_user_role_at_organization"),
]

operations = [
migrations.CreateModel(
name="Feature",
fields=[
(
"id",
models.UUIDField(
default=posthog.models.utils.UUIDT,
editable=False,
primary_key=True,
serialize=False,
),
),
("key", models.CharField(max_length=400, blank=False)),
("team", models.ForeignKey(on_delete=models.deletion.CASCADE, to="posthog.team")),
("name", models.CharField(max_length=400, blank=False)),
("description", models.TextField(default="", blank=True)),
(
"primary_early_access_feature",
models.ForeignKey(on_delete=models.deletion.RESTRICT, to="posthog.earlyaccessfeature"),
),
("created_at", models.DateTimeField(auto_now_add=True)),
(
"created_by",
models.ForeignKey(
blank=True, null=True, on_delete=models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL
),
),
("archived", models.BooleanField(default=False)),
("deleted", models.BooleanField(default=False)),
],
),
migrations.CreateModel(
name="FeatureAlertConfiguration",
fields=[
(
"id",
models.UUIDField(
default=posthog.models.utils.UUIDT,
editable=False,
primary_key=True,
serialize=False,
),
),
(
"team",
models.ForeignKey(on_delete=models.deletion.CASCADE, to="posthog.Team"),
),
(
"feature",
models.ForeignKey(on_delete=models.deletion.CASCADE, to="posthog.Feature"),
),
(
"alert_configuration",
models.ForeignKey(on_delete=models.deletion.CASCADE, to="posthog.AlertConfiguration"),
),
(
"feature_insight_type",
models.CharField(
max_length=32,
choices=[("success_metric", "Success Metric"), ("faiure_metric", "Failure Metric")],
),
),
],
),
migrations.AddField(
model_name="Feature",
name="alerts",
field=models.ManyToManyField(
to="posthog.AlertConfiguration",
through="posthog.FeatureAlertConfiguration",
),
),
]
2 changes: 1 addition & 1 deletion posthog/migrations/max_migration.txt
Original file line number Diff line number Diff line change
@@ -1 +1 @@
0539_user_role_at_organization
0540_feature_mgmt_tables
3 changes: 3 additions & 0 deletions posthog/models/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@
from .event_property import EventProperty
from .experiment import Experiment
from .exported_asset import ExportedAsset
from .feature import Feature, FeatureAlertConfiguration
from .feature_flag import FeatureFlag
from .feedback.survey import Survey
from .filters import Filter, RetentionFilter
Expand Down Expand Up @@ -119,6 +120,8 @@
"EventProperty",
"Experiment",
"ExportedAsset",
"Feature",
"FeatureAlertConfiguration",
"FeatureFlag",
"Filter",
"Group",
Expand Down
28 changes: 28 additions & 0 deletions posthog/models/feature.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
from django.db import models

from posthog.models.utils import UUIDModel, CreatedMetaFields


class Feature(CreatedMetaFields, UUIDModel):
# Key must be unique per team across all features and all feature flags,
# as the early access feature will create a flag with this same key.
key = models.CharField(max_length=400, blank=False)
team = models.ForeignKey("posthog.Team", on_delete=models.CASCADE)
name = models.CharField(max_length=400, blank=False)
description = models.TextField(default="", blank=True)
primary_early_access_feature = models.ForeignKey(
"posthog.EarlyAccessFeature", on_delete=models.RESTRICT, blank=False
)
alerts = models.ManyToManyField("posthog.AlertConfiguration", through="posthog.FeatureAlertConfiguration")
archived = models.BooleanField(default=False)
deleted = models.BooleanField(default=False)


class FeatureAlertConfiguration(UUIDModel):
team = models.ForeignKey("posthog.Team", on_delete=models.CASCADE)
feature = models.ForeignKey("posthog.Feature", on_delete=models.CASCADE)
alert_configuration = models.ForeignKey("posthog.AlertConfiguration", on_delete=models.CASCADE)
feature_insight_type = models.CharField(
max_length=32,
choices=[("success_metric", "Success Metric"), ("faiure_metric", "Failure Metric")],
)
1 change: 1 addition & 0 deletions posthog/models/scopes.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
"event_definition",
"experiment",
"export",
"feature",
"feature_flag",
"group",
"insight",
Expand Down
Loading