UNPKG

@web-widget/flags-kit

Version:

Flags SDK by Vercel - The feature flags toolkit for Next.js, SvelteKit, and Web Router - Enhanced fork with improved Web Router support

353 lines (349 loc) 11.1 kB
import { deserialize, normalizeOptions, serialize } from "./chunk-O6VYPARG.js"; import { HeadersAdapter, RequestCookiesAdapter, decryptFlagDefinitions, decryptFlagValues, decryptOverrides, encryptFlagDefinitions, encryptFlagValues, encryptOverrides, reportValue, verifyAccess, version } from "./chunk-BQ2THYCT.js"; import { safeJsonStringify } from "./chunk-2UDLZC33.js"; // src/sveltekit/index.ts import { error, json } from "@sveltejs/kit"; import { AsyncLocalStorage } from "node:async_hooks"; import { RequestCookies } from "@edge-runtime/cookies"; // src/sveltekit/precompute.ts async function evaluate(flags, request) { return Promise.all(flags.map((flag2) => flag2(request))); } async function precompute(flags, request, secret) { const values = await evaluate(flags, request); return serialize2(flags, values, secret); } function combine(flags, values) { return Object.fromEntries(flags.map((flag2, i) => [flag2.key, values[i]])); } async function serialize2(flags, values, secret) { return serialize(combine(flags, values), flags, secret); } async function deserialize2(flags, code, secret) { return deserialize(code, flags, secret); } async function getPrecomputed(flagKey, precomputeFlags, code, secret) { const flagSet = await deserialize2(precomputeFlags, code, secret); return flagSet[flagKey]; } function* cartesianIterator(items) { const remainder = items.length > 1 ? cartesianIterator(items.slice(1)) : [[]]; for (let r of remainder) for (let h of items.at(0)) yield [h, ...r]; } async function generatePermutations(flags, filter = null, secret) { const options = flags.map((flag2) => { if (!flag2.options) return [false, true]; return flag2.options.map((option) => option.value); }); const list = []; for (const permutation of cartesianIterator(options)) { const permObject = permutation.reduce( (acc, value, index) => { acc[flags[index].key] = value; return acc; }, {} ); if (!filter || filter(permObject)) list.push(permObject); } return Promise.all(list.map((values) => serialize(values, flags, secret))); } // src/sveltekit/env.ts var default_secret = process.env.FLAGS_SECRET; async function tryGetSecret(secret) { if (!default_secret) { try { const env = await import("$env/static/private"); default_secret = env.FLAGS_SECRET; } catch (e) { } } secret = secret || default_secret; if (!secret) { throw new Error( "flags: No secret provided. Set an environment variable FLAGS_SECRET or provide a secret to the function." ); } return secret; } // src/sveltekit/index.ts function hasOwnProperty(obj, prop) { return obj.hasOwnProperty(prop); } var headersMap = /* @__PURE__ */ new WeakMap(); var cookiesMap = /* @__PURE__ */ new WeakMap(); function sealHeaders(headers) { const cached = headersMap.get(headers); if (cached !== void 0) return cached; const sealed = HeadersAdapter.seal( headers, "Headers cannot be modified in SvelteKit. Headers are read-only during request processing." ); headersMap.set(headers, sealed); return sealed; } function sealCookies(headers) { const cached = cookiesMap.get(headers); if (cached !== void 0) return cached; const sealed = RequestCookiesAdapter.seal( new RequestCookies(headers), "Cookies cannot be modified in SvelteKit. Use the cookies API from @sveltejs/kit to set cookies. Read more: https://kit.svelte.dev/docs/load#cookies" ); cookiesMap.set(headers, sealed); return sealed; } async function resolveObjectPromises(obj) { const entries = Object.entries(obj); const resolvedEntries = await Promise.all( entries.map(async ([key, promise]) => { const value = await promise; return [key, value]; }) ); return Object.fromEntries(resolvedEntries); } function getDecide(definition) { return function decide(params) { if (typeof definition.decide === "function") { return definition.decide(params); } if (typeof definition.adapter?.decide === "function") { return definition.adapter.decide({ key: definition.key, ...params }); } throw new Error(`flags: No decide function provided for ${definition.key}`); }; } function getIdentify(definition) { if (typeof definition.identify === "function") { return definition.identify; } if (typeof definition.adapter?.identify === "function") { return definition.adapter.identify; } } var requestMap = /* @__PURE__ */ new WeakMap(); function flag(definition) { const decide = getDecide(definition); const identify = getIdentify(definition); const flagImpl = async function flagImpl2(requestOrCode, flagsArrayOrSecret) { let store = flagStorage.getStore(); if (!store) { if (requestOrCode instanceof Request) { store = requestMap.get(requestOrCode); if (!store) { store = createContext( requestOrCode, flagsArrayOrSecret ?? await tryGetSecret() ); requestMap.set(requestOrCode, store); } } else { throw new Error("flags: Neither context found nor Request provided"); } } if (typeof requestOrCode === "string" && Array.isArray(flagsArrayOrSecret)) { return getPrecomputed( definition.key, flagsArrayOrSecret, requestOrCode, store.secret ); } if (hasOwnProperty(store.usedFlags, definition.key)) { const valuePromise2 = store.usedFlags[definition.key]; if (typeof valuePromise2 !== "undefined") { return valuePromise2; } } const headers = sealHeaders(store.request.headers); const cookies = sealCookies(store.request.headers); const overridesCookie = cookies.get("vercel-flag-overrides")?.value; const overrides = overridesCookie ? await decryptOverrides(overridesCookie, store.secret) : void 0; if (overrides && hasOwnProperty(overrides, definition.key)) { const value2 = overrides[definition.key]; if (typeof value2 !== "undefined") { reportValue(definition.key, value2); store.usedFlags[definition.key] = Promise.resolve(value2); return value2; } } let entities; if (identify) { if (!store.identifiers.has(identify)) { const entities2 = identify({ headers, cookies }); store.identifiers.set(identify, entities2); } entities = await store.identifiers.get(identify); } const valuePromise = decide({ headers, cookies, entities }); store.usedFlags[definition.key] = valuePromise; const value = await valuePromise; reportValue(definition.key, value); return value; }; flagImpl.key = definition.key; flagImpl.defaultValue = definition.defaultValue; flagImpl.origin = definition.origin; flagImpl.description = definition.description; flagImpl.options = normalizeOptions(definition.options); flagImpl.decide = decide; flagImpl.identify = identify; return flagImpl; } function getProviderData(flags) { const definitions = Object.values(flags).reduce( (acc, d) => { acc[d.key] = { options: normalizeOptions(d.options), origin: d.origin, description: d.description }; return acc; }, {} ); return { definitions, hints: [] }; } function createContext(request, secret, params) { return { request, secret, params: params ?? {}, usedFlags: {}, identifiers: /* @__PURE__ */ new Map() }; } var flagStorage = new AsyncLocalStorage(); function createHandle({ secret, flags }) { return async function handle({ event, resolve }) { secret ?? (secret = await tryGetSecret(secret)); if (flags && // avoid creating the URL object for every request by checking with includes() first event.request.url.includes("/.well-known/") && new URL(event.request.url).pathname === "/.well-known/vercel/flags") { return handleWellKnownFlagsRoute(event, secret, flags); } const flagContext = createContext( event.request, secret, event.params ); return flagStorage.run( flagContext, () => resolve(event, { transformPageChunk: async ({ html }) => { const store = flagStorage.getStore(); if (!store || Object.keys(store.usedFlags).length === 0) return html; const encryptedFlagValues = await encryptFlagValues( await resolveObjectPromises(store.usedFlags), secret ); return html.replace( "</body>", `<script type="application/json" data-flag-values>${safeJsonStringify(encryptedFlagValues)}</script></body>` ); } }) ); }; } async function handleWellKnownFlagsRoute(event, secret, flags) { const access = await verifyAccess( event.request.headers.get("Authorization"), secret ); if (!access) return new Response(null, { status: 401 }); const providerData = getProviderData(flags); return Response.json(providerData, { headers: { "x-flags-sdk-version": version } }); } async function encryptFlagValues2(value, secret) { return encryptFlagValues(value, await tryGetSecret(secret)); } async function decryptFlagValues2(encryptedData, secret) { return decryptFlagValues(encryptedData, await tryGetSecret(secret)); } async function encryptOverrides2(overrides, secret) { return encryptOverrides(overrides, await tryGetSecret(secret)); } async function decryptOverrides2(encryptedData, secret) { return decryptOverrides(encryptedData, await tryGetSecret(secret)); } async function encryptFlagDefinitions2(value, secret) { return encryptFlagDefinitions(value, await tryGetSecret(secret)); } async function decryptFlagDefinitions2(encryptedData, secret) { return decryptFlagDefinitions(encryptedData, await tryGetSecret(secret)); } async function precompute2(flags, request, secret) { return precompute(flags, request, await tryGetSecret(secret)); } async function generatePermutations2(flags, filter = null, secret) { return generatePermutations(flags, filter, await tryGetSecret(secret)); } function createFlagsDiscoveryEndpoint(getApiData, options) { const requestHandler = async (event) => { const access = await verifyAccess( event.request.headers.get("Authorization"), options?.secret ); if (!access) error(401); const apiData = await getApiData(event); return json(apiData, { headers: { "x-flags-sdk-version": version } }); }; return requestHandler; } export { createFlagsDiscoveryEndpoint, createHandle, decryptFlagDefinitions2 as decryptFlagDefinitions, decryptFlagValues2 as decryptFlagValues, decryptOverrides2 as decryptOverrides, encryptFlagDefinitions2 as encryptFlagDefinitions, encryptFlagValues2 as encryptFlagValues, encryptOverrides2 as encryptOverrides, flag, generatePermutations2 as generatePermutations, getProviderData, precompute2 as precompute }; //# sourceMappingURL=sveltekit.js.map