From 4887ca261ebd2c2d87e0117349d3ba1ba66dfe2f Mon Sep 17 00:00:00 2001 From: Sam Rose Date: Fri, 9 Aug 2024 11:27:43 +0100 Subject: [PATCH] Improve testing of feature flags by not polluting production flags with test ones. --- packages/backend-core/src/environment.ts | 1 + packages/backend-core/src/features/index.ts | 265 ++++++++---------- .../src/features/tests/features.spec.ts | 87 +++--- .../worker/src/api/controllers/global/self.ts | 2 +- 4 files changed, 173 insertions(+), 182 deletions(-) diff --git a/packages/backend-core/src/environment.ts b/packages/backend-core/src/environment.ts index 5091a4971a..e064398153 100644 --- a/packages/backend-core/src/environment.ts +++ b/packages/backend-core/src/environment.ts @@ -145,6 +145,7 @@ const environment = { COOKIE_DOMAIN: process.env.COOKIE_DOMAIN, PLATFORM_URL: process.env.PLATFORM_URL || "", POSTHOG_TOKEN: process.env.POSTHOG_TOKEN, + POSTHOG_API_HOST: process.env.POSTHOG_API_HOST || "https://us.i.posthog.com", ENABLE_ANALYTICS: process.env.ENABLE_ANALYTICS, TENANT_FEATURE_FLAGS: process.env.TENANT_FEATURE_FLAGS, CLOUDFRONT_CDN: process.env.CLOUDFRONT_CDN, diff --git a/packages/backend-core/src/features/index.ts b/packages/backend-core/src/features/index.ts index 2ffa35974f..3f0a6cf6b0 100644 --- a/packages/backend-core/src/features/index.ts +++ b/packages/backend-core/src/features/index.ts @@ -1,6 +1,5 @@ import env from "../environment" import * as context from "../context" -import { cloneDeep } from "lodash" import { PostHog, PostHogOptions } from "posthog-node" import { IdentityType } from "@budibase/types" import tracer from "dd-trace" @@ -9,13 +8,13 @@ let posthog: PostHog | undefined export function init(opts?: PostHogOptions) { if (env.POSTHOG_TOKEN) { posthog = new PostHog(env.POSTHOG_TOKEN, { - host: "https://us.i.posthog.com", + host: env.POSTHOG_API_HOST, ...opts, }) } } -abstract class Flag { +export abstract class Flag { static boolean(defaultValue: boolean): Flag { return new BooleanFlag(defaultValue) } @@ -33,6 +32,16 @@ abstract class Flag { abstract parse(value: any): T } +type UnwrapFlag = F extends Flag ? U : never + +export type FlagValues = { + [K in keyof T]: UnwrapFlag +} + +type KeysOfType = { + [K in keyof T]: T[K] extends Flag ? K : never +}[keyof T] + class BooleanFlag extends Flag { parse(value: any) { if (typeof value === "string") { @@ -73,149 +82,123 @@ class NumberFlag extends Flag { } } +export class FlagSet, T extends { [key: string]: V }> { + private readonly flags: T + + constructor(flags: T) { + this.flags = flags + } + + defaults(): FlagValues { + return Object.keys(this.flags).reduce((acc, key) => { + const typedKey = key as keyof T + acc[typedKey] = this.flags[key].defaultValue + return acc + }, {} as FlagValues) + } + + isFlagName(name: string | number | symbol): name is keyof T { + return this.flags[name as keyof T] !== undefined + } + + async get(key: K): Promise[K]> { + const flags = await this.fetch() + return flags[key] + } + + async isEnabled>(key: K): Promise { + const flags = await this.fetch() + return flags[key] + } + + async fetch(): Promise> { + return await tracer.trace("features.fetch", async span => { + const tags: Record = {} + + const flags = this.defaults() + + const currentTenantId = context.getTenantId() + const specificallySetFalse = new Set() + + const split = (env.TENANT_FEATURE_FLAGS || "") + .split(",") + .map(x => x.split(":")) + for (const [tenantId, ...features] of split) { + if (!tenantId || (tenantId !== "*" && tenantId !== currentTenantId)) { + continue + } + + for (let feature of features) { + let value = true + if (feature.startsWith("!")) { + feature = feature.slice(1) + value = false + specificallySetFalse.add(feature) + } + + if (!this.isFlagName(feature)) { + throw new Error(`Feature: ${feature} is not an allowed option`) + } + + if (typeof flags[feature] !== "boolean") { + throw new Error(`Feature: ${feature} is not a boolean`) + } + + // @ts-ignore + flags[feature] = value + tags[`flags.${feature}.source`] = "environment" + } + } + + const identity = context.getIdentity() + if (posthog && identity?.type === IdentityType.USER) { + const posthogFlags = await posthog.getAllFlagsAndPayloads(identity._id) + for (const [name, value] of Object.entries(posthogFlags.featureFlags)) { + const flag = this.flags[name] + if (!flag) { + // We don't want an unexpected PostHog flag to break the app, so we + // just log it and continue. + console.warn(`Unexpected posthog flag "${name}": ${value}`) + continue + } + + if (flags[name] === true || specificallySetFalse.has(name)) { + // If the flag is already set to through environment variables, we + // don't want to override it back to false here. + continue + } + + const payload = posthogFlags.featureFlagPayloads?.[name] + + try { + // @ts-ignore + flags[name] = flag.parse(payload || value) + tags[`flags.${name}.source`] = "posthog" + } catch (err) { + // We don't want an invalid PostHog flag to break the app, so we just + // log it and continue. + console.warn(`Error parsing posthog flag "${name}": ${value}`, err) + } + } + } + + for (const [key, value] of Object.entries(flags)) { + tags[`flags.${key}.value`] = value + } + span?.addTags(tags) + + return flags + }) + } +} + // This is the primary source of truth for feature flags. If you want to add a // new flag, add it here and use the `fetch` and `get` functions to access it. // All of the machinery in this file is to make sure that flags have their // default values set correctly and their types flow through the system. -const FLAGS = { +export const flags = new FlagSet({ LICENSING: Flag.boolean(false), GOOGLE_SHEETS: Flag.boolean(false), USER_GROUPS: Flag.boolean(false), ONBOARDING_TOUR: Flag.boolean(false), - - _TEST_BOOLEAN: Flag.boolean(false), - _TEST_STRING: Flag.string("default value"), - _TEST_NUMBER: Flag.number(0), -} - -const DEFAULTS = Object.keys(FLAGS).reduce((acc, key) => { - const typedKey = key as keyof typeof FLAGS - // @ts-ignore - acc[typedKey] = FLAGS[typedKey].defaultValue - return acc -}, {} as Flags) - -type UnwrapFlag = F extends Flag ? U : never -export type Flags = { - [K in keyof typeof FLAGS]: UnwrapFlag<(typeof FLAGS)[K]> -} - -// Exported for use in tests, should not be used outside of this file. -export function defaultFlags(): Flags { - return cloneDeep(DEFAULTS) -} - -function isFlagName(name: string): name is keyof Flags { - return FLAGS[name as keyof typeof FLAGS] !== undefined -} - -/** - * Reads the TENANT_FEATURE_FLAGS environment variable and returns a Flags object - * populated with the flags for the current tenant, filling in the default values - * if the flag is not set. - * - * Check the tests for examples of how TENANT_FEATURE_FLAGS should be formatted. - * - * In future we plan to add more ways of setting feature flags, e.g. PostHog, and - * they will be accessed through this function as well. - */ -export async function fetch(): Promise { - return await tracer.trace("features.fetch", async span => { - const tags: Record = {} - - const flags = defaultFlags() - - const currentTenantId = context.getTenantId() - const specificallySetFalse = new Set() - - const split = (env.TENANT_FEATURE_FLAGS || "") - .split(",") - .map(x => x.split(":")) - for (const [tenantId, ...features] of split) { - if (!tenantId || (tenantId !== "*" && tenantId !== currentTenantId)) { - continue - } - - for (let feature of features) { - let value = true - if (feature.startsWith("!")) { - feature = feature.slice(1) - value = false - specificallySetFalse.add(feature) - } - - if (!isFlagName(feature)) { - throw new Error(`Feature: ${feature} is not an allowed option`) - } - - if (typeof flags[feature] !== "boolean") { - throw new Error(`Feature: ${feature} is not a boolean`) - } - - // @ts-ignore - flags[feature] = value - tags[`flags.${feature}.source`] = "environment" - } - } - - const identity = context.getIdentity() - if (posthog && identity?.type === IdentityType.USER) { - const posthogFlags = await posthog.getAllFlagsAndPayloads(identity._id) - for (const [name, value] of Object.entries(posthogFlags.featureFlags)) { - const key = name as keyof typeof FLAGS - const flag = FLAGS[key] - if (!flag) { - // We don't want an unexpected PostHog flag to break the app, so we - // just log it and continue. - console.warn(`Unexpected posthog flag "${name}": ${value}`) - continue - } - - if (flags[key] === true || specificallySetFalse.has(key)) { - // If the flag is already set to through environment variables, we - // don't want to override it back to false here. - continue - } - - const payload = posthogFlags.featureFlagPayloads?.[name] - - try { - // @ts-ignore - flags[key] = flag.parse(payload || value) - tags[`flags.${key}.source`] = "posthog" - } catch (err) { - // We don't want an invalid PostHog flag to break the app, so we just - // log it and continue. - console.warn(`Error parsing posthog flag "${name}": ${value}`, err) - } - } - } - - for (const [key, value] of Object.entries(flags)) { - tags[`flags.${key}.value`] = value - } - span?.addTags(tags) - - return flags - }) -} - -// Gets a single feature flag value. This is a convenience function for -// `fetch().then(flags => flags[name])`. -export async function get(name: K): Promise { - const flags = await fetch() - return flags[name] -} - -type BooleanFlags = { - [K in keyof typeof FLAGS]: (typeof FLAGS)[K] extends Flag ? K : never -}[keyof typeof FLAGS] - -// Convenience function for boolean flag values. This makes callsites more -// readable for boolean flags. -export async function isEnabled( - name: K -): Promise { - const flags = await fetch() - return flags[name] -} +}) diff --git a/packages/backend-core/src/features/tests/features.spec.ts b/packages/backend-core/src/features/tests/features.spec.ts index 1b85a74024..95819831b2 100644 --- a/packages/backend-core/src/features/tests/features.spec.ts +++ b/packages/backend-core/src/features/tests/features.spec.ts @@ -1,53 +1,60 @@ import { IdentityContext, IdentityType } from "@budibase/types" -import { defaultFlags, fetch, get, Flags, init } from "../" +import { Flag, FlagSet, FlagValues, init } from "../" import { context } from "../.." import { setEnv, withEnv } from "../../environment" import nodeFetch from "node-fetch" import nock from "nock" +const schema = { + TEST_BOOLEAN: Flag.boolean(false), + TEST_STRING: Flag.string("default value"), + TEST_NUMBER: Flag.number(0), +} +const flags = new FlagSet(schema) + describe("feature flags", () => { interface TestCase { tenant: string flags: string - expected: Partial + expected: Partial> } it.each([ { tenant: "tenant1", - flags: "tenant1:_TEST_BOOLEAN", - expected: { _TEST_BOOLEAN: true }, + flags: "tenant1:TEST_BOOLEAN", + expected: { TEST_BOOLEAN: true }, }, { tenant: "tenant1", - flags: "tenant1:!_TEST_BOOLEAN", - expected: { _TEST_BOOLEAN: false }, + flags: "tenant1:!TEST_BOOLEAN", + expected: { TEST_BOOLEAN: false }, }, { tenant: "tenant1", - flags: "*:_TEST_BOOLEAN", - expected: { _TEST_BOOLEAN: true }, + flags: "*:TEST_BOOLEAN", + expected: { TEST_BOOLEAN: true }, }, { tenant: "tenant1", - flags: "tenant2:_TEST_BOOLEAN", - expected: { _TEST_BOOLEAN: false }, + flags: "tenant2:TEST_BOOLEAN", + expected: { TEST_BOOLEAN: false }, }, { tenant: "tenant1", flags: "", - expected: defaultFlags(), + expected: flags.defaults(), }, ])( 'should find flags $expected for $tenant with string "$flags"', - ({ tenant, flags, expected }) => + ({ tenant, flags: envFlags, expected }) => context.doInTenant(tenant, async () => - withEnv({ TENANT_FEATURE_FLAGS: flags }, async () => { - const flags = await fetch() - expect(flags).toMatchObject(expected) + withEnv({ TENANT_FEATURE_FLAGS: envFlags }, async () => { + const values = await flags.fetch() + expect(values).toMatchObject(expected) for (const [key, expectedValue] of Object.entries(expected)) { - const value = await get(key as keyof Flags) + const value = await flags.get(key as keyof typeof schema) expect(value).toBe(expectedValue) } }) @@ -63,15 +70,15 @@ describe("feature flags", () => { it.each([ { tenant: "tenant1", - flags: "tenant1:_TEST_BOOLEAN,tenant1:FOO", + flags: "tenant1:TEST_BOOLEAN,tenant1:FOO", expected: "Feature: FOO is not an allowed option", }, ])( "should fail with message \"$expected\" for $tenant with string '$flags'", - ({ tenant, flags, expected }) => + ({ tenant, flags: envFlags, expected }) => context.doInTenant(tenant, () => - withEnv({ TENANT_FEATURE_FLAGS: flags }, () => - expect(fetch).rejects.toThrow(expected) + withEnv({ TENANT_FEATURE_FLAGS: envFlags }, () => + expect(flags.fetch()).rejects.toThrow(expected) ) ) ) @@ -118,45 +125,45 @@ describe("feature flags", () => { it("should be able to read flags from posthog", async () => { mockFlags({ featureFlags: { - _TEST_BOOLEAN: true, + TEST_BOOLEAN: true, }, }) await context.doInIdentityContext(identity, async () => { - const flags = await fetch() - expect(flags._TEST_BOOLEAN).toBe(true) + const values = await flags.fetch() + expect(values.TEST_BOOLEAN).toBe(true) }) }) it("should be able to read flags from posthog with payloads", async () => { mockFlags({ featureFlags: { - _TEST_STRING: true, + TEST_STRING: true, }, featureFlagPayloads: { - _TEST_STRING: "test payload", + TEST_STRING: "test payload", }, }) await context.doInIdentityContext(identity, async () => { - const flags = await fetch() - expect(flags._TEST_STRING).toBe("test payload") + const values = await flags.fetch() + expect(values.TEST_STRING).toBe("test payload") }) }) it("should be able to read flags from posthog with numbers", async () => { mockFlags({ featureFlags: { - _TEST_NUMBER: true, + TEST_NUMBER: true, }, featureFlagPayloads: { - _TEST_NUMBER: 123, + TEST_NUMBER: 123, }, }) await context.doInIdentityContext(identity, async () => { - const flags = await fetch() - expect(flags._TEST_NUMBER).toBe(123) + const values = await flags.fetch() + expect(values.TEST_NUMBER).toBe(123) }) }) @@ -168,23 +175,23 @@ describe("feature flags", () => { }) await context.doInIdentityContext(identity, async () => { - await expect(fetch()).resolves.not.toThrow() + await expect(flags.fetch()).resolves.not.toThrow() }) }) it("should not override flags set in the environment", async () => { mockFlags({ featureFlags: { - _TEST_BOOLEAN: false, + TEST_BOOLEAN: false, }, }) await withEnv( - { TENANT_FEATURE_FLAGS: `${identity.tenantId}:_TEST_BOOLEAN` }, + { TENANT_FEATURE_FLAGS: `${identity.tenantId}:TEST_BOOLEAN` }, async () => { await context.doInIdentityContext(identity, async () => { - const flags = await fetch() - expect(flags._TEST_BOOLEAN).toBe(true) + const values = await flags.fetch() + expect(values.TEST_BOOLEAN).toBe(true) }) } ) @@ -193,16 +200,16 @@ describe("feature flags", () => { it("should not override flags set in the environment with a ! prefix", async () => { mockFlags({ featureFlags: { - _TEST_BOOLEAN: true, + TEST_BOOLEAN: true, }, }) await withEnv( - { TENANT_FEATURE_FLAGS: `${identity.tenantId}:!_TEST_BOOLEAN` }, + { TENANT_FEATURE_FLAGS: `${identity.tenantId}:!TEST_BOOLEAN` }, async () => { await context.doInIdentityContext(identity, async () => { - const flags = await fetch() - expect(flags._TEST_BOOLEAN).toBe(false) + const values = await flags.fetch() + expect(values.TEST_BOOLEAN).toBe(false) }) } ) diff --git a/packages/worker/src/api/controllers/global/self.ts b/packages/worker/src/api/controllers/global/self.ts index ec154adf7f..14a2ce8c73 100644 --- a/packages/worker/src/api/controllers/global/self.ts +++ b/packages/worker/src/api/controllers/global/self.ts @@ -104,7 +104,7 @@ export async function getSelf(ctx: any) { ctx.body = await groups.enrichUserRolesFromGroups(user) // add the feature flags for this tenant - const flags = await features.fetch() + const flags = await features.flags.fetch() ctx.body.flags = flags addSessionAttributesToUser(ctx)