Skip to content

Commit 0dfffdc

Browse files
feat(go-feature-flag): Support exporter metadata during evaluation (open-feature#1186)
Signed-off-by: Thomas Poignant <[email protected]>
1 parent 9de707a commit 0dfffdc

File tree

3 files changed

+50
-1
lines changed

3 files changed

+50
-1
lines changed

libs/providers/go-feature-flag/src/lib/controller/goff-api.ts

+13
Original file line numberDiff line numberDiff line change
@@ -56,19 +56,32 @@ export class GoffApiController {
5656
* @param defaultValue
5757
* @param evaluationContext
5858
* @param expectedType
59+
* @param exporterMetadata
5960
*/
6061
async evaluate<T>(
6162
flagKey: string,
6263
defaultValue: T,
6364
evaluationContext: EvaluationContext,
6465
expectedType: string,
66+
exporterMetadata: Record<string, ExporterMetadataValue> = {},
6567
): Promise<{ resolutionDetails: ResolutionDetails<T>; isCacheable: boolean }> {
6668
const goffEvaluationContext = transformContext(evaluationContext);
6769

6870
// build URL to access to the endpoint
6971
const endpointURL = new URL(this.endpoint);
7072
endpointURL.pathname = `v1/feature/${flagKey}/eval`;
7173

74+
if (goffEvaluationContext.custom === undefined) {
75+
goffEvaluationContext.custom = {};
76+
}
77+
goffEvaluationContext.custom['gofeatureflag'] = {
78+
exporterMetadata: {
79+
openfeature: true,
80+
provider: 'js',
81+
...exporterMetadata,
82+
},
83+
};
84+
7285
const request: GoFeatureFlagProxyRequest<T> = {
7386
evaluationContext: goffEvaluationContext,
7487
defaultValue,

libs/providers/go-feature-flag/src/lib/go-feature-flag-provider.spec.ts

+33
Original file line numberDiff line numberDiff line change
@@ -212,6 +212,39 @@ describe('GoFeatureFlagProvider', () => {
212212
it('provider should be ready after after setting the provider to Open Feature', async () => {
213213
expect(cli.providerStatus).toEqual(ServerProviderStatus.READY);
214214
});
215+
216+
it('should send exporter metadata to the evaluation API', async () => {
217+
const flagName = 'random-flag';
218+
const targetingKey = 'user-key';
219+
const dns = `${endpoint}v1/feature/${flagName}/eval`;
220+
axiosMock.onPost(dns).reply(200, {
221+
value: true,
222+
variationType: 'trueVariation',
223+
reason: StandardResolutionReasons.TARGETING_MATCH,
224+
failed: false,
225+
trackEvents: true,
226+
version: '1.0.0',
227+
} as GoFeatureFlagProxyResponse<boolean>);
228+
229+
const provider = new GoFeatureFlagProvider({
230+
endpoint,
231+
exporterMetadata: { key1: 'value', key2: 123, key3: 123.45 },
232+
});
233+
await OpenFeature.setProviderAndWait('test-exporter-metadata', provider);
234+
const cli = OpenFeature.getClient('test-exporter-metadata');
235+
236+
await cli.getBooleanDetails(flagName, false, { targetingKey });
237+
const request = axiosMock.history.post[0];
238+
const want = {
239+
openfeature: true,
240+
provider: 'js',
241+
key1: 'value',
242+
key2: 123,
243+
key3: 123.45,
244+
};
245+
const got = JSON.parse(request.data).evaluationContext.custom.gofeatureflag.exporterMetadata;
246+
expect(got).toEqual(want);
247+
});
215248
});
216249
describe('resolveBooleanEvaluation', () => {
217250
it('should throw an error if we expect a boolean and got another type', async () => {

libs/providers/go-feature-flag/src/lib/go-feature-flag-provider.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import {
99
ServerProviderEvents,
1010
StandardResolutionReasons,
1111
} from '@openfeature/server-sdk';
12-
import { ConfigurationChange, GoFeatureFlagProviderOptions } from './model';
12+
import { ConfigurationChange, ExporterMetadataValue, GoFeatureFlagProviderOptions } from './model';
1313
import { GoFeatureFlagDataCollectorHook } from './data-collector-hook';
1414
import { GoffApiController } from './controller/goff-api';
1515
import { CacheController } from './controller/cache';
@@ -34,6 +34,7 @@ export class GoFeatureFlagProvider implements Provider {
3434
private readonly _cacheController?: CacheController;
3535
private _pollingIntervalId?: number;
3636
private _pollingInterval: number;
37+
private _exporterMetadata?: Record<string, ExporterMetadataValue>;
3738

3839
constructor(options: GoFeatureFlagProviderOptions, logger?: Logger) {
3940
this._goffApiController = new GoffApiController(options);
@@ -46,6 +47,7 @@ export class GoFeatureFlagProvider implements Provider {
4647
this._goffApiController,
4748
logger,
4849
);
50+
this._exporterMetadata = options.exporterMetadata;
4951
this._disableDataCollection = options.disableDataCollection || false;
5052
this._cacheController = new CacheController(options, logger);
5153
this._pollingInterval = options.pollInterval ?? this.DEFAULT_POLL_INTERVAL;
@@ -188,6 +190,7 @@ export class GoFeatureFlagProvider implements Provider {
188190
defaultValue,
189191
evaluationContext,
190192
expectedType,
193+
this._exporterMetadata ?? {},
191194
);
192195

193196
this._cacheController?.set(flagKey, evaluationContext, evaluationResponse);

0 commit comments

Comments
 (0)