Skip to content

Commit

Permalink
Cache feature flags per-request, set default values flag to false by …
Browse files Browse the repository at this point in the history
…default.
  • Loading branch information
samwho committed Aug 14, 2024
1 parent 4623fd4 commit 08a56ef
Show file tree
Hide file tree
Showing 3 changed files with 34 additions and 8 deletions.
3 changes: 3 additions & 0 deletions packages/backend-core/src/context/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,4 +18,7 @@ export type ContextMap = {
oauthClient: OAuth2Client
clients: Record<string, GoogleSpreadsheet>
}
featureFlagCache?: {
[key: string]: Record<string, any>
}
}
35 changes: 28 additions & 7 deletions packages/backend-core/src/features/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,10 @@ export function init(opts?: PostHogOptions) {
}
}

export function shutdown() {
posthog?.shutdown()
}

export abstract class Flag<T> {
static boolean(defaultValue: boolean): Flag<boolean> {
return new BooleanFlag(defaultValue)
Expand Down Expand Up @@ -87,7 +91,14 @@ class NumberFlag extends Flag<number> {
}

export class FlagSet<V extends Flag<any>, T extends { [key: string]: V }> {
constructor(private readonly flagSchema: T) {}
// This is used to safely cache flags sets in the current request context.
// Because multiple sets could theoretically exist, we don't want the cache of
// one to leak into another.
private readonly setId: string

constructor(private readonly flagSchema: T) {
this.setId = crypto.randomUUID()
}

defaults(): FlagValues<T> {
return Object.keys(this.flagSchema).reduce((acc, key) => {
Expand Down Expand Up @@ -119,6 +130,15 @@ export class FlagSet<V extends Flag<any>, T extends { [key: string]: V }> {

async fetch(ctx?: UserCtx): Promise<FlagValues<T>> {
return await tracer.trace("features.fetch", async span => {
const requestContext = context.getCurrentContext()
const cachedFlags = requestContext?.featureFlagCache?.[this.setId] as
| FlagValues<T>
| undefined
if (cachedFlags) {
span?.addTags({ fromCache: true })
return cachedFlags
}

const tags: Record<string, any> = {}
const flagValues = this.defaults()
const currentTenantId = context.getTenantId()
Expand Down Expand Up @@ -187,10 +207,7 @@ export class FlagSet<V extends Flag<any>, T extends { [key: string]: V }> {
tags[`identity.tenantId`] = identity?.tenantId
tags[`identity._id`] = identity?._id

// Until we're confident this performs well, we're only enabling it in QA
// and test environments.
const usePosthog = env.isTest() || env.isQA()
if (usePosthog && posthog && identity?.type === IdentityType.USER) {
if (posthog && identity?.type === IdentityType.USER) {
tags[`readFromPostHog`] = true

const personProperties: Record<string, string> = {}
Expand All @@ -204,7 +221,6 @@ export class FlagSet<V extends Flag<any>, T extends { [key: string]: V }> {
personProperties,
}
)
console.log("posthog flags", JSON.stringify(posthogFlags))

for (const [name, value] of Object.entries(posthogFlags.featureFlags)) {
if (!this.isFlagName(name)) {
Expand Down Expand Up @@ -236,6 +252,11 @@ export class FlagSet<V extends Flag<any>, T extends { [key: string]: V }> {
}
}

if (requestContext) {
requestContext.featureFlagCache ??= {}
requestContext.featureFlagCache[this.setId] = flagValues
}

for (const [key, value] of Object.entries(flagValues)) {
tags[`flags.${key}.value`] = value
}
Expand All @@ -255,5 +276,5 @@ export const flags = new FlagSet({
GOOGLE_SHEETS: Flag.boolean(false),
USER_GROUPS: Flag.boolean(false),
ONBOARDING_TOUR: Flag.boolean(false),
DEFAULT_VALUES: Flag.boolean(true),
DEFAULT_VALUES: Flag.boolean(false),
})
4 changes: 3 additions & 1 deletion packages/backend-core/src/features/tests/features.spec.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { IdentityContext, IdentityType, UserCtx } from "@budibase/types"
import { Flag, FlagSet, FlagValues, init } from "../"
import { Flag, FlagSet, FlagValues, init, shutdown } from "../"
import { context } from "../.."
import environment, { withEnv } from "../../environment"
import nodeFetch from "node-fetch"
Expand Down Expand Up @@ -197,6 +197,8 @@ describe("feature flags", () => {
throw new Error("No expected value")
}
})

shutdown()
})
}
)
Expand Down

0 comments on commit 08a56ef

Please sign in to comment.