GrowthBook provider for OpenFeature
Project description
GrowthBook OpenFeature Provider for Python
A Python implementation of an OpenFeature provider for GrowthBook, enabling standardized feature flag evaluation in Python applications.
Requirements
- Python 3.9 or higher
- OpenFeature SDK 0.8.1+
Features
- Full implementation of the OpenFeature provider interface
- Support for all flag types (boolean, string, integer, float, object)
- Proper context mapping between OpenFeature and GrowthBook
- Asynchronous and synchronous initialization options
- Comprehensive error handling
- Resource cleanup utilities
Installation
pip install growthbook-openfeature-provider
Quick Start
Async Usage (Recommended)
import asyncio
from openfeature.api import OpenFeatureAPI
from openfeature.evaluation_context import EvaluationContext
from growthbook_openfeature_provider import GrowthBookProvider, GrowthBookProviderOptions
async def main():
# Create and initialize the provider
provider = GrowthBookProvider(GrowthBookProviderOptions(
api_host="https://cdn.growthbook.io",
client_key="sdk-abc123" # Replace with your actual SDK key
))
# Initialize the provider
await provider.initialize()
# Register with OpenFeature
OpenFeatureAPI.set_provider(provider)
# Get a client
client = OpenFeatureAPI.get_client("my-app")
# Create an evaluation context with targeting information
context = EvaluationContext(
targeting_key="user-123",
attributes={
"country": "US",
"email": "user@example.com",
"premium": True
}
)
details = await provider.resolve_boolean_details_async("my-flag", False, context)
print(f"Flag value: {details.value}, reason: {details.reason}")
# Clean up resources when done
await provider.close()
# Run the async function
asyncio.run(main())
Synchronous Usage
from openfeature.api import OpenFeatureAPI
from openfeature.evaluation_context import EvaluationContext
from growthbook_openfeature_provider import GrowthBookProvider, GrowthBookProviderOptions
# Create provider
provider = GrowthBookProvider(GrowthBookProviderOptions(
api_host="https://cdn.growthbook.io",
client_key="sdk-abc123"
))
# Initialize synchronously
provider.initialize_sync()
# Register with OpenFeature
OpenFeatureAPI.set_provider(provider)
# Get a client and evaluate flags
client = OpenFeatureAPI.get_client("my-app")
context = EvaluationContext(
targeting_key="user-123",
attributes={"country": "US", "premium": True}
)
value = client.get_boolean_value("my-flag", False, context)
print(f"Flag value: {value}")
# Clean up (can be called from sync context)
import asyncio
asyncio.run(provider.close())
Configuration Options
The GrowthBookProviderOptions class accepts the following parameters:
| Parameter | Type | Description | Default |
|---|---|---|---|
api_host |
str |
URL of the GrowthBook API | Required |
client_key |
str |
API key for authentication | Required |
decryption_key |
str |
Key for encrypted features | "" |
cache_ttl |
int |
Cache duration in seconds | 60 |
enabled |
bool |
Whether GrowthBook is enabled | True |
qa_mode |
bool |
Enable QA mode for testing | False |
on_experiment_viewed |
Callable |
Callback when experiments are viewed | None |
sticky_bucket_service |
AbstractStickyBucketService |
Service for consistent experiment assignments | None |
Evaluation Context
The provider maps OpenFeature evaluation context to GrowthBook user context:
- The
targeting_keyis mapped to theidattribute - All other attributes are passed through directly
Example:
context = EvaluationContext(
targeting_key="user-123",
attributes={
"country": "US",
"deviceId": "device-456",
"premium": True
}
)
This creates a GrowthBook context with:
{
"id": "user-123",
"country": "US",
"deviceId": "device-456",
"premium": true
}
Flag Evaluation
The provider supports all OpenFeature flag types with both synchronous and asynchronous methods:
Synchronous Methods (for sync contexts)
# Boolean flags
boolean_value = client.get_boolean_value("my-boolean-flag", False, context)
# String flags
string_value = client.get_string_value("my-string-flag", "default", context)
# Integer flags
int_value = client.get_integer_value("my-number-flag", 0, context)
# Float flags
float_value = client.get_float_value("my-float-flag", 0.0, context)
# Object flags
object_value = client.get_object_value("my-object-flag", {"default": True}, context)
Asynchronous Methods (recommended for async contexts)
# Boolean flags
boolean_details = await provider.resolve_boolean_details_async("my-boolean-flag", False, context)
boolean_value = boolean_details.value
# String flags
string_details = await provider.resolve_string_details_async("my-string-flag", "default", context)
string_value = string_details.value
# Integer flags
int_details = await provider.resolve_integer_details_async("my-number-flag", 0, context)
int_value = int_details.value
# Float flags
float_details = await provider.resolve_float_details_async("my-float-flag", 0.0, context)
float_value = float_details.value
# Object flags
object_details = await provider.resolve_object_details_async("my-object-flag", {"default": True}, context)
object_value = object_details.value
Evaluation Results
Evaluation results include value, reason and variant:
# Asynchronous evaluation
details = await provider.resolve_boolean_details_async("my-flag", False, context)
print(f"Value: {details.value}")
print(f"Reason: {details.reason}")
print(f"Variant: {details.variant}")
Error Handling
The provider handles various error conditions gracefully:
- Uninitialized provider: Returns
PROVIDER_NOT_READYerror - Missing targeting key: Returns
TARGETING_KEY_MISSINGerror - Type conversion errors: Returns
TYPE_MISMATCHerror - Network failures: Returns default values with
ERRORreason - General exceptions: Returns
GENERALerror with message
Resource Cleanup
Always clean up resources when done:
# Async cleanup (recommended)
await provider.close()
# Sync cleanup
import asyncio
asyncio.run(provider.close())
Examples
See the examples directory for more usage examples, including:
- Basic usage patterns
- FastAPI integration
- Error handling
- Performance optimization
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
- Fork the repository
- Create your feature branch (
git checkout -b feature/amazing-feature) - Commit your changes (
git commit -m 'Add some amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request
License
This project is licensed under the MIT License - see the LICENSE file for details.
Acknowledgements
- OpenFeature - For the feature flag standard
- GrowthBook - For the feature flagging and experimentation platform
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file growthbook_openfeature_provider-0.0.6.tar.gz.
File metadata
- Download URL: growthbook_openfeature_provider-0.0.6.tar.gz
- Upload date:
- Size: 16.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.8.18
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d2af9e4ed52b367b22b3c76be666f00b71c8378392b4eb4700dcff82c1eb418e
|
|
| MD5 |
964e6e0037d3876eb5322535306ca96a
|
|
| BLAKE2b-256 |
551af9219925703d7272fb5fe148546d190706b7ea66ef8d6dc1e2113472e408
|
File details
Details for the file growthbook_openfeature_provider-0.0.6-py3-none-any.whl.
File metadata
- Download URL: growthbook_openfeature_provider-0.0.6-py3-none-any.whl
- Upload date:
- Size: 9.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.8.18
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f3c247939b109dbc138af173c5a9365d0abbede5bc053a871434cabe889574e1
|
|
| MD5 |
ba6db2a1976e29442c0c3c3aacf70c47
|
|
| BLAKE2b-256 |
27f4d9f0e358024b1fff6145aa82ad469c37befd29e25f69042971b33f616869
|