0
0
mirror of https://github.com/PostHog/posthog.git synced 2024-11-28 18:26:15 +01:00
posthog/ee/api/test/test_organization.py
Karl-Aksel Puulmann 7de761dff0
Update BE tracking to use groups (#7355)
* Update some posthoganalytics.capture calls

* Update report_user_logged_in tracking

* Update report_team_member_invited

* Update report_bulk_invited

* Update report_org_usage

* Update AnalyticsDestroyModelMixin

* Update action creation/updating tracking

* Update annotations tracking

* Update cohort tracking

* Update dashboards tracking

* Update feature flags tracking

* Update imports, sessions tracking

* More typeignores

* Default properties blank

* update metadata default

* Stringify

* Fix some tests

* Kill debug code

* Update FF tests

* Update test_organization

* Update test_organization_invites

* Update test_user

* Fix org usage report

* update test_send_license_usage

* Update organization tests

* Respect dry-run on errors as well, log

* Update organization properties on report

* Assert organization props

* Remove dead imports

* Fix test
2021-11-30 10:58:37 +02:00

225 lines
11 KiB
Python

import datetime as dt
import random
from unittest.mock import ANY, Mock, patch
from freezegun.api import freeze_time
from rest_framework import status
from ee.api.test.base import APILicensedTest
from ee.models.license import License
from posthog.celery import sync_all_organization_available_features
from posthog.models import Team, User
from posthog.models.organization import Organization, OrganizationMembership
class TestOrganizationEnterpriseAPI(APILicensedTest):
def test_create_organization(self):
response = self.client.post("/api/organizations/", {"name": "Test"})
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
self.assertEqual(Organization.objects.count(), 2)
response_data = response.json()
self.assertEqual(response_data.get("name"), "Test")
self.assertEqual(OrganizationMembership.objects.filter(organization_id=response_data.get("id")).count(), 1)
self.assertEqual(
OrganizationMembership.objects.get(organization_id=response_data.get("id"), user=self.user).level,
OrganizationMembership.Level.OWNER,
)
def test_create_two_similarly_named_organizations(self):
random.seed(0)
response = self.client.post("/api/organizations/", {"name": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"})
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
self.assertDictContainsSubset(
{
"name": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
"slug": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
},
response.json(),
)
response = self.client.post(
"/api/organizations/", {"name": "#XXxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxX"}
)
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
self.assertDictContainsSubset(
{
"name": "#XXxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxX",
"slug": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx-yWAc",
},
response.json(),
)
@patch("posthoganalytics.capture")
def test_delete_second_managed_organization(self, mock_capture):
organization, _, team = Organization.objects.bootstrap(self.user, name="X")
organization_props = organization.get_analytics_metadata()
self.assertTrue(Organization.objects.filter(id=organization.id).exists())
self.assertTrue(Team.objects.filter(id=team.id).exists())
response = self.client.delete(f"/api/organizations/{organization.id}")
self.assertEqual(response.status_code, 204)
self.assertFalse(Organization.objects.filter(id=organization.id).exists())
self.assertFalse(Team.objects.filter(id=team.id).exists())
mock_capture.assert_called_once_with(
self.user.distinct_id,
"organization deleted",
organization_props,
groups={"instance": ANY, "organization": str(organization.id)},
)
@patch("posthoganalytics.capture")
def test_delete_last_organization(self, mock_capture):
org_id = self.organization.id
organization_props = self.organization.get_analytics_metadata()
self.assertTrue(Organization.objects.filter(id=org_id).exists())
self.organization_membership.level = OrganizationMembership.Level.OWNER
self.organization_membership.save()
response = self.client.delete(f"/api/organizations/{org_id}")
self.assertEqual(response.status_code, 204, "Did not successfully delete last organization on the instance")
self.assertFalse(Organization.objects.filter(id=org_id).exists())
self.assertFalse(Organization.objects.exists())
response_bis = self.client.delete(f"/api/organizations/{org_id}")
self.assertEqual(response_bis.status_code, 404, "Did not return a 404 on trying to delete a nonexistent org")
mock_capture.assert_called_once_with(
self.user.distinct_id,
"organization deleted",
organization_props,
groups={"instance": ANY, "organization": str(org_id)},
)
def test_no_delete_organization_not_owning(self):
for level in (OrganizationMembership.Level.MEMBER, OrganizationMembership.Level.ADMIN):
self.organization_membership.level = level
self.organization_membership.save()
response = self.client.delete(f"/api/organizations/{self.organization.id}")
potential_err_message = f"Somehow managed to delete the org as a level {level} (which is not owner)"
self.assertEqual(
response.json(),
{
"attr": None,
"detail": "Your organization access level is insufficient.",
"code": "permission_denied",
"type": "authentication_error",
},
potential_err_message,
)
self.assertEqual(response.status_code, 403, potential_err_message)
self.assertTrue(self.organization.name, self.CONFIG_ORGANIZATION_NAME)
def test_delete_organization_owning(self):
self.organization_membership.level = OrganizationMembership.Level.OWNER
self.organization_membership.save()
membership_ids = OrganizationMembership.objects.filter(organization=self.organization).values_list(
"id", flat=True
)
response = self.client.delete(f"/api/organizations/{self.organization.id}")
potential_err_message = f"Somehow did not delete the org as the owner"
self.assertEqual(response.status_code, 204, potential_err_message)
self.assertFalse(Organization.objects.filter(id=self.organization.id).exists(), potential_err_message)
self.assertFalse(OrganizationMembership.objects.filter(id__in=membership_ids).exists())
self.assertTrue(User.objects.filter(id=self.user.pk).exists())
def test_no_delete_organization_not_belonging_to(self):
for level in OrganizationMembership.Level:
self.organization_membership.level = level
self.organization_membership.save()
organization = Organization.objects.create(name="Some Other Org")
response = self.client.delete(f"/api/organizations/{organization.id}")
potential_err_message = f"Somehow managed to delete someone else's org as a level {level} in own org"
self.assertEqual(
response.json(),
{"attr": None, "detail": "Not found.", "code": "not_found", "type": "invalid_request"},
potential_err_message,
)
self.assertEqual(response.status_code, 404, potential_err_message)
self.assertTrue(Organization.objects.filter(id=organization.id).exists(), potential_err_message)
def test_update_org(self):
for level in OrganizationMembership.Level:
self.organization_membership.level = level
self.organization_membership.save()
response_rename = self.client.patch(f"/api/organizations/{self.organization.id}", {"name": "Woof"})
response_email = self.client.patch(
f"/api/organizations/{self.organization.id}", {"is_member_join_email_enabled": False}
)
self.organization.refresh_from_db()
expected_response = {
"attr": None,
"detail": "Your organization access level is insufficient.",
"code": "permission_denied",
"type": "authentication_error",
}
if level < OrganizationMembership.Level.ADMIN:
potential_err_message = f"Somehow managed to update the org as a level {level} (which is below admin)"
self.assertEqual(
response_rename.json(), expected_response, potential_err_message,
)
self.assertEqual(response_rename.status_code, 403, potential_err_message)
self.assertTrue(self.organization.name, self.CONFIG_ORGANIZATION_NAME)
self.assertEqual(
response_email.json(), expected_response, potential_err_message,
)
self.assertEqual(response_email.status_code, 403, potential_err_message)
else:
potential_err_message = f"Somehow did not update the org as a level {level} (which is at least admin)"
self.assertEqual(response_rename.status_code, 200, potential_err_message)
self.assertEqual(response_email.status_code, 200, potential_err_message)
self.assertTrue(self.organization.name, "Woof")
def test_no_update_organization_not_belonging_to(self):
for level in OrganizationMembership.Level:
self.organization_membership.level = level
self.organization_membership.save()
organization = Organization.objects.create(name="Meow")
response = self.client.patch(f"/api/organizations/{organization.id}", {"name": "Mooooooooo"})
potential_err_message = f"Somehow managed to update someone else's org as a level {level} in own org"
self.assertEqual(
response.json(),
{"attr": None, "detail": "Not found.", "code": "not_found", "type": "invalid_request"},
potential_err_message,
)
self.assertEqual(response.status_code, 404, potential_err_message)
organization.refresh_from_db()
self.assertTrue(organization.name, "Meow")
@patch("posthog.models.organization.License.PLANS", {"enterprise": ["whatever"]})
@patch("ee.models.license.requests.post")
def test_feature_available_self_hosted_has_license(self, patch_post):
with self.settings(MULTI_TENANCY=False):
mock = Mock()
mock.json.return_value = {"plan": "enterprise", "valid_until": dt.datetime.now() + dt.timedelta(days=1)}
patch_post.return_value = mock
License.objects.create(key="key")
# Still only old, empty available_features field value known
self.assertFalse(self.organization.is_feature_available("whatever"))
self.assertFalse(self.organization.is_feature_available("feature-doesnt-exist"))
# New available_features field value that was updated in DB on license creation is known after refresh
self.organization.refresh_from_db()
self.assertTrue(self.organization.is_feature_available("whatever"))
self.assertFalse(self.organization.is_feature_available("feature-doesnt-exist"))
@patch("posthog.models.organization.License.PLANS", {"enterprise": ["whatever"]})
def test_feature_available_self_hosted_no_license(self):
self.assertFalse(self.organization.is_feature_available("whatever"))
self.assertFalse(self.organization.is_feature_available("feature-doesnt-exist"))
@patch("posthog.models.organization.License.PLANS", {"enterprise": ["whatever"]})
@patch("ee.models.license.requests.post")
def test_feature_available_self_hosted_license_expired(self, patch_post):
with freeze_time("2070-01-01T12:00:00.000Z"): # LicensedTestMixin enterprise license expires in 2038
sync_all_organization_available_features() # This is normally ran every hour
self.organization.refresh_from_db()
self.assertFalse(self.organization.is_feature_available("whatever"))