diff --git a/.gitmodules b/.gitmodules index 3ef2544c8..fcf6412d8 100644 --- a/.gitmodules +++ b/.gitmodules @@ -5,14 +5,14 @@ path = libs/providers/flagd-web/schemas url = https://github.com/open-feature/schemas [submodule "libs/providers/flagd/spec"] - path = libs/providers/flagd/spec - url = https://github.com/open-feature/spec.git -[submodule "libs/providers/flagd-web/spec"] - path = libs/providers/flagd-web/spec - url = https://github.com/open-feature/spec.git + path = libs/providers/flagd/spec + url = https://github.com/open-feature/spec.git [submodule "libs/shared/flagd-core/flagd-schemas"] path = libs/shared/flagd-core/flagd-schemas url = https://github.com/open-feature/flagd-schemas.git -[submodule "libs/providers/flagd/flagd-testbed"] - path = libs/providers/flagd/flagd-testbed - url = https://github.com/open-feature/flagd-testbed.git +[submodule "libs/shared/flagd-core/test-harness"] + path = libs/shared/flagd-core/test-harness + url = https://github.com/open-feature/flagd-testbed +[submodule "libs/shared/flagd-core/spec"] + path = libs/shared/flagd-core/spec + url = https://github.com/open-feature/spec diff --git a/libs/providers/flagd-web/project.json b/libs/providers/flagd-web/project.json index 428f1b7db..aa02170bc 100644 --- a/libs/providers/flagd-web/project.json +++ b/libs/providers/flagd-web/project.json @@ -58,18 +58,6 @@ } ] }, - "pullTestHarness": { - "executor": "nx:run-commands", - "options": { - "commands": [ - "git submodule update --init spec", - "rm -f -r ./src/e2e/features/*", - "cp -v ./spec/specification/assets/gherkin/evaluation.feature ./src/e2e/features/" - ], - "cwd": "libs/providers/flagd-web", - "parallel": false - } - }, "e2e": { "executor": "nx:run-commands", "options": { @@ -84,7 +72,7 @@ "target": "generate" }, { - "target": "pullTestHarness" + "target": "flagd-core:pullTestHarness" } ] }, diff --git a/libs/providers/flagd-web/schemas b/libs/providers/flagd-web/schemas index 1ba4b0324..76d611fd9 160000 --- a/libs/providers/flagd-web/schemas +++ b/libs/providers/flagd-web/schemas @@ -1 +1 @@ -Subproject commit 1ba4b0324771273e6db7d4a808f6dc87a0b3c717 +Subproject commit 76d611fd94689d906af316105ac12670d40f7648 diff --git a/libs/providers/flagd-web/spec b/libs/providers/flagd-web/spec deleted file mode 160000 index eaa44da21..000000000 --- a/libs/providers/flagd-web/spec +++ /dev/null @@ -1 +0,0 @@ -Subproject commit eaa44da2110d97376d3292ea3963c54bea498a77 diff --git a/libs/providers/flagd-web/src/e2e/constants.ts b/libs/providers/flagd-web/src/e2e/constants.ts index 41d547459..d51d19f0e 100644 --- a/libs/providers/flagd-web/src/e2e/constants.ts +++ b/libs/providers/flagd-web/src/e2e/constants.ts @@ -1,3 +1,5 @@ +import { getGherkinTestPath } from '@openfeature/flagd-core'; + export const FLAGD_NAME = 'flagd-web'; -export const IMAGE_VERSION = 'v0.5.6'; +export const GHERKIN_EVALUATION_FEATURE = getGherkinTestPath('flagd.feature'); diff --git a/libs/providers/flagd-web/src/e2e/features/.gitignore b/libs/providers/flagd-web/src/e2e/features/.gitignore deleted file mode 100644 index f9e3307dd..000000000 --- a/libs/providers/flagd-web/src/e2e/features/.gitignore +++ /dev/null @@ -1 +0,0 @@ -*.feature diff --git a/libs/providers/flagd-web/src/e2e/features/.gitkeep b/libs/providers/flagd-web/src/e2e/features/.gitkeep deleted file mode 100644 index e69de29bb..000000000 diff --git a/libs/providers/flagd-web/src/e2e/index.ts b/libs/providers/flagd-web/src/e2e/index.ts new file mode 100644 index 000000000..ac4cdfd4c --- /dev/null +++ b/libs/providers/flagd-web/src/e2e/index.ts @@ -0,0 +1,2 @@ +export * from './constants'; +export * from './step-definitions'; diff --git a/libs/providers/flagd-web/src/e2e/jest.config.ts b/libs/providers/flagd-web/src/e2e/jest.config.ts index 29de43ff6..8fec2f6ce 100644 --- a/libs/providers/flagd-web/src/e2e/jest.config.ts +++ b/libs/providers/flagd-web/src/e2e/jest.config.ts @@ -4,6 +4,7 @@ export default { '^.+\\.[tj]s$': ['ts-jest', { tsconfig: './tsconfig.lib.json' }], }, moduleNameMapper: { + '@openfeature/flagd-core': ['/../../../../shared/flagd-core/src'], '^(.*)\\.js$': ['$1.js', '$1.ts', '$1'], }, testEnvironment: 'node', diff --git a/libs/providers/flagd-web/src/e2e/step-definitions/evaluation.ts b/libs/providers/flagd-web/src/e2e/step-definitions/evaluation.ts deleted file mode 100644 index 27ebd7b46..000000000 --- a/libs/providers/flagd-web/src/e2e/step-definitions/evaluation.ts +++ /dev/null @@ -1,339 +0,0 @@ -import { - EvaluationContext, - EvaluationDetails, - JsonObject, - JsonValue, - OpenFeature, - ProviderEvents, - ResolutionDetails, - StandardResolutionReasons, -} from '@openfeature/web-sdk'; -import { defineFeature, loadFeature } from 'jest-cucumber'; - -export function evaluation() { - // load the feature file. - const feature = loadFeature('features/evaluation.feature'); - - // get a client (flagd provider registered in setup) - const client = OpenFeature.getClient(); - - const givenAnOpenfeatureClientIsRegistered = ( - given: (stepMatcher: string, stepDefinitionCallback: () => void) => void, - ) => { - given('a provider is registered', () => undefined); - }; - - defineFeature(feature, (test) => { - beforeAll((done) => { - client.addHandler(ProviderEvents.Ready, async () => { - done(); - }); - }); - - test('Resolves boolean value', ({ given, when, then }) => { - let value: boolean; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, - (key: string, defaultValue: string) => { - value = client.getBooleanValue(key, defaultValue === 'true'); - }, - ); - - then(/^the resolved boolean value should be "(.*)"$/, (expectedValue: string) => { - expect(value).toEqual(expectedValue === 'true'); - }); - }); - - test('Resolves string value', ({ given, when, then }) => { - let value: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a string flag with key "(.*)" is evaluated with default value "(.*)"$/, - (key: string, defaultValue: string) => { - value = client.getStringValue(key, defaultValue); - }, - ); - - then(/^the resolved string value should be "(.*)"$/, (expectedValue: string) => { - expect(value).toEqual(expectedValue); - }); - }); - - test('Resolves integer value', ({ given, when, then }) => { - let value: number; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^an integer flag with key "(.*)" is evaluated with default value (\d+)$/, - (key: string, defaultValue: number) => { - value = client.getNumberValue(key, defaultValue); - }, - ); - - then(/^the resolved integer value should be (\d+)$/, (expectedStringValue: string) => { - const expectedNumberValue = parseInt(expectedStringValue); - expect(value).toEqual(expectedNumberValue); - }); - }); - - test('Resolves float value', ({ given, when, then }) => { - let value: number; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a float flag with key "(.*)" is evaluated with default value (\d+\.?\d*)$/, - (key: string, defaultValue: string) => { - value = client.getNumberValue(key, Number.parseFloat(defaultValue)); - }, - ); - - then(/^the resolved float value should be (\d+\.?\d*)$/, (expectedValue: string) => { - expect(value).toEqual(Number.parseFloat(expectedValue)); - }); - }); - - test('Resolves object value', ({ given, when, then }) => { - let value: JsonValue; - givenAnOpenfeatureClientIsRegistered(given); - - when(/^an object flag with key "(.*)" is evaluated with a null default value$/, (key: string) => { - value = client.getObjectValue(key, {}); - }); - - then( - /^the resolved object value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, - (field1: string, field2: string, field3: string, boolVal: string, strVal: string, intVal: string) => { - const jsonObject = value as JsonObject; - expect(jsonObject[field1]).toEqual(boolVal === 'true'); - expect(jsonObject[field2]).toEqual(strVal); - expect(jsonObject[field3]).toEqual(Number.parseInt(intVal)); - }, - ); - }); - - test('Resolves boolean details', ({ given, when, then }) => { - let details: EvaluationDetails; - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a boolean flag with key "(.*)" is evaluated with details and default value "(.*)"$/, - (key: string, defaultValue: string) => { - details = client.getBooleanDetails(key, defaultValue === 'true'); - }, - ); - - then( - /^the resolved boolean details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(expectedValue === 'true'); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves string details', ({ given, when, then }) => { - let details: EvaluationDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a string flag with key "(.*)" is evaluated with details and default value "(.*)"$/, - (key: string, defaultValue: string) => { - details = client.getStringDetails(key, defaultValue); - }, - ); - - then( - /^the resolved string details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(expectedValue); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves integer details', ({ given, when, then }) => { - let details: EvaluationDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^an integer flag with key "(.*)" is evaluated with details and default value (\d+)$/, - (key: string, defaultValue: string) => { - details = client.getNumberDetails(key, Number.parseInt(defaultValue)); - }, - ); - - then( - /^the resolved integer details value should be (\d+), the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(Number.parseInt(expectedValue)); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves float details', ({ given, when, then }) => { - let details: EvaluationDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a float flag with key "(.*)" is evaluated with details and default value (\d+\.?\d*)$/, - (key: string, defaultValue: string) => { - details = client.getNumberDetails(key, Number.parseFloat(defaultValue)); - }, - ); - - then( - /^the resolved float details value should be (\d+\.?\d*), the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(Number.parseFloat(expectedValue)); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves object details', ({ given, when, then, and }) => { - let details: EvaluationDetails; // update this after merge - - givenAnOpenfeatureClientIsRegistered(given); - - when(/^an object flag with key "(.*)" is evaluated with details and a null default value$/, (key: string) => { - details = client.getObjectDetails(key, {}); // update this after merge - }); - - then( - /^the resolved object details value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, - (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { - const jsonObject = details.value as JsonObject; - - expect(jsonObject[field1]).toEqual(boolValue === 'true'); - expect(jsonObject[field2]).toEqual(stringValue); - expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); - }, - ); - - and( - /^the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedVariant: string, expectedReason: string) => { - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves based on context', ({ given, when, and, then }) => { - const context: EvaluationContext = {}; - let value: string; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^context contains keys "(.*)", "(.*)", "(.*)", "(.*)" with values "(.*)", "(.*)", (\d+), "(.*)"$/, - async ( - stringField1: string, - stringField2: string, - intField: string, - boolField: string, - stringValue1: string, - stringValue2: string, - intValue: string, - boolValue: string, - ) => { - context[stringField1] = stringValue1; - context[stringField2] = stringValue2; - context[intField] = Number.parseInt(intValue); - context[boolField] = boolValue === 'true'; - - await OpenFeature.setContext(context); - }, - ); - - and(/^a flag with key "(.*)" is evaluated with default value "(.*)"$/, (key: string, defaultValue: string) => { - flagKey = key; - value = client.getStringValue(flagKey, defaultValue); - }); - - then(/^the resolved string response should be "(.*)"$/, (expectedValue: string) => { - expect(value).toEqual(expectedValue); - }); - - and(/^the resolved flag value is "(.*)" when the context is empty$/, async (expectedValue) => { - await OpenFeature.setContext({}); - const emptyContextValue = client.getStringValue(flagKey, 'nope', {}); - expect(emptyContextValue).toEqual(expectedValue); - }); - }); - - test('Flag not found', ({ given, when, then, and }) => { - let flagKey: string; - let fallbackValue: string; - let details: ResolutionDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a non-existent string flag with key "(.*)" is evaluated with details and a default value "(.*)"$/, - (key: string, defaultValue: string) => { - flagKey = key; - fallbackValue = defaultValue; - details = client.getStringDetails(flagKey, defaultValue); - }, - ); - - then(/^the default string value should be returned$/, () => { - expect(details.value).toEqual(fallbackValue); - }); - - and( - /^the reason should indicate an error and the error code should indicate a missing flag with "(.*)"$/, - (errorCode: string) => { - expect(details.reason).toEqual(StandardResolutionReasons.ERROR); - expect(details.errorCode).toEqual(errorCode); - }, - ); - }); - - test('Type error', ({ given, when, then, and }) => { - let flagKey: string; - let fallbackValue: number; - let details: ResolutionDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a string flag with key "(.*)" is evaluated as an integer, with details and a default value (\d+)$/, - (key: string, defaultValue: string) => { - flagKey = key; - fallbackValue = Number.parseInt(defaultValue); - details = client.getNumberDetails(flagKey, Number.parseInt(defaultValue)); - }, - ); - - then(/^the default integer value should be returned$/, () => { - expect(details.value).toEqual(fallbackValue); - }); - - and( - /^the reason should indicate an error and the error code should indicate a type mismatch with "(.*)"$/, - (errorCode: string) => { - expect(details.reason).toEqual(StandardResolutionReasons.ERROR); - expect(details.errorCode).toEqual(errorCode); - }, - ); - }); - }); -} diff --git a/libs/providers/flagd-web/src/e2e/step-definitions/flag.ts b/libs/providers/flagd-web/src/e2e/step-definitions/flag.ts new file mode 100644 index 000000000..ec40cbf76 --- /dev/null +++ b/libs/providers/flagd-web/src/e2e/step-definitions/flag.ts @@ -0,0 +1,350 @@ +import { StepDefinitions } from 'jest-cucumber'; +import { + EvaluationContext, + EvaluationDetails, + FlagValue, + JsonObject, + OpenFeature, + ProviderEvents, + StandardResolutionReasons, +} from '@openfeature/web-sdk'; +import { E2E_CLIENT_NAME } from '@openfeature/flagd-core'; + +export const flagStepDefinitions: StepDefinitions = ({ given, and, when, then }) => { + let flagKey: string; + let value: FlagValue; + let context: EvaluationContext = {}; + let details: EvaluationDetails; + let fallback: FlagValue; + let flagsChanged: string[]; + + const client = OpenFeature.getClient(E2E_CLIENT_NAME); + + beforeAll((done) => { + client.addHandler(ProviderEvents.Ready, () => { + done(); + }); + }); + + beforeEach(() => { + context = {}; + }); + + given('a provider is registered', () => undefined); + given('a flagd provider is set', () => undefined); + + when( + /^a boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + value = await client.getBooleanValue(key, defaultValue === 'true'); + }, + ); + + then(/^the resolved boolean value should be "(.*)"$/, (expectedValue: string) => { + expect(value).toEqual(expectedValue === 'true'); + }); + + when( + /^a string flag with key "(.*)" is evaluated with default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + value = await client.getStringValue(key, defaultValue); + }, + ); + + then(/^the resolved string value should be "(.*)"$/, (expectedValue: string) => { + expect(value).toEqual(expectedValue); + }); + + when( + /^an integer flag with key "(.*)" is evaluated with default value (\d+)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = Number(defaultValue); + value = await client.getNumberValue(key, Number.parseInt(defaultValue)); + }, + ); + + then(/^the resolved integer value should be (\d+)$/, (expectedValue: string) => { + expect(value).toEqual(Number.parseInt(expectedValue)); + }); + + when( + /^a float flag with key "(.*)" is evaluated with default value (\d+\.?\d*)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = Number(defaultValue); + value = await client.getNumberValue(key, Number.parseFloat(defaultValue)); + }, + ); + + then(/^the resolved float value should be (\d+\.?\d*)$/, (expectedValue: string) => { + expect(value).toEqual(Number.parseFloat(expectedValue)); + }); + + when(/^an object flag with key "(.*)" is evaluated with a null default value$/, async (key: string) => { + const defaultValue = {}; + flagKey = key; + fallback = ''; + value = await client.getObjectValue(key, defaultValue); + }); + + then( + /^the resolved object value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, + (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { + const jsonObject = value as JsonObject; + expect(jsonObject[field1]).toEqual(boolValue === 'true'); + expect(jsonObject[field2]).toEqual(stringValue); + expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); + }, + ); + + when( + /^a boolean flag with key "(.*)" is evaluated with details and default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getBooleanDetails(key, defaultValue === 'true'); + }, + ); + + then( + /^the resolved boolean details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(expectedValue === 'true'); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^a string flag with key "(.*)" is evaluated with details and default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getStringDetails(key, defaultValue); + }, + ); + + then( + /^the resolved string details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(expectedValue); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^an integer flag with key "(.*)" is evaluated with details and default value (\d+)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getNumberDetails(key, Number.parseInt(defaultValue)); + }, + ); + + then( + /^the resolved integer details value should be (\d+), the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(Number.parseInt(expectedValue)); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^a float flag with key "(.*)" is evaluated with details and default value (\d+\.?\d*)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getNumberDetails(key, Number.parseFloat(defaultValue)); + }, + ); + + then( + /^the resolved float details value should be (\d+\.?\d*), the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(Number.parseFloat(expectedValue)); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when(/^an object flag with key "(.*)" is evaluated with details and a null default value$/, async (key: string) => { + flagKey = key; + fallback = {}; + details = await client.getObjectDetails(key, {}); + }); + + then( + /^the resolved object details value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, + (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { + expect(details).toBeDefined(); + const jsonObject = details.value as JsonObject; + + expect(jsonObject[field1]).toEqual(boolValue === 'true'); + expect(jsonObject[field2]).toEqual(stringValue); + expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); + }, + ); + + and( + /^the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + then(/^the resolved string response should be "(.*)"$/, (expectedValue: string) => { + expect(value).toEqual(expectedValue); + }); + + when( + /^a non-existent string flag with key "(.*)" is evaluated with details and a default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getStringDetails(flagKey, defaultValue); + }, + ); + + then(/^the default string value should be returned$/, () => { + expect(details).toBeDefined(); + expect(details.value).toEqual(fallback); + }); + + and( + /^the reason should indicate an error and the error code should indicate a missing flag with "(.*)"$/, + (errorCode: string) => { + expect(details).toBeDefined(); + expect(details.reason).toEqual(StandardResolutionReasons.ERROR); + expect(details.errorCode).toEqual(errorCode); + }, + ); + + when( + /^a string flag with key "(.*)" is evaluated as an integer, with details and a default value (\d+)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = Number.parseInt(defaultValue); + details = await client.getNumberDetails(flagKey, Number.parseInt(defaultValue)); + }, + ); + + then(/^the default integer value should be returned$/, () => { + expect(details).toBeDefined(); + expect(details.value).toEqual(fallback); + }); + + and( + /^the reason should indicate an error and the error code should indicate a type mismatch with "(.*)"$/, + (errorCode: string) => { + expect(details).toBeDefined(); + expect(details.reason).toEqual(StandardResolutionReasons.ERROR); + expect(details.errorCode).toEqual(errorCode); + }, + ); + + let ran: boolean; + when('a PROVIDER_READY handler is added', () => { + client.addHandler(ProviderEvents.Ready, async () => { + ran = true; + }); + }); + then('the PROVIDER_READY handler must run', () => { + expect(ran).toBeTruthy(); + }); + + when('a PROVIDER_CONFIGURATION_CHANGED handler is added', () => { + client.addHandler(ProviderEvents.ConfigurationChanged, async (details) => { + // file writes are not atomic, so we get a few events in quick succession from the testbed + // some will not contain changes, this tolerates that; at least 1 should have our change + + // TODO: enable this for testing of issue + //if (details?.flagsChanged?.length) { + // flagsChanged = details?.flagsChanged; + // ran = true; + //} + + // TODO: remove this for testing of issue + ran = true; + }); + }); + + and(/^a flag with key "(.*)" is modified$/, async () => { + // this happens every 1s in the associated container, so wait 3s + await new Promise((resolve) => setTimeout(resolve, 3000)); + }); + + then('the PROVIDER_CONFIGURATION_CHANGED handler must run', async () => { + expect(ran).toBeTruthy(); + }); + + and(/^the event details must indicate "(.*)" was altered$/, (flagName) => { + // TODO: enable this for testing of issue + //expect(flagsChanged).toContain(flagName); + }); + + when( + /^a zero-value boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, + (key, defaultVal: string) => { + flagKey = key; + fallback = defaultVal === 'true'; + }, + ); + + then(/^the resolved boolean zero-value should be "(.*)"$/, async (expectedVal: string) => { + const expectedValue = expectedVal === 'true'; + const value = await client.getBooleanValue(flagKey, fallback as boolean); + expect(value).toEqual(expectedValue); + }); + + when(/^a zero-value string flag with key "(.*)" is evaluated with default value "(.*)"$/, (key, defaultVal) => { + flagKey = key; + fallback = defaultVal; + }); + + then('the resolved string zero-value should be ""', async () => { + const value = await client.getStringValue(flagKey, fallback as string); + expect(value).toEqual(''); + }); + + when(/^a zero-value integer flag with key "(.*)" is evaluated with default value (\d+)$/, (key, defaultVal) => { + flagKey = key; + fallback = defaultVal; + }); + + then(/^the resolved integer zero-value should be (\d+)$/, async (expectedValueString) => { + const expectedValue = Number.parseInt(expectedValueString); + const value = await client.getNumberValue(flagKey, fallback as number); + expect(value).toEqual(expectedValue); + }); + + when( + /^a zero-value float flag with key "(.*)" is evaluated with default value (\d+\.\d+)$/, + (key, defaultValueString) => { + flagKey = key; + fallback = Number.parseFloat(defaultValueString); + }, + ); + + then(/^the resolved float zero-value should be (\d+\.\d+)$/, async (expectedValueString) => { + const expectedValue = Number.parseFloat(expectedValueString); + const value = await client.getNumberValue(flagKey, fallback as number); + expect(value).toEqual(expectedValue); + }); + + then(/^the returned reason should be "(.*)"$/, (expectedReason) => { + expect(details.reason).toEqual(expectedReason); + }); +}; diff --git a/libs/providers/flagd-web/src/e2e/step-definitions/index.ts b/libs/providers/flagd-web/src/e2e/step-definitions/index.ts new file mode 100644 index 000000000..7a2dcd6c9 --- /dev/null +++ b/libs/providers/flagd-web/src/e2e/step-definitions/index.ts @@ -0,0 +1 @@ +export * from './flag'; diff --git a/libs/providers/flagd-web/src/e2e/tests/provider.spec.ts b/libs/providers/flagd-web/src/e2e/tests/provider.spec.ts index 927a5474b..1c786f1a2 100644 --- a/libs/providers/flagd-web/src/e2e/tests/provider.spec.ts +++ b/libs/providers/flagd-web/src/e2e/tests/provider.spec.ts @@ -1,9 +1,11 @@ import assert from 'assert'; import { OpenFeature } from '@openfeature/web-sdk'; -import { FLAGD_NAME, IMAGE_VERSION } from '../constants'; import { GenericContainer, StartedTestContainer } from 'testcontainers'; import { FlagdWebProvider } from '../../lib/flagd-web-provider'; -import { evaluation } from '../step-definitions/evaluation'; +import { autoBindSteps, loadFeature } from 'jest-cucumber'; +import { FLAGD_NAME, GHERKIN_EVALUATION_FEATURE } from '../constants'; +import { flagStepDefinitions } from '../step-definitions'; +import { E2E_CLIENT_NAME, IMAGE_VERSION } from '@openfeature/flagd-core'; // register the flagd provider before the tests. async function setup() { @@ -15,17 +17,18 @@ async function setup() { .withExposedPorts(8013) .start(); containers.push(stable); - OpenFeature.setProvider( - new FlagdWebProvider({ - host: stable.getHost(), - port: stable.getMappedPort(8013), - tls: false, - maxRetries: -1, - }), - ); + const flagdWebProvider = new FlagdWebProvider({ + host: stable.getHost(), + port: stable.getMappedPort(8013), + tls: false, + maxRetries: -1, + }); + await OpenFeature.setProviderAndWait(E2E_CLIENT_NAME, flagdWebProvider); assert( - OpenFeature.providerMetadata.name === FLAGD_NAME, - new Error(`Expected ${FLAGD_NAME} provider to be configured, instead got: ${OpenFeature.providerMetadata.name}`), + OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name === FLAGD_NAME, + new Error( + `Expected ${E2E_CLIENT_NAME} provider to be configured, instead got: ${OpenFeature.providerMetadata.name}`, + ), ); console.log('flagd provider configured!'); return containers; @@ -42,5 +45,7 @@ describe('web provider', () => { container.stop(); } }); - evaluation(); + + const features = [loadFeature(GHERKIN_EVALUATION_FEATURE)]; + autoBindSteps(features, [flagStepDefinitions]); }); diff --git a/libs/providers/flagd-web/src/e2e/tsconfig.lib.json b/libs/providers/flagd-web/src/e2e/tsconfig.lib.json index 244a71e92..1211629d6 100644 --- a/libs/providers/flagd-web/src/e2e/tsconfig.lib.json +++ b/libs/providers/flagd-web/src/e2e/tsconfig.lib.json @@ -1,11 +1,12 @@ { "extends": "../../tsconfig.json", "compilerOptions": { - "lib": ["ES2015", "DOM"], + "lib": ["ES2022", "DOM"], "outDir": "../../../dist/out-tsc", "declaration": true, "types": ["jest"], "allowSyntheticDefaultImports": true, - "allowJs" :true + "allowJs" :true, + "resolveJsonModule": true } } diff --git a/libs/providers/flagd-web/tsconfig.lib.json b/libs/providers/flagd-web/tsconfig.lib.json index 677013480..f961a4643 100644 --- a/libs/providers/flagd-web/tsconfig.lib.json +++ b/libs/providers/flagd-web/tsconfig.lib.json @@ -8,6 +8,6 @@ "allowSyntheticDefaultImports": true, }, "include": ["**/*.ts"], - "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts"] + "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts", "src/e2e/"] } diff --git a/libs/providers/flagd-web/tsconfig.spec.json b/libs/providers/flagd-web/tsconfig.spec.json index 11470f5d3..5a458f8e3 100644 --- a/libs/providers/flagd-web/tsconfig.spec.json +++ b/libs/providers/flagd-web/tsconfig.spec.json @@ -6,5 +6,5 @@ "types": ["jest"], "allowJs": true, }, - "include": ["jest.config.ts", "**/*.test.ts", "**/*.spec.ts", "**/*.d.ts"] + "include": ["jest.config.ts", "**/*.test.ts", "**/*.spec.ts", "**/*.d.ts","src/e2e/"] } diff --git a/libs/providers/flagd/flagd-testbed b/libs/providers/flagd/flagd-testbed deleted file mode 160000 index ed7e0ba66..000000000 --- a/libs/providers/flagd/flagd-testbed +++ /dev/null @@ -1 +0,0 @@ -Subproject commit ed7e0ba660b01e1a22849e1b28ec37453921552e diff --git a/libs/providers/flagd/project.json b/libs/providers/flagd/project.json index 556647124..4cc8ae90f 100644 --- a/libs/providers/flagd/project.json +++ b/libs/providers/flagd/project.json @@ -58,25 +58,11 @@ } ] }, - "pullTestHarness": { - "executor": "nx:run-commands", - "options": { - "commands": [ - "git submodule update --init spec", - "git submodule update --init flagd-testbed", - "rm -f -r ./src/e2e/features/*", - "cp -v ./spec/specification/assets/gherkin/evaluation.feature ./src/e2e/features/", - "cp -v ./flagd-testbed/gherkin/*.feature ./src/e2e/features/" - ], - "cwd": "libs/providers/flagd", - "parallel": false - } - }, "e2e": { "executor": "nx:run-commands", "options": { "commands": [ - "npx jest --runInBand --detectOpenHandles" + "npx jest" ], "cwd": "libs/providers/flagd/src/e2e", "parallel": false @@ -86,7 +72,7 @@ "target": "generate" }, { - "target": "pullTestHarness" + "target": "flagd-core:pullTestHarness" } ] }, diff --git a/libs/providers/flagd/schemas b/libs/providers/flagd/schemas index e72b08b71..2aa89b314 160000 --- a/libs/providers/flagd/schemas +++ b/libs/providers/flagd/schemas @@ -1 +1 @@ -Subproject commit e72b08b71ad8654e8a31ec6f75a9c8b4d47db8ca +Subproject commit 2aa89b31432284507af3873de9b0bb7b68dd02c7 diff --git a/libs/providers/flagd/spec b/libs/providers/flagd/spec index eaa44da21..ffebdecd7 160000 --- a/libs/providers/flagd/spec +++ b/libs/providers/flagd/spec @@ -1 +1 @@ -Subproject commit eaa44da2110d97376d3292ea3963c54bea498a77 +Subproject commit ffebdecd725ed1a19c96927f66472c86e97ce551 diff --git a/libs/providers/flagd/src/e2e/constants.ts b/libs/providers/flagd/src/e2e/constants.ts index eb83fd17f..f0633ac8e 100644 --- a/libs/providers/flagd/src/e2e/constants.ts +++ b/libs/providers/flagd/src/e2e/constants.ts @@ -1,6 +1,13 @@ +import { getGherkinTestPath } from '@openfeature/flagd-core'; + export const FLAGD_NAME = 'flagd Provider'; -export const E2E_CLIENT_NAME = 'e2e'; export const UNSTABLE_CLIENT_NAME = 'unstable'; export const UNAVAILABLE_CLIENT_NAME = 'unavailable'; -export const IMAGE_VERSION = 'v0.5.6'; +export const GHERKIN_FLAGD_FEATURE = getGherkinTestPath('flagd.feature'); +export const GHERKIN_FLAGD_JSON_EVALUATOR_FEATURE = getGherkinTestPath('flagd-json-evaluator.feature'); +export const GHERKIN_FLAGD_RECONNECT_FEATURE = getGherkinTestPath('flagd-reconnect.feature'); +export const GHERKIN_EVALUATION_FEATURE = getGherkinTestPath( + 'evaluation.feature', + 'spec/specification/assets/gherkin/', +); diff --git a/libs/providers/flagd/src/e2e/features/.gitignore b/libs/providers/flagd/src/e2e/features/.gitignore deleted file mode 100644 index f9e3307dd..000000000 --- a/libs/providers/flagd/src/e2e/features/.gitignore +++ /dev/null @@ -1 +0,0 @@ -*.feature diff --git a/libs/providers/flagd/src/e2e/features/.gitkeep b/libs/providers/flagd/src/e2e/features/.gitkeep deleted file mode 100644 index e69de29bb..000000000 diff --git a/libs/providers/flagd/src/e2e/index.ts b/libs/providers/flagd/src/e2e/index.ts new file mode 100644 index 000000000..ac4cdfd4c --- /dev/null +++ b/libs/providers/flagd/src/e2e/index.ts @@ -0,0 +1,2 @@ +export * from './constants'; +export * from './step-definitions'; diff --git a/libs/providers/flagd/src/e2e/step-definitions/evaluation.ts b/libs/providers/flagd/src/e2e/step-definitions/evaluation.ts deleted file mode 100644 index 7d392630e..000000000 --- a/libs/providers/flagd/src/e2e/step-definitions/evaluation.ts +++ /dev/null @@ -1,364 +0,0 @@ -import { - EvaluationContext, - EvaluationDetails, - JsonObject, - JsonValue, - OpenFeature, - ProviderEvents, - ResolutionDetails, - StandardResolutionReasons, -} from '@openfeature/server-sdk'; -import { defineFeature, loadFeature } from 'jest-cucumber'; -import { E2E_CLIENT_NAME } from '../constants'; - -export function evaluation() { - // load the feature file. - const feature = loadFeature('features/evaluation.feature'); - - // get a client (flagd provider registered in setup) - const client = OpenFeature.getClient(E2E_CLIENT_NAME); - - const givenAnOpenfeatureClientIsRegistered = ( - given: (stepMatcher: string, stepDefinitionCallback: () => void) => void, - ) => { - given('a provider is registered', () => undefined); - }; - - defineFeature(feature, (test) => { - beforeAll((done) => { - client.addHandler(ProviderEvents.Ready, async () => { - done(); - }); - }); - - test('Resolves boolean value', ({ given, when, then }) => { - let value: boolean; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, - async (key: string, defaultValue: string) => { - flagKey = key; - value = await client.getBooleanValue(flagKey, defaultValue === 'true'); - }, - ); - - then(/^the resolved boolean value should be "(.*)"$/, (expectedValue: string) => { - expect(value).toEqual(expectedValue === 'true'); - }); - }); - - test('Resolves string value', ({ given, when, then }) => { - let value: string; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a string flag with key "(.*)" is evaluated with default value "(.*)"$/, - async (key: string, defaultValue: string) => { - flagKey = key; - value = await client.getStringValue(flagKey, defaultValue); - }, - ); - - then(/^the resolved string value should be "(.*)"$/, (expectedValue: string) => { - expect(value).toEqual(expectedValue); - }); - }); - - test('Resolves integer value', ({ given, when, then }) => { - let value: number; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^an integer flag with key "(.*)" is evaluated with default value (\d+)$/, - async (key: string, defaultValue: string) => { - flagKey = key; - value = await client.getNumberValue(flagKey, Number.parseInt(defaultValue)); - }, - ); - - then(/^the resolved integer value should be (\d+)$/, (expectedValue: string) => { - expect(value).toEqual(Number.parseInt(expectedValue)); - }); - }); - - test('Resolves float value', ({ given, when, then }) => { - let value: number; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a float flag with key "(.*)" is evaluated with default value (\d+\.?\d*)$/, - async (key: string, defaultValue: string) => { - flagKey = key; - value = await client.getNumberValue(flagKey, Number.parseFloat(defaultValue)); - }, - ); - - then(/^the resolved float value should be (\d+\.?\d*)$/, (expectedValue: string) => { - expect(value).toEqual(Number.parseFloat(expectedValue)); - }); - }); - - test('Resolves object value', ({ given, when, then }) => { - let value: JsonValue; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when(/^an object flag with key "(.*)" is evaluated with a null default value$/, async (key: string) => { - flagKey = key; - value = await client.getObjectValue(flagKey, {}); - }); - - then( - /^the resolved object value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, - (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { - const jsonObject = value as JsonObject; - expect(jsonObject[field1]).toEqual(boolValue === 'true'); - expect(jsonObject[field2]).toEqual(stringValue); - expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); - }, - ); - }); - - test('Resolves boolean details', ({ given, when, then }) => { - let details: EvaluationDetails; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a boolean flag with key "(.*)" is evaluated with details and default value "(.*)"$/, - async (key: string, defaultValue: string) => { - flagKey = key; - details = await client.getBooleanDetails(flagKey, defaultValue === 'true'); - }, - ); - - then( - /^the resolved boolean details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(expectedValue === 'true'); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves string details', ({ given, when, then }) => { - let details: EvaluationDetails; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a string flag with key "(.*)" is evaluated with details and default value "(.*)"$/, - async (key: string, defaultValue: string) => { - flagKey = key; - details = await client.getStringDetails(flagKey, defaultValue); - }, - ); - - then( - /^the resolved string details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(expectedValue); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves integer details', ({ given, when, then }) => { - let details: EvaluationDetails; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^an integer flag with key "(.*)" is evaluated with details and default value (\d+)$/, - async (key: string, defaultValue: string) => { - flagKey = key; - details = await client.getNumberDetails(flagKey, Number.parseInt(defaultValue)); - }, - ); - - then( - /^the resolved integer details value should be (\d+), the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(Number.parseInt(expectedValue)); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves float details', ({ given, when, then }) => { - let details: EvaluationDetails; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a float flag with key "(.*)" is evaluated with details and default value (\d+\.?\d*)$/, - async (key: string, defaultValue: string) => { - flagKey = key; - details = await client.getNumberDetails(flagKey, Number.parseFloat(defaultValue)); - }, - ); - - then( - /^the resolved float details value should be (\d+\.?\d*), the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedValue: string, expectedVariant: string, expectedReason: string) => { - expect(details.value).toEqual(Number.parseFloat(expectedValue)); - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves object details', ({ given, when, then, and }) => { - let details: EvaluationDetails; // update this after merge - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^an object flag with key "(.*)" is evaluated with details and a null default value$/, - async (key: string) => { - flagKey = key; - details = await client.getObjectDetails(flagKey, {}); // update this after merge - }, - ); - - then( - /^the resolved object details value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, - (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { - const jsonObject = details.value as JsonObject; - - expect(jsonObject[field1]).toEqual(boolValue === 'true'); - expect(jsonObject[field2]).toEqual(stringValue); - expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); - }, - ); - - and( - /^the variant should be "(.*)", and the reason should be "(.*)"$/, - (expectedVariant: string, expectedReason: string) => { - expect(details.variant).toEqual(expectedVariant); - expect(details.reason).toEqual(expectedReason); - }, - ); - }); - - test('Resolves based on context', ({ given, when, and, then }) => { - const context: EvaluationContext = {}; - let value: string; - let flagKey: string; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^context contains keys "(.*)", "(.*)", "(.*)", "(.*)" with values "(.*)", "(.*)", (\d+), "(.*)"$/, - ( - stringField1: string, - stringField2: string, - intField: string, - boolField: string, - stringValue1: string, - stringValue2: string, - intValue: string, - boolValue: string, - ) => { - context[stringField1] = stringValue1; - context[stringField2] = stringValue2; - context[intField] = Number.parseInt(intValue); - context[boolField] = boolValue === 'true'; - }, - ); - - and( - /^a flag with key "(.*)" is evaluated with default value "(.*)"$/, - async (key: string, defaultValue: string) => { - flagKey = key; - value = await client.getStringValue(flagKey, defaultValue, context); - }, - ); - - then(/^the resolved string response should be "(.*)"$/, (expectedValue: string) => { - expect(value).toEqual(expectedValue); - }); - - and(/^the resolved flag value is "(.*)" when the context is empty$/, async (expectedValue) => { - const emptyContextValue = await client.getStringValue(flagKey, 'nope', {}); - expect(emptyContextValue).toEqual(expectedValue); - }); - }); - - test('Flag not found', ({ given, when, then, and }) => { - let flagKey: string; - let fallbackValue: string; - let details: ResolutionDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a non-existent string flag with key "(.*)" is evaluated with details and a default value "(.*)"$/, - async (key: string, defaultValue: string) => { - flagKey = key; - fallbackValue = defaultValue; - details = await client.getStringDetails(flagKey, defaultValue); - }, - ); - - then(/^the default string value should be returned$/, () => { - expect(details.value).toEqual(fallbackValue); - }); - - and( - /^the reason should indicate an error and the error code should indicate a missing flag with "(.*)"$/, - (errorCode: string) => { - expect(details.reason).toEqual(StandardResolutionReasons.ERROR); - expect(details.errorCode).toEqual(errorCode); - }, - ); - }); - - test('Type error', ({ given, when, then, and }) => { - let flagKey: string; - let fallbackValue: number; - let details: ResolutionDetails; - - givenAnOpenfeatureClientIsRegistered(given); - - when( - /^a string flag with key "(.*)" is evaluated as an integer, with details and a default value (\d+)$/, - async (key: string, defaultValue: string) => { - flagKey = key; - fallbackValue = Number.parseInt(defaultValue); - details = await client.getNumberDetails(flagKey, Number.parseInt(defaultValue)); - }, - ); - - then(/^the default integer value should be returned$/, () => { - expect(details.value).toEqual(fallbackValue); - }); - - and( - /^the reason should indicate an error and the error code should indicate a type mismatch with "(.*)"$/, - (errorCode: string) => { - expect(details.reason).toEqual(StandardResolutionReasons.ERROR); - expect(details.errorCode).toEqual(errorCode); - }, - ); - }); - }); -} diff --git a/libs/providers/flagd/src/e2e/step-definitions/flag.ts b/libs/providers/flagd/src/e2e/step-definitions/flag.ts new file mode 100644 index 000000000..92e85da9b --- /dev/null +++ b/libs/providers/flagd/src/e2e/step-definitions/flag.ts @@ -0,0 +1,406 @@ +import { StepDefinitions } from 'jest-cucumber'; +import { + EvaluationContext, + EvaluationDetails, + FlagValue, + JsonObject, + OpenFeature, + ProviderEvents, + StandardResolutionReasons, +} from '@openfeature/server-sdk'; +import { E2E_CLIENT_NAME } from '@openfeature/flagd-core'; + +export const flagStepDefinitions: StepDefinitions = ({ given, and, when, then }) => { + let flagKey: string; + let value: FlagValue; + let context: EvaluationContext = {}; + let details: EvaluationDetails; + let fallback: FlagValue; + let flagsChanged: string[]; + + const client = OpenFeature.getClient(E2E_CLIENT_NAME); + + beforeAll((done) => { + client.addHandler(ProviderEvents.Ready, () => { + done(); + }); + }); + + beforeEach(() => { + context = {}; + }); + + given('a provider is registered', () => undefined); + given('a flagd provider is set', () => undefined); + + when( + /^a boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + value = await client.getBooleanValue(key, defaultValue === 'true'); + }, + ); + + then(/^the resolved boolean value should be "(.*)"$/, (expectedValue: string) => { + expect(value).toEqual(expectedValue === 'true'); + }); + + when( + /^a string flag with key "(.*)" is evaluated with default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + value = await client.getStringValue(key, defaultValue); + }, + ); + + then(/^the resolved string value should be "(.*)"$/, (expectedValue: string) => { + expect(value).toEqual(expectedValue); + }); + + when( + /^an integer flag with key "(.*)" is evaluated with default value (\d+)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = Number(defaultValue); + value = await client.getNumberValue(key, Number.parseInt(defaultValue)); + }, + ); + + then(/^the resolved integer value should be (\d+)$/, (expectedValue: string) => { + expect(value).toEqual(Number.parseInt(expectedValue)); + }); + + when( + /^a float flag with key "(.*)" is evaluated with default value (\d+\.?\d*)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = Number(defaultValue); + value = await client.getNumberValue(key, Number.parseFloat(defaultValue)); + }, + ); + + then(/^the resolved float value should be (\d+\.?\d*)$/, (expectedValue: string) => { + expect(value).toEqual(Number.parseFloat(expectedValue)); + }); + + when(/^an object flag with key "(.*)" is evaluated with a null default value$/, async (key: string) => { + const defaultValue = {}; + flagKey = key; + fallback = ''; + value = await client.getObjectValue(key, defaultValue); + }); + + then( + /^the resolved object value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, + (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { + const jsonObject = value as JsonObject; + expect(jsonObject[field1]).toEqual(boolValue === 'true'); + expect(jsonObject[field2]).toEqual(stringValue); + expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); + }, + ); + + when( + /^a boolean flag with key "(.*)" is evaluated with details and default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getBooleanDetails(key, defaultValue === 'true'); + }, + ); + + then( + /^the resolved boolean details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(expectedValue === 'true'); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^a string flag with key "(.*)" is evaluated with details and default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getStringDetails(key, defaultValue); + }, + ); + + then( + /^the resolved string details value should be "(.*)", the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(expectedValue); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^an integer flag with key "(.*)" is evaluated with details and default value (\d+)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getNumberDetails(key, Number.parseInt(defaultValue)); + }, + ); + + then( + /^the resolved integer details value should be (\d+), the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(Number.parseInt(expectedValue)); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^a float flag with key "(.*)" is evaluated with details and default value (\d+\.?\d*)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getNumberDetails(key, Number.parseFloat(defaultValue)); + }, + ); + + then( + /^the resolved float details value should be (\d+\.?\d*), the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedValue: string, expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.value).toEqual(Number.parseFloat(expectedValue)); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when(/^an object flag with key "(.*)" is evaluated with details and a null default value$/, async (key: string) => { + flagKey = key; + fallback = {}; + details = await client.getObjectDetails(key, {}); + }); + + then( + /^the resolved object details value should be contain fields "(.*)", "(.*)", and "(.*)", with values "(.*)", "(.*)" and (\d+), respectively$/, + (field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => { + expect(details).toBeDefined(); + const jsonObject = details.value as JsonObject; + + expect(jsonObject[field1]).toEqual(boolValue === 'true'); + expect(jsonObject[field2]).toEqual(stringValue); + expect(jsonObject[field3]).toEqual(Number.parseInt(intValue)); + }, + ); + + and( + /^the variant should be "(.*)", and the reason should be "(.*)"$/, + (expectedVariant: string, expectedReason: string) => { + expect(details).toBeDefined(); + expect(details.variant).toEqual(expectedVariant); + expect(details.reason).toEqual(expectedReason); + }, + ); + + when( + /^context contains keys "(.*)", "(.*)", "(.*)", "(.*)" with values "(.*)", "(.*)", (\d+), "(.*)"$/, + ( + stringField1: string, + stringField2: string, + intField: string, + boolField: string, + stringValue1: string, + stringValue2: string, + intValue: string, + boolValue: string, + ) => { + context[stringField1] = stringValue1; + context[stringField2] = stringValue2; + context[intField] = Number.parseInt(intValue); + context[boolField] = boolValue === 'true'; + }, + ); + + and(/^a flag with key "(.*)" is evaluated with default value "(.*)"$/, async (key: string, defaultValue: string) => { + flagKey = key; + value = await client.getStringValue(flagKey, defaultValue, context); + }); + + then(/^the resolved string response should be "(.*)"$/, (expectedValue: string) => { + expect(value).toEqual(expectedValue); + }); + + and(/^the resolved flag value is "(.*)" when the context is empty$/, async (expectedValue) => { + const emptyContextValue = await client.getStringValue(flagKey, 'nope', {}); + expect(emptyContextValue).toEqual(expectedValue); + }); + + when( + /^a non-existent string flag with key "(.*)" is evaluated with details and a default value "(.*)"$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = defaultValue; + details = await client.getStringDetails(flagKey, defaultValue); + }, + ); + + then(/^the default string value should be returned$/, () => { + expect(details).toBeDefined(); + expect(details.value).toEqual(fallback); + }); + + and( + /^the reason should indicate an error and the error code should indicate a missing flag with "(.*)"$/, + (errorCode: string) => { + expect(details).toBeDefined(); + expect(details.reason).toEqual(StandardResolutionReasons.ERROR); + expect(details.errorCode).toEqual(errorCode); + }, + ); + + when( + /^a string flag with key "(.*)" is evaluated as an integer, with details and a default value (\d+)$/, + async (key: string, defaultValue: string) => { + flagKey = key; + fallback = Number.parseInt(defaultValue); + details = await client.getNumberDetails(flagKey, Number.parseInt(defaultValue)); + }, + ); + + then(/^the default integer value should be returned$/, () => { + expect(details).toBeDefined(); + expect(details.value).toEqual(fallback); + }); + + and( + /^the reason should indicate an error and the error code should indicate a type mismatch with "(.*)"$/, + (errorCode: string) => { + expect(details).toBeDefined(); + expect(details.reason).toEqual(StandardResolutionReasons.ERROR); + expect(details.errorCode).toEqual(errorCode); + }, + ); + + let ran: boolean; + when('a PROVIDER_READY handler is added', () => { + client.addHandler(ProviderEvents.Ready, async () => { + ran = true; + }); + }); + then('the PROVIDER_READY handler must run', () => { + expect(ran).toBeTruthy(); + }); + + when('a PROVIDER_CONFIGURATION_CHANGED handler is added', () => { + client.addHandler(ProviderEvents.ConfigurationChanged, async (details) => { + // file writes are not atomic, so we get a few events in quick succession from the testbed + // some will not contain changes, this tolerates that; at least 1 should have our change + if (details?.flagsChanged?.length) { + flagsChanged = details?.flagsChanged; + + ran = true; + } + }); + }); + + and(/^a flag with key "(.*)" is modified$/, async () => { + // this happens every 1s in the associated container, so wait 3s + await new Promise((resolve) => setTimeout(resolve, 3000)); + }); + + then('the PROVIDER_CONFIGURATION_CHANGED handler must run', () => { + expect(ran).toBeTruthy(); + }); + + and(/^the event details must indicate "(.*)" was altered$/, (flagName) => { + expect(flagsChanged).toContain(flagName); + }); + + when( + /^a zero-value boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, + (key, defaultVal: string) => { + flagKey = key; + fallback = defaultVal === 'true'; + }, + ); + + then(/^the resolved boolean zero-value should be "(.*)"$/, async (expectedVal: string) => { + const expectedValue = expectedVal === 'true'; + const value = await client.getBooleanValue(flagKey, fallback as boolean); + expect(value).toEqual(expectedValue); + }); + + when(/^a zero-value string flag with key "(.*)" is evaluated with default value "(.*)"$/, (key, defaultVal) => { + flagKey = key; + fallback = defaultVal; + }); + + then('the resolved string zero-value should be ""', async () => { + const value = await client.getStringValue(flagKey, fallback as string); + expect(value).toEqual(''); + }); + + when(/^a zero-value integer flag with key "(.*)" is evaluated with default value (\d+)$/, (key, defaultVal) => { + flagKey = key; + fallback = defaultVal; + }); + + then(/^the resolved integer zero-value should be (\d+)$/, async (expectedValueString) => { + const expectedValue = Number.parseInt(expectedValueString); + const value = await client.getNumberValue(flagKey, fallback as number); + expect(value).toEqual(expectedValue); + }); + + when( + /^a zero-value float flag with key "(.*)" is evaluated with default value (\d+\.\d+)$/, + (key, defaultValueString) => { + flagKey = key; + fallback = Number.parseFloat(defaultValueString); + }, + ); + + then(/^the resolved float zero-value should be (\d+\.\d+)$/, async (expectedValueString) => { + const expectedValue = Number.parseFloat(expectedValueString); + const value = await client.getNumberValue(flagKey, fallback as number); + expect(value).toEqual(expectedValue); + }); + + // evaluator + and(/^a context containing a key "(.*)", with value "?([^"]*)"?$/, (key: string, value: string) => { + context[key] = value; + }); + + then(/^the returned value should be (.*)$/, async (expectedValue) => { + if (!isNaN(Number(expectedValue))) { + const value = await client.getNumberValue(flagKey, fallback as number, context); + expect(value).toEqual(Number(expectedValue)); + } else if (typeof expectedValue == 'string') { + const value = await client.getStringValue(flagKey, fallback as string, context); + expect(value).toEqual(expectedValue.replaceAll('"', '')); + } + }); + + and( + /^a context containing a nested property with outer key "(.*)" and inner key "(.*)", with value (.*)$/, + (outerKey: string, innerKey: string, value: string) => { + // we have to support string and non-string params in this test (we test invalid context value 3) + const valueNoQuotes = value.replaceAll('"', ''); + context[outerKey] = { + [innerKey]: parseInt(value) || valueNoQuotes, + }; + }, + ); + + and(/^a context containing a targeting key with value "(.*)"$/, async (targetingKeyValue) => { + context.targetingKey = targetingKeyValue; + details = await client.getStringDetails(flagKey, fallback as string, context); + value = details.value; + }); + + then(/^the returned reason should be "(.*)"$/, (expectedReason) => { + expect(details.reason).toEqual(expectedReason); + }); +}; diff --git a/libs/providers/flagd/src/e2e/step-definitions/flagd-json-evaluator.ts b/libs/providers/flagd/src/e2e/step-definitions/flagd-json-evaluator.ts index 8751b05d3..d7da17daa 100644 --- a/libs/providers/flagd/src/e2e/step-definitions/flagd-json-evaluator.ts +++ b/libs/providers/flagd/src/e2e/step-definitions/flagd-json-evaluator.ts @@ -1,7 +1,7 @@ import { EvaluationContext, EvaluationDetails, OpenFeature, ProviderEvents } from '@openfeature/server-sdk'; import { defineFeature, loadFeature } from 'jest-cucumber'; import { StepsDefinitionCallbackFunction } from 'jest-cucumber/dist/src/feature-definition-creation'; -import { E2E_CLIENT_NAME } from '../constants'; +import { E2E_CLIENT_NAME } from '@openfeature/flagd-core'; export function flagdJsonEvaluator() { // load the feature file. diff --git a/libs/providers/flagd/src/e2e/step-definitions/flagd-reconnect.unstable.ts b/libs/providers/flagd/src/e2e/step-definitions/flagd-reconnect.unstable.ts deleted file mode 100644 index 9462b7d8f..000000000 --- a/libs/providers/flagd/src/e2e/step-definitions/flagd-reconnect.unstable.ts +++ /dev/null @@ -1,76 +0,0 @@ -import { OpenFeature, ProviderEvents } from '@openfeature/server-sdk'; -import { defineFeature, loadFeature } from 'jest-cucumber'; -import { UNAVAILABLE_CLIENT_NAME, UNSTABLE_CLIENT_NAME } from '../constants'; - -jest.setTimeout(30000); - -export function flagdRecconnectUnstable() { - // load the feature file. - const feature = loadFeature('features/flagd-reconnect.feature'); - - // get a client (flagd provider registered in setup) - const client = OpenFeature.getClient(UNSTABLE_CLIENT_NAME); - - defineFeature(feature, (test) => { - let readyRunCount = 0; - let errorRunCount = 0; - - beforeAll((done) => { - client.addHandler(ProviderEvents.Ready, async () => { - readyRunCount++; - done(); - }); - }); - - describe('retry', () => { - /** - * This describe block and retry settings are calibrated to gRPC's retry time - * and our testing container's restart cadence. - */ - const retryTimes = 240; - const retryDelayMs = 1000; - jest.retryTimes(retryTimes); - - test('Provider reconnection', ({ given, when, then, and }) => { - given('a flagd provider is set', () => { - // handled in beforeAll - }); - when('a PROVIDER_READY handler and a PROVIDER_ERROR handler are added', () => { - client.addHandler(ProviderEvents.Error, () => { - errorRunCount++; - }); - }); - then('the PROVIDER_READY handler must run when the provider connects', async () => { - // should already be at 1 from `beforeAll` - expect(readyRunCount).toEqual(1); - }); - and("the PROVIDER_ERROR handler must run when the provider's connection is lost", async () => { - await new Promise((resolve) => setTimeout(resolve, retryDelayMs)); - expect(errorRunCount).toBeGreaterThan(0); - }); - and('when the connection is reestablished the PROVIDER_READY handler must run again', async () => { - await new Promise((resolve) => setTimeout(resolve, retryDelayMs)); - expect(readyRunCount).toBeGreaterThan(1); - }); - }); - }); - - test('Provider unavailable', ({ given, when, then }) => { - let errorHandlerRun = 0; - - given('flagd is unavailable', async () => { - // handled in setup - }); - - when('a flagd provider is set and initialization is awaited', () => { - OpenFeature.getClient(UNAVAILABLE_CLIENT_NAME).addHandler(ProviderEvents.Error, () => { - errorHandlerRun++; - }); - }); - - then('an error should be indicated within the configured deadline', () => { - expect(errorHandlerRun).toBeGreaterThan(0); - }); - }); - }); -} diff --git a/libs/providers/flagd/src/e2e/step-definitions/flagd.ts b/libs/providers/flagd/src/e2e/step-definitions/flagd.ts deleted file mode 100644 index 0a59c073b..000000000 --- a/libs/providers/flagd/src/e2e/step-definitions/flagd.ts +++ /dev/null @@ -1,133 +0,0 @@ -import { OpenFeature, ProviderEvents } from '@openfeature/server-sdk'; -import { defineFeature, loadFeature } from 'jest-cucumber'; -import { E2E_CLIENT_NAME } from '../constants'; - -export function flagd() { - // load the feature file. - const feature = loadFeature('features/flagd.feature'); - - // get a client (flagd provider registered in setup) - const client = OpenFeature.getClient(E2E_CLIENT_NAME); - - const aFlagProviderIsSet = (given: (stepMatcher: string, stepDefinitionCallback: () => void) => void) => { - given('a flagd provider is set', () => undefined); - }; - - defineFeature(feature, (test) => { - beforeAll((done) => { - client.addHandler(ProviderEvents.Ready, async () => { - done(); - }); - }); - - test('Provider ready event', ({ given, when, then }) => { - let ran = false; - - aFlagProviderIsSet(given); - when('a PROVIDER_READY handler is added', () => { - client.addHandler(ProviderEvents.Ready, async () => { - ran = true; - }); - }); - then('the PROVIDER_READY handler must run', () => { - expect(ran).toBeTruthy(); - }); - }); - - test('Flag change event', ({ given, when, and, then }) => { - let ran = false; - let flagsChanged: string[]; - - aFlagProviderIsSet(given); - when('a PROVIDER_CONFIGURATION_CHANGED handler is added', () => { - client.addHandler(ProviderEvents.ConfigurationChanged, async (details) => { - ran = true; - // file writes are not atomic, so we get a few events in quick succession from the testbed - // some will not contain changes, this tolerates that; at least 1 should have our change - if (details?.flagsChanged?.length) { - flagsChanged = details?.flagsChanged; - } - }); - }); - and(/^a flag with key "(.*)" is modified$/, async () => { - // this happens every 1s in the associated container, so wait 3s - await new Promise((resolve) => setTimeout(resolve, 3000)); - }); - then('the PROVIDER_CONFIGURATION_CHANGED handler must run', () => { - expect(ran).toBeTruthy(); - }); - and(/^the event details must indicate "(.*)" was altered$/, (flagName) => { - expect(flagsChanged).toContain(flagName); - }); - }); - - test('Resolves boolean zero value', ({ given, when, then }) => { - let flagKey: string; - let defaultValue: boolean; - - aFlagProviderIsSet(given); - when( - /^a zero-value boolean flag with key "(.*)" is evaluated with default value "(.*)"$/, - (key, defaultVal: string) => { - flagKey = key; - defaultValue = defaultVal === 'true'; - }, - ); - then(/^the resolved boolean zero-value should be "(.*)"$/, async (expectedVal: string) => { - const expectedValue = expectedVal === 'true'; - const value = await client.getBooleanValue(flagKey, defaultValue); - expect(value).toEqual(expectedValue); - }); - }); - - test('Resolves string zero value', ({ given, when, then }) => { - let flagKey: string; - let defaultValue: string; - - aFlagProviderIsSet(given); - when(/^a zero-value string flag with key "(.*)" is evaluated with default value "(.*)"$/, (key, defaultVal) => { - flagKey = key; - defaultValue = defaultVal; - }); - then('the resolved string zero-value should be ""', async () => { - const value = await client.getStringValue(flagKey, defaultValue); - expect(value).toEqual(''); - }); - }); - - test('Resolves integer zero value', ({ given, when, then }) => { - let flagKey: string; - let defaultValue: number; - - aFlagProviderIsSet(given); - when(/^a zero-value integer flag with key "(.*)" is evaluated with default value (\d+)$/, (key, defaultVal) => { - flagKey = key; - defaultValue = defaultVal; - }); - then(/^the resolved integer zero-value should be (\d+)$/, async (expectedValueString) => { - const expectedValue = Number.parseInt(expectedValueString); - const value = await client.getNumberValue(flagKey, defaultValue); - expect(value).toEqual(expectedValue); - }); - }); - - test('Resolves float zero value', ({ given, when, then }) => { - let flagKey: string; - let defaultValue: number; - - aFlagProviderIsSet(given); - when( - /^a zero-value float flag with key "(.*)" is evaluated with default value (\d+\.\d+)$/, - (key, defaultValueString) => { - flagKey = key; - defaultValue = Number.parseFloat(defaultValueString); - }, - ); - then(/^the resolved float zero-value should be (\d+\.\d+)$/, async (expectedValueString) => { - const expectedValue = Number.parseFloat(expectedValueString); - const value = await client.getNumberValue(flagKey, defaultValue); - expect(value).toEqual(expectedValue); - }); - }); - }); -} diff --git a/libs/providers/flagd/src/e2e/step-definitions/index.ts b/libs/providers/flagd/src/e2e/step-definitions/index.ts new file mode 100644 index 000000000..f5b24218f --- /dev/null +++ b/libs/providers/flagd/src/e2e/step-definitions/index.ts @@ -0,0 +1,3 @@ +export * from './flag'; +export * from './flagd-json-evaluator'; +export * from './reconnect'; diff --git a/libs/providers/flagd/src/e2e/step-definitions/reconnect.ts b/libs/providers/flagd/src/e2e/step-definitions/reconnect.ts new file mode 100644 index 000000000..9b3e73955 --- /dev/null +++ b/libs/providers/flagd/src/e2e/step-definitions/reconnect.ts @@ -0,0 +1,58 @@ +import { StepDefinitions } from 'jest-cucumber'; +import { OpenFeature, ProviderEvents } from '@openfeature/server-sdk'; +import { UNAVAILABLE_CLIENT_NAME, UNSTABLE_CLIENT_NAME } from '../constants'; + +export const reconnectStepDefinitions: StepDefinitions = ({ given, and, when, then }) => { + const client = OpenFeature.getClient(UNSTABLE_CLIENT_NAME); + + given('a flagd provider is set', () => undefined); + given('flagd is unavailable', () => undefined); + + let errorRunCount = 0; + let readyRunCount = 0; + let errorHandlerRun = 0; + + /** + * This describe block and retry settings are calibrated to gRPC's retry time + * and our testing container's restart cadence. + */ + const retryTimes = 240; + jest.retryTimes(retryTimes); + const retryDelayMs = 5000; + + beforeAll((done) => { + client.addHandler(ProviderEvents.Ready, () => { + readyRunCount++; + done(); + }); + }); + + when('a PROVIDER_READY handler and a PROVIDER_ERROR handler are added', () => { + client.addHandler(ProviderEvents.Error, () => { + errorRunCount++; + }); + }); + then('the PROVIDER_READY handler must run when the provider connects', async () => { + // should already be at 1 from `beforeAll` + expect(readyRunCount).toEqual(1); + }); + and("the PROVIDER_ERROR handler must run when the provider's connection is lost", async () => { + await new Promise((resolve) => setTimeout(resolve, retryDelayMs)); + expect(errorRunCount).toBeGreaterThan(0); + }); + and('when the connection is reestablished the PROVIDER_READY handler must run again', async () => { + await new Promise((resolve) => setTimeout(resolve, retryDelayMs)); + expect(readyRunCount).toBeGreaterThan(1); + }); + when('a flagd provider is set and initialization is awaited', () => { + OpenFeature.getClient(UNAVAILABLE_CLIENT_NAME).addHandler(ProviderEvents.Error, () => { + errorHandlerRun++; + }); + }); + + then('an error should be indicated within the configured deadline', () => { + expect(errorHandlerRun).toBeGreaterThan(0); + }); +}; + +export default reconnectStepDefinitions; diff --git a/libs/providers/flagd/src/e2e/tests/in-process-provider.spec.ts b/libs/providers/flagd/src/e2e/tests/in-process-reconnect.spec.ts similarity index 60% rename from libs/providers/flagd/src/e2e/tests/in-process-provider.spec.ts rename to libs/providers/flagd/src/e2e/tests/in-process-reconnect.spec.ts index eef1dd558..a74d3ac17 100644 --- a/libs/providers/flagd/src/e2e/tests/in-process-provider.spec.ts +++ b/libs/providers/flagd/src/e2e/tests/in-process-reconnect.spec.ts @@ -1,39 +1,27 @@ import assert from 'assert'; import { OpenFeature } from '@openfeature/server-sdk'; import { FlagdProvider } from '../../lib/flagd-provider'; +import { GenericContainer, StartedTestContainer } from 'testcontainers'; +import { autoBindSteps, loadFeature } from 'jest-cucumber'; import { - E2E_CLIENT_NAME, FLAGD_NAME, - UNSTABLE_CLIENT_NAME, + GHERKIN_FLAGD_RECONNECT_FEATURE, UNAVAILABLE_CLIENT_NAME, - IMAGE_VERSION, + UNSTABLE_CLIENT_NAME, } from '../constants'; -import { evaluation } from '../step-definitions/evaluation'; -import { GenericContainer, StartedTestContainer, TestContainer } from 'testcontainers'; -import { flagd } from '../step-definitions/flagd'; -import { flagdJsonEvaluator } from '../step-definitions/flagd-json-evaluator'; -import { flagdRecconnectUnstable } from '../step-definitions/flagd-reconnect.unstable'; +import { IMAGE_VERSION } from '@openfeature/flagd-core'; +import { reconnectStepDefinitions } from '../step-definitions'; // register the flagd provider before the tests. async function setup() { const containers: StartedTestContainer[] = []; console.log('Setting flagd provider...'); - - const stable = await new GenericContainer(`ghcr.io/open-feature/sync-testbed:${IMAGE_VERSION}`) - .withExposedPorts(9090) - .start(); - containers.push(stable); - OpenFeature.setProvider( - E2E_CLIENT_NAME, - new FlagdProvider({ resolverType: 'in-process', host: 'localhost', port: stable.getFirstMappedPort() }), - ); - const unstable = await new GenericContainer(`ghcr.io/open-feature/sync-testbed-unstable:${IMAGE_VERSION}`) .withExposedPorts(9090) .start(); containers.push(unstable); - OpenFeature.setProvider( + await OpenFeature.setProviderAndWait( UNSTABLE_CLIENT_NAME, new FlagdProvider({ resolverType: 'in-process', host: 'localhost', port: unstable.getFirstMappedPort() }), ); @@ -42,14 +30,6 @@ async function setup() { UNAVAILABLE_CLIENT_NAME, new FlagdProvider({ resolverType: 'in-process', host: 'localhost', port: 9092 }), ); - assert( - OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name === FLAGD_NAME, - new Error( - `Expected ${FLAGD_NAME} provider to be configured, instead got: ${ - OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name - }`, - ), - ); assert( OpenFeature.getProviderMetadata(UNSTABLE_CLIENT_NAME).name === FLAGD_NAME, new Error( @@ -66,10 +46,13 @@ async function setup() { }`, ), ); + console.log('flagd provider configured!'); return containers; } +jest.setTimeout(30000); + describe('in process', () => { let containers: StartedTestContainer[] = []; beforeAll(async () => { @@ -78,11 +61,9 @@ describe('in process', () => { afterAll(async () => { await OpenFeature.close(); for (const container of containers) { - container.stop(); + await container.stop(); } }); - evaluation(); - flagd(); - flagdJsonEvaluator(); - flagdRecconnectUnstable(); + const features = [loadFeature(GHERKIN_FLAGD_RECONNECT_FEATURE)]; + autoBindSteps(features, [reconnectStepDefinitions]); }); diff --git a/libs/providers/flagd/src/e2e/tests/in-process.spec.ts b/libs/providers/flagd/src/e2e/tests/in-process.spec.ts new file mode 100644 index 000000000..273d91e98 --- /dev/null +++ b/libs/providers/flagd/src/e2e/tests/in-process.spec.ts @@ -0,0 +1,60 @@ +import assert from 'assert'; +import { OpenFeature } from '@openfeature/server-sdk'; +import { FlagdProvider } from '../../lib/flagd-provider'; +import { GenericContainer, StartedTestContainer } from 'testcontainers'; +import { autoBindSteps, loadFeature } from 'jest-cucumber'; +import { + FLAGD_NAME, + GHERKIN_EVALUATION_FEATURE, + GHERKIN_FLAGD_FEATURE, + GHERKIN_FLAGD_JSON_EVALUATOR_FEATURE, +} from '../constants'; +import { flagStepDefinitions } from '../step-definitions'; +import { E2E_CLIENT_NAME, IMAGE_VERSION } from '@openfeature/flagd-core'; + +// register the flagd provider before the tests. +async function setup() { + const containers: StartedTestContainer[] = []; + + console.log('Setting flagd provider...'); + + const stable = await new GenericContainer(`ghcr.io/open-feature/sync-testbed:${IMAGE_VERSION}`) + .withExposedPorts(9090) + .start(); + containers.push(stable); + OpenFeature.setProvider( + E2E_CLIENT_NAME, + new FlagdProvider({ resolverType: 'in-process', host: 'localhost', port: stable.getFirstMappedPort() }), + ); + + assert( + OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name === FLAGD_NAME, + new Error( + `Expected ${FLAGD_NAME} provider to be configured, instead got: ${ + OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name + }`, + ), + ); + + console.log('flagd provider configured!'); + return containers; +} + +describe('in process', () => { + let containers: StartedTestContainer[] = []; + beforeAll(async () => { + containers = await setup(); + }, 60000); + afterAll(async () => { + await OpenFeature.close(); + for (const container of containers) { + await container.stop(); + } + }); + const features = [ + loadFeature(GHERKIN_FLAGD_FEATURE), + loadFeature(GHERKIN_EVALUATION_FEATURE), + loadFeature(GHERKIN_FLAGD_JSON_EVALUATOR_FEATURE), + ]; + autoBindSteps(features, [flagStepDefinitions]); +}); diff --git a/libs/providers/flagd/src/e2e/tests/rpc-provider.spec.ts b/libs/providers/flagd/src/e2e/tests/rpc-reconnect.spec.ts similarity index 64% rename from libs/providers/flagd/src/e2e/tests/rpc-provider.spec.ts rename to libs/providers/flagd/src/e2e/tests/rpc-reconnect.spec.ts index 3e17f7d9b..782028f4b 100644 --- a/libs/providers/flagd/src/e2e/tests/rpc-provider.spec.ts +++ b/libs/providers/flagd/src/e2e/tests/rpc-reconnect.spec.ts @@ -1,31 +1,22 @@ import assert from 'assert'; import { OpenFeature } from '@openfeature/server-sdk'; import { FlagdProvider } from '../../lib/flagd-provider'; +import { GenericContainer, StartedTestContainer } from 'testcontainers'; +import { autoBindSteps, loadFeature } from 'jest-cucumber'; import { - E2E_CLIENT_NAME, FLAGD_NAME, - UNSTABLE_CLIENT_NAME, + GHERKIN_FLAGD_RECONNECT_FEATURE, UNAVAILABLE_CLIENT_NAME, - IMAGE_VERSION, + UNSTABLE_CLIENT_NAME, } from '../constants'; -import { evaluation } from '../step-definitions/evaluation'; -import { GenericContainer, StartedTestContainer, TestContainer } from 'testcontainers'; -import { flagd } from '../step-definitions/flagd'; -import { flagdJsonEvaluator } from '../step-definitions/flagd-json-evaluator'; -import { flagdRecconnectUnstable } from '../step-definitions/flagd-reconnect.unstable'; +import { reconnectStepDefinitions } from '../step-definitions'; +import { IMAGE_VERSION } from '@openfeature/flagd-core'; // register the flagd provider before the tests. async function setup() { const containers: StartedTestContainer[] = []; console.log('Setting flagd provider...'); - - const stable = await new GenericContainer(`ghcr.io/open-feature/flagd-testbed:${IMAGE_VERSION}`) - .withExposedPorts(8013) - .start(); - containers.push(stable); - OpenFeature.setProvider(E2E_CLIENT_NAME, new FlagdProvider({ cache: 'disabled', port: stable.getFirstMappedPort() })); - const unstable = await new GenericContainer(`ghcr.io/open-feature/flagd-testbed-unstable:${IMAGE_VERSION}`) .withExposedPorts(8013) .start(); @@ -35,14 +26,6 @@ async function setup() { new FlagdProvider({ cache: 'disabled', port: unstable.getFirstMappedPort() }), ); OpenFeature.setProvider(UNAVAILABLE_CLIENT_NAME, new FlagdProvider({ cache: 'disabled', port: 8015 })); - assert( - OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name === FLAGD_NAME, - new Error( - `Expected ${FLAGD_NAME} provider to be configured, instead got: ${ - OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name - }`, - ), - ); assert( OpenFeature.getProviderMetadata(UNSTABLE_CLIENT_NAME).name === FLAGD_NAME, new Error( @@ -59,10 +42,12 @@ async function setup() { }`, ), ); + console.log('flagd provider configured!'); return containers; } +jest.setTimeout(30000); describe('rpc', () => { let containers: StartedTestContainer[] = []; beforeAll(async () => { @@ -78,8 +63,7 @@ describe('rpc', () => { } } }); - evaluation(); - flagd(); - flagdJsonEvaluator(); - flagdRecconnectUnstable(); + + const features = [loadFeature(GHERKIN_FLAGD_RECONNECT_FEATURE)]; + autoBindSteps(features, [reconnectStepDefinitions]); }); diff --git a/libs/providers/flagd/src/e2e/tests/rpc.spec.ts b/libs/providers/flagd/src/e2e/tests/rpc.spec.ts new file mode 100644 index 000000000..59090c1d9 --- /dev/null +++ b/libs/providers/flagd/src/e2e/tests/rpc.spec.ts @@ -0,0 +1,56 @@ +import assert from 'assert'; +import { OpenFeature } from '@openfeature/server-sdk'; +import { FlagdProvider } from '../../lib/flagd-provider'; +import { GenericContainer, StartedTestContainer } from 'testcontainers'; +import { autoBindSteps, loadFeature } from 'jest-cucumber'; +import { + FLAGD_NAME, + GHERKIN_EVALUATION_FEATURE, + GHERKIN_FLAGD_FEATURE, + GHERKIN_FLAGD_JSON_EVALUATOR_FEATURE, +} from '../constants'; +import { E2E_CLIENT_NAME, IMAGE_VERSION } from '@openfeature/flagd-core'; +import { flagStepDefinitions } from '../step-definitions'; + +// register the flagd provider before the tests. +async function setup() { + const containers: StartedTestContainer[] = []; + + console.log('Setting flagd provider...'); + const stable = await new GenericContainer(`ghcr.io/open-feature/flagd-testbed:${IMAGE_VERSION}`) + .withExposedPorts(8013) + .start(); + containers.push(stable); + OpenFeature.setProvider(E2E_CLIENT_NAME, new FlagdProvider({ cache: 'disabled', port: stable.getFirstMappedPort() })); + + assert( + OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name === FLAGD_NAME, + new Error( + `Expected ${FLAGD_NAME} provider to be configured, instead got: ${ + OpenFeature.getProviderMetadata(E2E_CLIENT_NAME).name + }`, + ), + ); + + console.log('flagd provider configured!'); + return containers; +} + +describe('rpc', () => { + let containers: StartedTestContainer[] = []; + beforeAll(async () => { + containers = await setup(); + }, 60000); + afterAll(async () => { + await OpenFeature.close(); + for (const container of containers) { + await container.stop(); + } + }); + const features = [ + loadFeature(GHERKIN_FLAGD_FEATURE), + loadFeature(GHERKIN_EVALUATION_FEATURE), + loadFeature(GHERKIN_FLAGD_JSON_EVALUATOR_FEATURE), + ]; + autoBindSteps(features, [flagStepDefinitions]); +}); diff --git a/libs/providers/flagd/tsconfig.lib.json b/libs/providers/flagd/tsconfig.lib.json index 4403f2ace..928a12324 100644 --- a/libs/providers/flagd/tsconfig.lib.json +++ b/libs/providers/flagd/tsconfig.lib.json @@ -6,5 +6,5 @@ "types": [] }, "include": ["**/*.ts"], - "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts"] + "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts", "./src/e2e"] } diff --git a/libs/providers/flagd/tsconfig.spec.json b/libs/providers/flagd/tsconfig.spec.json index 99ef89807..3243bdeca 100644 --- a/libs/providers/flagd/tsconfig.spec.json +++ b/libs/providers/flagd/tsconfig.spec.json @@ -5,5 +5,5 @@ "module": "commonjs", "types": ["jest", "node"] }, - "include": ["jest.config.ts", "**/*.test.ts", "**/*.spec.ts", "**/*.d.ts"] + "include": ["jest.config.ts", "**/*.test.ts", "**/*.spec.ts", "**/*.d.ts", "./src/e2e"] } diff --git a/libs/shared/flagd-core/flagd-schemas b/libs/shared/flagd-core/flagd-schemas index 1ba4b0324..76d611fd9 160000 --- a/libs/shared/flagd-core/flagd-schemas +++ b/libs/shared/flagd-core/flagd-schemas @@ -1 +1 @@ -Subproject commit 1ba4b0324771273e6db7d4a808f6dc87a0b3c717 +Subproject commit 76d611fd94689d906af316105ac12670d40f7648 diff --git a/libs/shared/flagd-core/package-lock.json b/libs/shared/flagd-core/package-lock.json new file mode 100644 index 000000000..37cbe347e --- /dev/null +++ b/libs/shared/flagd-core/package-lock.json @@ -0,0 +1,68 @@ +{ + "name": "@openfeature/flagd-core", + "version": "0.2.5", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "@openfeature/flagd-core", + "version": "0.2.5", + "dependencies": { + "ajv": "^8.12.0", + "tslib": "^2.3.0" + }, + "peerDependencies": { + "@openfeature/core": ">=0.0.16" + } + }, + "node_modules/@openfeature/core": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/@openfeature/core/-/core-1.4.0.tgz", + "integrity": "sha512-Cd5eeAouAYaj1RMgVq4gfasoAc4TSkN4fuhloZ3yCQA2t74IdVMAT0iadq1Seqy+G7PZoN2jy706ei9HT55PIg==", + "peer": true + }, + "node_modules/ajv": { + "version": "8.17.1", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", + "integrity": "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==", + "dependencies": { + "fast-deep-equal": "^3.1.3", + "fast-uri": "^3.0.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" + }, + "node_modules/fast-uri": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.3.tgz", + "integrity": "sha512-aLrHthzCjH5He4Z2H9YZ+v6Ujb9ocRuW6ZzkJQOrTxleEijANq4v1TsaPaVG1PZcuurEzrLcWRyYBYXD5cEiaw==" + }, + "node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" + }, + "node_modules/require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tslib": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.0.tgz", + "integrity": "sha512-jWVzBLplnCmoaTr13V9dYbiQ99wvZRd0vNWaDRg+aVYRcjDF3nDksxFDE/+fkXnKhpnUUkmx5pK/v8mCtLVqZA==" + } + } +} diff --git a/libs/shared/flagd-core/project.json b/libs/shared/flagd-core/project.json index 6169a62f8..628518d78 100644 --- a/libs/shared/flagd-core/project.json +++ b/libs/shared/flagd-core/project.json @@ -14,6 +14,17 @@ "build" ] }, + "pullTestHarness": { + "executor": "nx:run-commands", + "options": { + "commands": [ + "git submodule update --init spec", + "git submodule update --init test-harness", + ], + "cwd": "libs/shared/flagd-core", + "parallel": false + } + }, "lint": { "executor": "@nx/linter:eslint", "outputs": [ @@ -46,6 +57,9 @@ "outputs": [ "{options.outputPath}" ], + "dependsOn": [ + "pullTestHarness" + ], "options": { "project": "libs/shared/flagd-core/package.json", "outputPath": "dist/libs/shared/flagd-core", diff --git a/libs/shared/flagd-core/spec b/libs/shared/flagd-core/spec new file mode 160000 index 000000000..ffebdecd7 --- /dev/null +++ b/libs/shared/flagd-core/spec @@ -0,0 +1 @@ +Subproject commit ffebdecd725ed1a19c96927f66472c86e97ce551 diff --git a/libs/shared/flagd-core/src/e2e/index.ts b/libs/shared/flagd-core/src/e2e/index.ts new file mode 100644 index 000000000..32ac194f1 --- /dev/null +++ b/libs/shared/flagd-core/src/e2e/index.ts @@ -0,0 +1,7 @@ +export const E2E_CLIENT_NAME = 'e2e'; + +export const IMAGE_VERSION = 'v0.5.13'; + +export function getGherkinTestPath(file: string, modulePath = 'test-harness/gherkin/'): string { + return `/../../../../../shared/flagd-core/${modulePath}${file}`; +} diff --git a/libs/shared/flagd-core/src/index.ts b/libs/shared/flagd-core/src/index.ts index 71e12e9e7..7d97e85da 100644 --- a/libs/shared/flagd-core/src/index.ts +++ b/libs/shared/flagd-core/src/index.ts @@ -1,3 +1,4 @@ export * from './lib/flagd-core'; export * from './lib/feature-flag'; export * from './lib/storage'; +export * from './e2e'; diff --git a/libs/shared/flagd-core/test-harness b/libs/shared/flagd-core/test-harness new file mode 160000 index 000000000..36b07d5bd --- /dev/null +++ b/libs/shared/flagd-core/test-harness @@ -0,0 +1 @@ +Subproject commit 36b07d5bddbe854e05e60f5ea93ca70cf1a7b5c5 diff --git a/libs/shared/flagd-core/tsconfig.lib.json b/libs/shared/flagd-core/tsconfig.lib.json index 36e88999e..075a857b0 100644 --- a/libs/shared/flagd-core/tsconfig.lib.json +++ b/libs/shared/flagd-core/tsconfig.lib.json @@ -6,5 +6,5 @@ "types": [] }, "include": ["src/**/*.ts"], - "exclude": ["jest.config.ts", "src/**/*.spec.ts", "src/**/*.test.ts"] + "exclude": ["jest.config.ts", "src/**/*.spec.ts", "src/**/*.test.ts", "src/e2e"] } diff --git a/libs/shared/flagd-core/tsconfig.spec.json b/libs/shared/flagd-core/tsconfig.spec.json index 97775bd24..0dceea5a8 100644 --- a/libs/shared/flagd-core/tsconfig.spec.json +++ b/libs/shared/flagd-core/tsconfig.spec.json @@ -3,7 +3,7 @@ "compilerOptions": { "outDir": "../../../dist/out-tsc", "module": "commonjs", - "types": ["jest"] + "types": ["jest", "node"] }, - "include": ["jest.config.ts", "src/**/*.test.ts", "src/**/*.spec.ts", "src/**/*.d.ts"] + "include": ["jest.config.ts", "src/**/*.test.ts", "src/**/*.spec.ts", "src/**/*.d.ts", "src/e2e"] }