Skip to content

Commit 9a9950d

Browse files
authored
chore: add integration tests (#225)
Add integration test suite. Signed-off-by: Todd Baert <[email protected]>
1 parent ebb6c37 commit 9a9950d

14 files changed

+13830
-5039
lines changed

.github/workflows/pr-checks.yaml

+11-2
Original file line numberDiff line numberDiff line change
@@ -13,18 +13,27 @@ jobs:
1313
build-test-lint:
1414
runs-on: ubuntu-latest
1515

16+
services:
17+
flagd:
18+
image: ghcr.io/open-feature/flagd-testbed:latest
19+
ports:
20+
- 8013:8013
21+
1622
steps:
1723
- uses: actions/checkout@v3
1824
- uses: actions/setup-node@v3
1925

2026
- name: Install
2127
run: npm ci
2228

29+
- name: Build
30+
run: npm run build
31+
2332
- name: Lint
2433
run: npm run lint
2534

26-
- name: Build
27-
run: npm run build
35+
- name: Integration
36+
run: npm run integration
2837

2938
- name: Test
3039
run: npm run test

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -102,3 +102,7 @@ dist
102102

103103
# TernJS port file
104104
.tern-port
105+
106+
# yalc stuff
107+
yalc.lock
108+
.yalc/

.gitmodules

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
[submodule "test-harness"]
2+
path = test-harness
3+
url = https://github.com/open-feature/test-harness.git

README.md

+4
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,10 @@ We value having as few runtime dependencies as possible. The addition of any dep
4848

4949
Run tests with `npm test`.
5050

51+
### Integration tests
52+
53+
The continuous integration runs a set of [gherkin integration tests](https://github.com/open-feature/test-harness/blob/main/features/evaluation.feature) using [`flagd`](https://github.com/open-feature/flagd). These tests run with the "integration" npm script. If you'd like to run them locally, you can start the flagd testbed with `docker run -p 8013:8013 ghcr.io/open-feature/flagd-testbed:latest` and then run `npm run integration`.
54+
5155
### Packaging
5256

5357
Both ES modules and CommonJS modules are supported, so consumers can use both `require` and `import` functions to utilize this module. This is accomplished by building 2 variations of the output, under `dist/esm` and `dist/cjs`, respectively. To force resolution of the `dist/esm/**.js*` files as modules, a package json with only the context `{"type": "module"}` is included at a in a `postbuild` step. Type declarations are included at `/dist/types/`

integration/features/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
evaluation.feature

integration/features/.gitkeep

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,315 @@
1+
import { defineFeature, loadFeature } from 'jest-cucumber';
2+
import { OpenFeature } from '../../src/open-feature';
3+
import {
4+
EvaluationContext,
5+
EvaluationDetails,
6+
JsonObject,
7+
JsonValue,
8+
ResolutionDetails, StandardResolutionReasons
9+
} from '../../src/types';
10+
11+
// load the feature file.
12+
const feature = loadFeature('integration/features/evaluation.feature');
13+
14+
// get a client (flagd provider registered in setup)
15+
const client = OpenFeature.getClient();
16+
17+
defineFeature(feature, (test) => {
18+
test('Resolves boolean value', ({ when, then }) => {
19+
let value: boolean;
20+
let flagKey: string;
21+
22+
when(
23+
/^a boolean flag with key '(.*)' is evaluated with default value '(.*)'$/,
24+
async (key: string, defaultValue: string) => {
25+
flagKey = key;
26+
value = await client.getBooleanValue(flagKey, defaultValue === 'true');
27+
}
28+
);
29+
30+
then(/^the resolved boolean value should be '(.*)'$/, (expectedValue: string) => {
31+
expect(value).toEqual(expectedValue === 'true');
32+
});
33+
});
34+
35+
test('Resolves string value', ({ when, then }) => {
36+
let value: string;
37+
let flagKey: string;
38+
39+
when(
40+
/^a string flag with key '(.*)' is evaluated with default value '(.*)'$/,
41+
async (key: string, defaultValue: string) => {
42+
flagKey = key;
43+
value = await client.getStringValue(flagKey, defaultValue);
44+
}
45+
);
46+
47+
then(/^the resolved string value should be '(.*)'$/, (expectedValue: string) => {
48+
expect(value).toEqual(expectedValue);
49+
});
50+
});
51+
52+
test('Resolves integer value', ({ when, then }) => {
53+
let value: number;
54+
let flagKey: string;
55+
56+
when(
57+
/^an integer flag with key '(.*)' is evaluated with default value (\d+)$/,
58+
async (key: string, defaultValue: string) => {
59+
flagKey = key;
60+
value = await client.getNumberValue(flagKey, Number.parseInt(defaultValue));
61+
}
62+
);
63+
64+
then(/^the resolved integer value should be (\d+)$/, (expectedValue: string) => {
65+
expect(value).toEqual(Number.parseInt(expectedValue));
66+
});
67+
});
68+
69+
test('Resolves float value', ({ when, then }) => {
70+
let value: number;
71+
let flagKey: string;
72+
73+
when(
74+
/^a float flag with key '(.*)' is evaluated with default value (\d+\.?\d*)$/,
75+
async (key: string, defaultValue: string) => {
76+
flagKey = key;
77+
value = await client.getNumberValue(flagKey, Number.parseFloat(defaultValue));
78+
}
79+
);
80+
81+
then(/^the resolved float value should be (\d+\.?\d*)$/, (expectedValue: string) => {
82+
expect(value).toEqual(Number.parseFloat(expectedValue));
83+
});
84+
});
85+
86+
test('Resolves object value', ({ when, then }) => {
87+
let value: JsonValue;
88+
let flagKey: string;
89+
90+
when(/^an object flag with key '(.*)' is evaluated with a null default value$/, async (key: string) => {
91+
flagKey = key;
92+
value = await client.getObjectValue(flagKey, {});
93+
});
94+
95+
then(
96+
/^the resolved object value should be contain fields '(.*)', '(.*)', and '(.*)', with values '(.*)', '(.*)' and (\d+), respectively$/,
97+
(field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => {
98+
const jsonObject = value as JsonObject;
99+
expect(jsonObject[field1]).toEqual(boolValue === 'true');
100+
expect(jsonObject[field2]).toEqual(stringValue);
101+
expect(jsonObject[field3]).toEqual(Number.parseInt(intValue));
102+
}
103+
);
104+
});
105+
106+
test('Resolves boolean details', ({ when, then }) => {
107+
let details: EvaluationDetails<boolean>;
108+
let flagKey: string;
109+
110+
when(
111+
/^a boolean flag with key '(.*)' is evaluated with details and default value '(.*)'$/,
112+
async (key: string, defaultValue: string) => {
113+
flagKey = key;
114+
details = await client.getBooleanDetails(flagKey, defaultValue === 'true');
115+
}
116+
);
117+
118+
then(
119+
/^the resolved boolean details value should be '(.*)', the variant should be '(.*)', and the reason should be '(.*)'$/,
120+
(expectedValue: string, expectedVariant: string, expectedReason: string) => {
121+
expect(details.value).toEqual(expectedValue === 'true');
122+
expect(details.variant).toEqual(expectedVariant);
123+
expect(details.reason).toEqual(expectedReason);
124+
}
125+
);
126+
});
127+
128+
test('Resolves string details', ({ when, then }) => {
129+
let details: EvaluationDetails<string>;
130+
let flagKey: string;
131+
132+
when(
133+
/^a string flag with key '(.*)' is evaluated with details and default value '(.*)'$/,
134+
async (key: string, defaultValue: string) => {
135+
flagKey = key;
136+
details = await client.getStringDetails(flagKey, defaultValue);
137+
}
138+
);
139+
140+
then(
141+
/^the resolved string details value should be '(.*)', the variant should be '(.*)', and the reason should be '(.*)'$/,
142+
(expectedValue: string, expectedVariant: string, expectedReason: string) => {
143+
expect(details.value).toEqual(expectedValue);
144+
expect(details.variant).toEqual(expectedVariant);
145+
expect(details.reason).toEqual(expectedReason);
146+
}
147+
);
148+
});
149+
150+
test('Resolves integer details', ({ when, then }) => {
151+
let details: EvaluationDetails<number>;
152+
let flagKey: string;
153+
154+
when(
155+
/^an integer flag with key '(.*)' is evaluated with details and default value (\d+)$/,
156+
async (key: string, defaultValue: string) => {
157+
flagKey = key;
158+
details = await client.getNumberDetails(flagKey, Number.parseInt(defaultValue));
159+
}
160+
);
161+
162+
then(
163+
/^the resolved integer details value should be (\d+), the variant should be '(.*)', and the reason should be '(.*)'$/,
164+
(expectedValue: string, expectedVariant: string, expectedReason: string) => {
165+
expect(details.value).toEqual(Number.parseInt(expectedValue));
166+
expect(details.variant).toEqual(expectedVariant);
167+
expect(details.reason).toEqual(expectedReason);
168+
}
169+
);
170+
});
171+
172+
test('Resolves float details', ({ when, then }) => {
173+
let details: EvaluationDetails<number>;
174+
let flagKey: string;
175+
176+
when(
177+
/^a float flag with key '(.*)' is evaluated with details and default value (\d+\.?\d*)$/,
178+
async (key: string, defaultValue: string) => {
179+
flagKey = key;
180+
details = await client.getNumberDetails(flagKey, Number.parseFloat(defaultValue));
181+
}
182+
);
183+
184+
then(
185+
/^the resolved float details value should be (\d+\.?\d*), the variant should be '(.*)', and the reason should be '(.*)'$/,
186+
(expectedValue: string, expectedVariant: string, expectedReason: string) => {
187+
expect(details.value).toEqual(Number.parseFloat(expectedValue));
188+
expect(details.variant).toEqual(expectedVariant);
189+
expect(details.reason).toEqual(expectedReason);
190+
}
191+
);
192+
});
193+
194+
test('Resolves object details', ({ when, then, and }) => {
195+
let details: EvaluationDetails<JsonValue>; // update this after merge
196+
let flagKey: string;
197+
198+
when(/^an object flag with key '(.*)' is evaluated with details and a null default value$/, async (key: string) => {
199+
flagKey = key;
200+
details = await client.getObjectDetails(flagKey, {}); // update this after merge
201+
});
202+
203+
then(
204+
/^the resolved object details value should be contain fields '(.*)', '(.*)', and '(.*)', with values '(.*)', '(.*)' and (\d+), respectively$/,
205+
(field1: string, field2: string, field3: string, boolValue: string, stringValue: string, intValue: string) => {
206+
const jsonObject = details.value as JsonObject;
207+
208+
expect(jsonObject[field1]).toEqual(boolValue === 'true');
209+
expect(jsonObject[field2]).toEqual(stringValue);
210+
expect(jsonObject[field3]).toEqual(Number.parseInt(intValue));
211+
}
212+
);
213+
214+
and(
215+
/^the variant should be '(.*)', and the reason should be '(.*)'$/,
216+
(expectedVariant: string, expectedReason: string) => {
217+
expect(details.variant).toEqual(expectedVariant);
218+
expect(details.reason).toEqual(expectedReason);
219+
}
220+
);
221+
});
222+
223+
test('Resolves based on context', ({ when, and, then }) => {
224+
const context: EvaluationContext = {};
225+
let value: string;
226+
let flagKey: string;
227+
228+
when(
229+
/^context contains keys '(.*)', '(.*)', '(.*)', '(.*)' with values '(.*)', '(.*)', (\d+), '(.*)'$/,
230+
(
231+
stringField1: string,
232+
stringField2: string,
233+
intField: string,
234+
boolField: string,
235+
stringValue1: string,
236+
stringValue2: string,
237+
intValue: string,
238+
boolValue: string
239+
) => {
240+
context[stringField1] = stringValue1;
241+
context[stringField2] = stringValue2;
242+
context[intField] = Number.parseInt(intValue);
243+
context[boolField] = boolValue === 'true';
244+
}
245+
);
246+
247+
and(/^a flag with key '(.*)' is evaluated with default value '(.*)'$/, async (key: string, defaultValue: string) => {
248+
flagKey = key;
249+
value = await client.getStringValue(flagKey, defaultValue, context);
250+
});
251+
252+
then(/^the resolved string response should be '(.*)'$/, (expectedValue: string) => {
253+
expect(value).toEqual(expectedValue);
254+
});
255+
256+
and(/^the resolved flag value is '(.*)' when the context is empty$/, async (expectedValue) => {
257+
const emptyContextValue = await client.getStringValue(flagKey, 'nope', {});
258+
expect(emptyContextValue).toEqual(expectedValue);
259+
});
260+
});
261+
262+
test('Flag not found', ({ when, then, and }) => {
263+
let flagKey: string;
264+
let fallbackValue: string;
265+
let details: ResolutionDetails<string>;
266+
267+
when(
268+
/^a non-existent string flag with key '(.*)' is evaluated with details and a default value '(.*)'$/,
269+
async (key: string, defaultValue: string) => {
270+
flagKey = key;
271+
fallbackValue = defaultValue;
272+
details = await client.getStringDetails(flagKey, defaultValue);
273+
}
274+
);
275+
276+
then(/^then the default string value should be returned$/, () => {
277+
expect(details.value).toEqual(fallbackValue);
278+
});
279+
280+
and(
281+
/^the reason should indicate an error and the error code should indicate a missing flag with '(.*)'$/,
282+
(errorCode: string) => {
283+
expect(details.reason).toEqual(StandardResolutionReasons.ERROR);
284+
expect(details.errorCode).toEqual(errorCode);
285+
}
286+
);
287+
});
288+
289+
test('Type error', ({ when, then, and }) => {
290+
let flagKey: string;
291+
let fallbackValue: number;
292+
let details: ResolutionDetails<number>;
293+
294+
when(
295+
/^a string flag with key '(.*)' is evaluated as an integer, with details and a default value (\d+)$/,
296+
async (key: string, defaultValue: string) => {
297+
flagKey = key;
298+
fallbackValue = Number.parseInt(defaultValue);
299+
details = await client.getNumberDetails(flagKey, Number.parseInt(defaultValue));
300+
}
301+
);
302+
303+
then(/^then the default integer value should be returned$/, () => {
304+
expect(details.value).toEqual(fallbackValue);
305+
});
306+
307+
and(
308+
/^the reason should indicate an error and the error code should indicate a type mismatch with '(.*)'$/,
309+
(errorCode: string) => {
310+
expect(details.reason).toEqual(StandardResolutionReasons.ERROR);
311+
expect(details.errorCode).toEqual(errorCode);
312+
}
313+
);
314+
});
315+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
export default {
2+
clearMocks: true,
3+
collectCoverage: true,
4+
coverageDirectory: 'coverage',
5+
coverageProvider: 'v8',
6+
globals: {
7+
'ts-jest': {
8+
tsConfig: 'integration/step-definitions/tsconfig.json',
9+
},
10+
},
11+
moduleNameMapper: {
12+
'^(.*)\\.js$': ['$1', '$1.js'],
13+
},
14+
setupFiles: ['./setup.ts'],
15+
preset: 'ts-jest',
16+
};

0 commit comments

Comments
 (0)