Skip to content

Commit

Permalink
Merge branch 'master' into sort-imports
Browse files Browse the repository at this point in the history
  • Loading branch information
Twixes committed Nov 17, 2023
2 parents 663392c + 4557575 commit fe3d792
Show file tree
Hide file tree
Showing 264 changed files with 2,429 additions and 3,825 deletions.
55 changes: 31 additions & 24 deletions .eslintrc.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,12 +27,12 @@ module.exports = {
},
extends: [
'eslint:recommended',
'plugin:@typescript-eslint/recommended',
'plugin:@typescript-eslint/recommended-type-checked',
'plugin:react/recommended',
'plugin:eslint-comments/recommended',
'plugin:storybook/recommended',
'prettier',
'plugin:compat/recommended',
'prettier',
],
globals,
parser: '@typescript-eslint/parser',
Expand All @@ -42,6 +42,7 @@ module.exports = {
},
ecmaVersion: 2018,
sourceType: 'module',
project: 'tsconfig.json'
},
plugins: [
'prettier',
Expand Down Expand Up @@ -84,7 +85,27 @@ module.exports = {
'@typescript-eslint/no-empty-function': 'off',
'@typescript-eslint/no-inferrable-types': 'off',
'@typescript-eslint/ban-ts-comment': 'off',
'@typescript-eslint/no-non-null-assertion': 'error',
'@typescript-eslint/require-await': 'off', // TODO: Enable - this rule is useful, but doesn't have an autofix
'@typescript-eslint/no-unsafe-assignment': 'off',
'@typescript-eslint/no-unsafe-member-access': 'off',
'@typescript-eslint/no-unsafe-enum-comparison': 'off',
'@typescript-eslint/no-unsafe-argument': 'off',
'@typescript-eslint/no-unsafe-return': 'off',
'@typescript-eslint/no-unsafe-call': 'off',
'@typescript-eslint/no-explicit-any': 'off',
'@typescript-eslint/restrict-template-expressions': 'off',
'@typescript-eslint/explicit-function-return-type': [
'error',
{
allowExpressions: true,
},
],
'@typescript-eslint/explicit-module-boundary-types': [
'error',
{
allowArgumentsExplicitlyTypedAsAny: true,
},
],
curly: 'error',
'no-restricted-imports': [
'error',
Expand Down Expand Up @@ -242,43 +263,29 @@ module.exports = {
...globals,
given: 'readonly',
},
rules: {
// The below complains needlessly about expect(api.createInvite).toHaveBeenCalledWith(...)
'@typescript-eslint/unbound-method': 'off',
}
},
{
// disable these rules for files generated by kea-typegen
files: ['*Type.ts', '*Type.tsx'],
rules: {
'@typescript-eslint/no-explicit-any': ['off'],
'no-restricted-imports': 'off',
'@typescript-eslint/ban-types': ['off'],
},
},
{
// enable the rule specifically for TypeScript files
files: ['*.ts', '*.tsx'],
rules: {
'@typescript-eslint/no-explicit-any': ['off'],
'@typescript-eslint/explicit-function-return-type': [
'error',
{
allowExpressions: true,
},
],
'@typescript-eslint/explicit-module-boundary-types': [
'error',
{
allowArgumentsExplicitlyTypedAsAny: true,
},
],
},
},
{
files: ['*.js'],
rules: {
'@typescript-eslint/no-var-requires': 'off',
'@typescript-eslint/explicit-function-return-type': 'off',
'@typescript-eslint/explicit-module-boundary-types': 'off',
},
},
{
files: 'eslint-rules/**/*',
extends: ['eslint:recommended'],
rules: {
'@typescript-eslint/no-var-requires': 'off',
},
Expand Down
6 changes: 3 additions & 3 deletions .github/workflows/ci-frontend.yml
Original file line number Diff line number Diff line change
Expand Up @@ -49,15 +49,15 @@ jobs:
- name: Check formatting with prettier
run: pnpm prettier:check

- name: Lint with ESLint
run: pnpm lint:js

- name: Lint with Stylelint
run: pnpm lint:css

- name: Generate logic types and run typescript with strict
run: pnpm typegen:write && pnpm typescript:check

- name: Lint with ESLint
run: pnpm lint:js

- name: Check if "schema.json" is up to date
run: pnpm schema:build:json && git diff --exit-code

Expand Down
2 changes: 1 addition & 1 deletion cypress/e2e/actions.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ const createAction = (actionName: string): void => {
cy.get('[data-attr=action-name-create]').should('exist')

cy.get('[data-attr=action-name-create]').type(actionName)
cy.get('.ant-radio-group > :nth-child(3)').click()
cy.get('.LemonSegmentedButton > ul > :nth-child(3)').click()
cy.get('[data-attr=edit-action-url-input]').click().type(Cypress.config().baseUrl)

Check failure on line 9 in cypress/e2e/actions.cy.ts

View workflow job for this annotation

GitHub Actions / Code quality checks

Argument of type 'string | null' is not assignable to parameter of type 'string'.

cy.get('[data-attr=save-action-button]').click()
Expand Down
15 changes: 15 additions & 0 deletions ee/api/test/test_billing.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ def create_missing_billing_customer(**kwargs) -> CustomerInfo:
usage_summary={
"events": {"limit": None, "usage": 0},
"recordings": {"limit": None, "usage": 0},
"rows_synced": {"limit": None, "usage": 0},
},
free_trial_until=None,
available_features=[],
Expand Down Expand Up @@ -96,6 +97,7 @@ def create_billing_customer(**kwargs) -> CustomerInfo:
usage_summary={
"events": {"limit": None, "usage": 0},
"recordings": {"limit": None, "usage": 0},
"rows_synced": {"limit": None, "usage": 0},
},
free_trial_until=None,
)
Expand Down Expand Up @@ -292,6 +294,7 @@ def mock_implementation(url: str, headers: Any = None, params: Any = None) -> Ma
"usage_summary": {
"events": {"limit": None, "usage": 0},
"recordings": {"limit": None, "usage": 0},
"rows_synced": {"limit": None, "usage": 0},
},
"free_trial_until": None,
}
Expand Down Expand Up @@ -363,6 +366,7 @@ def mock_implementation(url: str, headers: Any = None, params: Any = None) -> Ma
"usage_summary": {
"events": {"limit": None, "usage": 0},
"recordings": {"limit": None, "usage": 0},
"rows_synced": {"limit": None, "usage": 0},
},
"free_trial_until": None,
"current_total_amount_usd": "0.00",
Expand Down Expand Up @@ -521,6 +525,11 @@ def mock_implementation(url: str, headers: Any = None, params: Any = None) -> Ma
"todays_usage": 0,
"usage": 0,
},
"rows_synced": {
"limit": None,
"todays_usage": 0,
"usage": 0,
},
"period": ["2022-10-07T11:12:48", "2022-11-07T11:12:48"],
}

Expand Down Expand Up @@ -556,6 +565,11 @@ def mock_implementation_missing_customer(url: str, headers: Any = None, params:
"todays_usage": 0,
"usage": 0,
},
"rows_synced": {
"limit": None,
"todays_usage": 0,
"usage": 0,
},
"period": ["2022-10-07T11:12:48", "2022-11-07T11:12:48"],
}
assert self.organization.customer_id == "cus_123"
Expand Down Expand Up @@ -613,5 +627,6 @@ def mock_implementation(url: str, headers: Any = None, params: Any = None) -> Ma
assert self.organization.usage == {
"events": {"limit": None, "usage": 0, "todays_usage": 0},
"recordings": {"limit": None, "usage": 0, "todays_usage": 0},
"rows_synced": {"limit": None, "usage": 0, "todays_usage": 0},
"period": ["2022-10-07T11:12:48", "2022-11-07T11:12:48"],
}
1 change: 1 addition & 0 deletions ee/billing/billing_manager.py
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,7 @@ def update_org_details(self, organization: Organization, billing_status: Billing
usage_info = OrganizationUsageInfo(
events=usage_summary["events"],
recordings=usage_summary["recordings"],
rows_synced=usage_summary.get("rows_synced", None),
period=[
data["billing_period"]["current_period_start"],
data["billing_period"]["current_period_end"],
Expand Down
59 changes: 44 additions & 15 deletions ee/billing/quota_limiting.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
convert_team_usage_rows_to_dict,
get_teams_with_billable_event_count_in_period,
get_teams_with_recording_count_in_period,
get_teams_with_rows_synced_in_period,
)
from posthog.utils import get_current_day

Expand All @@ -26,11 +27,13 @@
class QuotaResource(Enum):
EVENTS = "events"
RECORDINGS = "recordings"
ROWS_SYNCED = "rows_synced"


OVERAGE_BUFFER = {
QuotaResource.EVENTS: 0,
QuotaResource.RECORDINGS: 1000,
QuotaResource.ROWS_SYNCED: 0,
}


Expand All @@ -53,7 +56,7 @@ def remove_limited_team_tokens(resource: QuotaResource, tokens: List[str]) -> No


@cache_for(timedelta(seconds=30), background_refresh=True)
def list_limited_team_tokens(resource: QuotaResource) -> List[str]:
def list_limited_team_attributes(resource: QuotaResource) -> List[str]:
now = timezone.now()
redis_client = get_client()
results = redis_client.zrangebyscore(f"{QUOTA_LIMITER_CACHE_KEY}{resource.value}", min=now.timestamp(), max="+inf")
Expand All @@ -63,13 +66,16 @@ def list_limited_team_tokens(resource: QuotaResource) -> List[str]:
class UsageCounters(TypedDict):
events: int
recordings: int
rows_synced: int


def org_quota_limited_until(organization: Organization, resource: QuotaResource) -> Optional[int]:
if not organization.usage:
return None

summary = organization.usage.get(resource.value, {})
if not summary:
return None
usage = summary.get("usage", 0)
todays_usage = summary.get("todays_usage", 0)
limit = summary.get("limit")
Expand All @@ -93,19 +99,34 @@ def sync_org_quota_limits(organization: Organization):
if not organization.usage:
return None

team_tokens: List[str] = [x for x in list(organization.teams.values_list("api_token", flat=True)) if x]

if not team_tokens:
capture_exception(Exception(f"quota_limiting: No team tokens found for organization: {organization.id}"))
return

for resource in [QuotaResource.EVENTS, QuotaResource.RECORDINGS]:
for resource in [QuotaResource.EVENTS, QuotaResource.RECORDINGS, QuotaResource.ROWS_SYNCED]:
team_attributes = get_team_attribute_by_quota_resource(organization, resource)
quota_limited_until = org_quota_limited_until(organization, resource)

if quota_limited_until:
add_limited_team_tokens(resource, {x: quota_limited_until for x in team_tokens})
add_limited_team_tokens(resource, {x: quota_limited_until for x in team_attributes})
else:
remove_limited_team_tokens(resource, team_tokens)
remove_limited_team_tokens(resource, team_attributes)


def get_team_attribute_by_quota_resource(organization: Organization, resource: QuotaResource):
if resource in [QuotaResource.EVENTS, QuotaResource.RECORDINGS]:
team_tokens: List[str] = [x for x in list(organization.teams.values_list("api_token", flat=True)) if x]

if not team_tokens:
capture_exception(Exception(f"quota_limiting: No team tokens found for organization: {organization.id}"))
return

return team_tokens

if resource == QuotaResource.ROWS_SYNCED:
team_ids: List[str] = [x for x in list(organization.teams.values_list("id", flat=True)) if x]

if not team_ids:
capture_exception(Exception(f"quota_limiting: No team ids found for organization: {organization.id}"))
return

return team_ids


def set_org_usage_summary(
Expand All @@ -125,8 +146,10 @@ def set_org_usage_summary(

new_usage = copy.deepcopy(new_usage)

for field in ["events", "recordings"]:
for field in ["events", "recordings", "rows_synced"]:
resource_usage = new_usage[field] # type: ignore
if not resource_usage:
continue

if todays_usage:
resource_usage["todays_usage"] = todays_usage[field] # type: ignore
Expand Down Expand Up @@ -155,6 +178,9 @@ def update_all_org_billing_quotas(dry_run: bool = False) -> Dict[str, Dict[str,
teams_with_recording_count_in_period=convert_team_usage_rows_to_dict(
get_teams_with_recording_count_in_period(period_start, period_end)
),
teams_with_rows_synced_in_period=convert_team_usage_rows_to_dict(
get_teams_with_rows_synced_in_period(period_start, period_end)
),
)

teams: Sequence[Team] = list(
Expand All @@ -171,6 +197,7 @@ def update_all_org_billing_quotas(dry_run: bool = False) -> Dict[str, Dict[str,
team_report = UsageCounters(
events=all_data["teams_with_event_count_in_period"].get(team.id, 0),
recordings=all_data["teams_with_recording_count_in_period"].get(team.id, 0),
rows_synced=all_data["teams_with_rows_synced_in_period"].get(team.id, 0),
)

org_id = str(team.organization.id)
Expand All @@ -183,7 +210,7 @@ def update_all_org_billing_quotas(dry_run: bool = False) -> Dict[str, Dict[str,
for field in team_report:
org_report[field] += team_report[field] # type: ignore

quota_limited_orgs: Dict[str, Dict[str, int]] = {"events": {}, "recordings": {}}
quota_limited_orgs: Dict[str, Dict[str, int]] = {"events": {}, "recordings": {}, "rows_synced": {}}

# We find all orgs that should be rate limited
for org_id, todays_report in todays_usage_report.items():
Expand All @@ -195,7 +222,7 @@ def update_all_org_billing_quotas(dry_run: bool = False) -> Dict[str, Dict[str,
if set_org_usage_summary(org, todays_usage=todays_report):
org.save(update_fields=["usage"])

for field in ["events", "recordings"]:
for field in ["events", "recordings", "rows_synced"]:
quota_limited_until = org_quota_limited_until(org, QuotaResource(field))

if quota_limited_until:
Expand All @@ -207,12 +234,13 @@ def update_all_org_billing_quotas(dry_run: bool = False) -> Dict[str, Dict[str,
previously_quota_limited_team_tokens: Dict[str, Dict[str, int]] = {
"events": {},
"recordings": {},
"rows_synced": {},
}

for field in quota_limited_orgs:
previously_quota_limited_team_tokens[field] = list_limited_team_tokens(QuotaResource(field))
previously_quota_limited_team_tokens[field] = list_limited_team_attributes(QuotaResource(field))

quota_limited_teams: Dict[str, Dict[str, int]] = {"events": {}, "recordings": {}}
quota_limited_teams: Dict[str, Dict[str, int]] = {"events": {}, "recordings": {}, "rows_synced": {}}

# Convert the org ids to team tokens
for team in teams:
Expand All @@ -233,6 +261,7 @@ def update_all_org_billing_quotas(dry_run: bool = False) -> Dict[str, Dict[str,
properties = {
"quota_limited_events": quota_limited_orgs["events"].get(org_id, None),
"quota_limited_recordings": quota_limited_orgs["events"].get(org_id, None),
"quota_limited_rows_synced": quota_limited_orgs["rows_synced"].get(org_id, None),
}

report_organization_action(
Expand Down
Loading

0 comments on commit fe3d792

Please sign in to comment.