title | description | toc_max_heading_level |
---|---|---|
Flag Evaluation API |
The specification that defines the developer facing feature flag evaluation API. |
4 |
The evaluation API
allows for the evaluation of feature flag values, independent of any flag control plane or vendor. In the absence of a provider the evaluation API
uses the "No-op provider", which simply returns the supplied default flag value.
The
API
, and any state it maintains SHOULD exist as a global singleton, even in cases wherein multiple versions of theAPI
are present at runtime.
It's important that multiple instances of the API
not be active, so that state stored therein, such as the registered provider
, static global evaluation context
, and globally configured hooks
allow the API
to behave predictably. This can be difficult in some runtimes or languages, but implementors should make their best effort to ensure that only a single instance of the API
is used.
The
API
MUST define aprovider mutator
, a function to set the defaultprovider
, which accepts an API-conformantprovider
implementation.
// example provider mutator
OpenFeature.setProvider(new MyProvider());
This provider is used if there is not a more specific client name binding. (see later requirements).
See provider for details.
The
provider mutator
function MUST await the terminationinitialize
function on the newly registered provider before using it to resolve flag values.
The provider's
readiness can state can be determined from its status
member/accessor.
Provider instances which are already active (because they have been bound to other names
or otherwise) need not be initialized again.
The
provider mutator
function MUST invoke theshutdown
function on the previously registered provider once it's no longer being used to resolve flag values.
When a provider is no longer in use, it should be disposed of using its shutdown
mechanism.
Provider instances which are bound to multiple names won't be shut down until the last binding is removed.
see: shutdown, setting a provider
The
API
MUST provide a function to bind a givenprovider
to one or more clientname
s. If the client-name already has a bound provider, it is overwritten with the new mapping.
OpenFeature.setProvider("client-name", new MyProvider());
The
API
MUST provide a function to addhooks
which accepts one or more API-conformanthooks
, and appends them to the collection of any previously added hooks. When new hooks are added, previously added hooks are not removed.
// example hook attachment
OpenFeature.addHooks([new MyHook()]);
See hooks for details.
The
API
MUST provide a function for retrieving the metadata field of the configuredprovider
.
// example provider accessor
OpenFeature.getProviderMetadata();
See provider for details.
The
API
MUST provide a function for creating aclient
which accepts the following options:
- name (optional): A logical string identifier for the client.
// example client creation and retrieval
OpenFeature.getClient({
name: "my-openfeature-client",
});
The name is a logical identifier for the client.
The client creation function MUST NOT throw, or otherwise abnormally terminate.
Clients may be created in critical code paths, and even per-request in server-side HTTP contexts. Therefore, in keeping with the principle that OpenFeature should never cause abnormal execution of the first party application, this function should never throw. Abnormal execution in initialization should instead occur during provider registration.
The client MUST provide a method to add
hooks
which accepts one or more API-conformanthooks
, and appends them to the collection of any previously added hooks. When new hooks are added, previously added hooks are not removed.
// example hook attachment
client.addHooks([new MyHook()]);
See hooks for details.
The client interface MUST define a
metadata
member or accessor, containing an immutablename
field or accessor of type string, which corresponds to thename
value supplied during client creation.
client.getMetadata().getName(); // "my-client"
The
client
MUST provide methods for typed flag evaluation, including boolean, numeric, string, and structure, with parametersflag key
(string, required),default value
(boolean | number | string | structure, required),evaluation context
(optional), andevaluation options
(optional), which returns the flag value.
// example boolean flag evaluation
boolean myBool = client.getBooleanValue('bool-flag', false);
// example overloaded string flag evaluation with optional params
string myString = client.getStringValue('string-flag', 'N/A', evaluationContext, options);
// example number flag evaluation
number myNumber = client.getNumberValue('number-flag', 75);
// example overloaded structure flag evaluation with optional params
MyStruct myStruct = client.getObjectValue<MyStruct>('structured-flag', { text: 'N/A', percentage: 75 }, evaluationContext, options);
See evaluation context for details.
The implementation language differentiates between floating-point numbers and integers.
The client SHOULD provide functions for floating-point numbers and integers, consistent with language idioms.
int getIntValue(String flag, int defaultValue);
long getFloatValue(String flag, long defaultValue);
See types for details.
The
client
SHOULD guarantee the returned value of any typed flag evaluation method is of the expected type. If the value returned by the underlying provider implementation does not match the expected type, it's to be considered abnormal execution, and the supplieddefault value
should be returned.
The
client
MUST provide methods for detailed flag value evaluation with parametersflag key
(string, required),default value
(boolean | number | string | structure, required),evaluation context
(optional), andevaluation options
(optional), which returns anevaluation details
structure.
// example detailed boolean flag evaluation
FlagEvaluationDetails<boolean> myBoolDetails = client.getBooleanDetails('bool-flag', false);
// example detailed string flag evaluation
FlagEvaluationDetails<string> myStringDetails = client.getStringDetails('string-flag', 'N/A', evaluationContext, options);
// example detailed number flag evaluation
FlagEvaluationDetails<number> myNumberDetails = client.getNumberDetails('number-flag', 75);
// example detailed structure flag evaluation
FlagEvaluationDetails<MyStruct> myStructDetails = client.getObjectDetails<MyStruct>('structured-flag', { text: 'N/A', percentage: 75 }, evaluationContext, options);
The
evaluation details
structure'svalue
field MUST contain the evaluated flag value.
The language supports generics (or an equivalent feature).
The
evaluation details
structure SHOULD accept a generic argument (or use an equivalent language feature) which indicates the type of the wrappedvalue
field.
The
evaluation details
structure'sflag key
field MUST contain theflag key
argument passed to the detailed flag evaluation method.
In cases of normal execution, the
evaluation details
structure'svariant
field MUST contain the value of thevariant
field in theflag resolution
structure returned by the configuredprovider
, if the field is set.
In cases of normal execution, the
evaluation details
structure'sreason
field MUST contain the value of thereason
field in theflag resolution
structure returned by the configuredprovider
, if the field is set.
In cases of abnormal execution, the
evaluation details
structure'serror code
field MUST contain anerror code
.
See error code for details.
In cases of abnormal execution (network failure, unhandled error, etc) the
reason
field in theevaluation details
SHOULD indicate an error.
Methods, functions, or operations on the client MUST NOT throw exceptions, or otherwise abnormally terminate. Flag evaluation calls must always return the
default value
in the event of abnormal execution. Exceptions include functions or methods for the purposes for configuration or setup.
Configuration code includes code to set the provider, instantiate providers, and configure the global API object.
In the case of abnormal execution, the client SHOULD log an informative error message.
Implementations may define a standard logging interface that can be supplied as an optional argument to the client creation function, which may wrap standard logging functionality of the implementation language.
The
client
SHOULD provide asynchronous or non-blocking mechanisms for flag evaluation.
It's recommended to provide non-blocking mechanisms for flag evaluation, particularly in languages or environments wherein there's a single thread of execution.
In cases of abnormal execution, the
evaluation details
structure'serror message
field MAY contain a string containing additional details about the nature of the error.
If the
flag metadata
field in theflag resolution
structure returned by the configuredprovider
is set, theevaluation details
structure'sflag metadata
field MUST contain that value. Otherwise, it MUST contain an empty record.
This flag metadata
field is intended as a mechanism for providers to surface additional information about a feature flag (or its evaluation) beyond what is defined within the OpenFeature spec itself. The primary consumer of this information is a provider-specific hook.
The implementation language supports a mechanism for marking data as immutable.
Condition:
Flag metadata
MUST be immutable.
The
evaluation options
structure'shooks
field denotes an ordered collection of hooks that the client MUST execute for the respective flag evaluation, in addition to those already configured.
See hooks for details.
The API MUST define a mechanism to propagate a shutdown request to active providers.
The global API object might expose a shutdown
function, which will call the respective shutdown
function on the registered providers.
Alternatively, implementations might leverage language idioms such as auto-disposable interfaces or some means of cancellation signal propagation to allow for graceful shutdown.
see: shutdown