featurit / featurit-sdk-symfony
Symfony SDK for FeaturIT
Installs: 1 165
Dependents: 0
Suggesters: 0
Security: 0
Stars: 4
Watchers: 3
Forks: 0
Open Issues: 0
Type:symfony-bundle
pkg:composer/featurit/featurit-sdk-symfony
Requires
- php: ^8.1
- featurit/featurit-sdk-php: ^1.0.0
- nyholm/psr7: ^1.5
- symfony/framework-bundle: ^5.2 || ^6.0 || ^7.0
Requires (Dev)
- matthiasnoback/symfony-dependency-injection-test: ^4.3
- predis/predis: ^2.3
- symfony/http-client: ^5.2-dev
- symfony/phpunit-bridge: ^5.2-dev
- symfony/security-core: ^5.2-dev
- twig/twig: ^3.3
README
Symfony wrapper of the PHP client for the FeaturIT Feature Flag management platform.
Description
This package aims to simplify the integration of the FeaturIT API in a Symfony project.
Getting started
Dependencies
- PHP >= 8.0.2
- symfony/framework-bundle >= 5.2
- psr/http-client-implementation
- psr/simple-cache-implementation
Installing
composer require featurit/featurit-sdk-symfony -W
If there's no package providing psr/http-client-implementation, visit https://packagist.org/providers/psr/http-client-implementation and choose the package that better suits your project.
If there's no package providing psr/simple-cache-implementation, visit https://packagist.org/providers/psr/simple-cache-implementation and choose the package that better suits your project.
Inside your config/bundles.php file, add:
    Featurit\Client\Symfony\FeaturitBundle::class => ['all' => true],
If you want to create your own configuration file in order to customize things
like the default FeaturitUserContextProvider, create a file in config/packages/featurit.yaml
with the following contents:
featurit:
    tenant_identifier: '%env(string:FEATURIT_TENANT_IDENTIFIER)%'
    environment_key: '%env(string:FEATURIT_ENVIRONMENT_KEY)%'
    enable_analytics: '%env(bool:FEATURIT_ENABLE_ANALYTICS)%'
    enable_tracking: '%env(bool:FEATURIT_ENABLE_TRACKING)%'
    cache_ttl_minutes: '%env(int:FEATURIT_CACHE_TTL_MINUTES)%'
    send_analytics_interval_minutes: '%env(int:FEATURIT_SEND_ANALYTICS_INTERVAL_MINUTES)%'
    featurit_user_context_provider: '@my_service_implementing_featurit_user_context_provider'
You also need to create the proper environment variables in your .env file.
Basic Usage
That's how you would use Featurit in one of your controllers, services, or anywhere inside your PHP codebase:
your_method(Featurit $featurit)
{
    if ($featurit->isActive('YOUR_FEATURE_NAME')) {
        your_feature_code();
    }
}
Or in order to check which is the version of your feature:
your_method(Featurit $featurit)
{
    if ($featurit->version('YOUR_FEATURE_NAME') == 'v1') {
        your_feature_code_for_v1();
    } else if ($featurit->version('YOUR_FEATURE_NAME') == 'v2') {
        your_feature_code_for_v2();
    }
}
Twig extension
For convenience we provide 2 twig functions which allow to render html depending on the Feature Flag values.
Inside your twig template, you can use them like this:
<div>
    <h2>This code will always be visible</h2>
    {% if feature_is_active('MY_ACTIVE_FEATURE') %}
        <h2>Welcome to MY_ACTIVE_FEATURE!</h2>
    {% endif %}
    {% if feature_version_equals('FEATURE_WITH_VERSIONS', 'v1') %}
        <h2>Welcome to v1!</h2>
    {% elseif feature_version_equals('FEATURE_WITH_VERSIONS', 'v2') %}
        <h2>Welcome to v2!</h2>
    {% endif %}
</div>
Defining your FeaturitUserContext
In order to show different versions of a feature to different users, Featurit needs to know about the attributes your user has in a certain context.
You can define the context using the as follows:
your_method(Featurit $featurit)
{
    $contextData = get_your_user_context_data();
    $featurit->setUserContext(
        new DefaultFeaturitUserContext(
            $contextData['userId'],
            $contextData['sessionId'],
            $contextData['ipAddress'],
            [
                'role' => $contextData['role'],
                ...
            ]
        )
    );
}
Defining a custom FeaturitUserContextProvider
This is an alternative to using $featurit->setUserContext(...);.
By default, Featurit SDK for Symfony comes with a default FeaturitUserContextProvider
adapted for Symfony, but if you want to create your own, create a service un your services.yaml file as follows:
services:
...
    Namespace\For\MyFeaturitUserContextProvider
        arguments:
            - arg1
            - arg2
            - ...
And then add it to the featurit.yaml config file as:
featurit:
    ...
    featurit_user_context_provider: '@Namespace\For\MyFeaturitUserContextProvider'
Let's say that your platform users have a "role" attribute that you use to decide which features you show to each user. In that case you could create an implementation like:
<?php
namespace My\Namespace\Of\Choice;
use Featurit\Client\Modules\Segmentation\DefaultFeaturitUserContext;
use Featurit\Client\Modules\Segmentation\FeaturitUserContext;
use Featurit\Client\Modules\Segmentation\FeaturitUserContextProvider;
class MyCustomFeaturitUserContextProvider implements FeaturitUserContextProvider
{
    public function getUserContext(): FeaturitUserContext
    {
        $userId = my_logic_to_get_the_user_identifier();
        $sessionId = my_logic_to_get_the_session_id();
        $ipAddress = my_logic_to_get_the_ip_address();
        
        $role = my_logic_to_get_the_user_role();
        return new DefaultFeaturitUserContext(
            $userId,
            $sessionId,
            $ipAddress,
            [
                'role' => $role,
            ]
        );
    }
}
Then you must replace your implementation in the featurit.yaml file as explained before.
And that should do it, from now on your segmentation rules will use the role attribute.
Event Tracking
In order to track some event in your application, you can add this once the event has happened:
your_method(Featurit $featurit)
{
    $contextData = get_your_user_context_data();
    $featurit->setUserContext(
        new DefaultFeaturitUserContext(
            $contextData['userId'],
            $contextData['sessionId'],
            $contextData['ipAddress'],
            [
                'role' => $contextData['role'],
                ...
            ]
        )
    );
    
    $featurit->trackPerson();
    
    // trackPerson will track a new Person with the data set in the FeaturitUserContext.
    
    $featurit->track('YOUR_EVENT_NAME', [
        'an_event_property_name' => 'an_event_property_value',
        'another_event_property_name' => 'another_event_property_value',
    ]);
}
All the events you track in the same request will be accumulated and associated to the current FeaturitUserContext, if for some reason you want to send the event immediately, you can do as follows:
$featurit->flush();
Custom cache implementation
By default, FeaturIT SDK for Symfony provides a simple File cache.
Once you have multiple servers, you will want to inject your custom Psr\SimpleCache\CacheInterface implementation
using some distributed cache system.
This can be done as follows:
featurit:
    tenant_identifier: '%env(string:FEATURIT_TENANT_IDENTIFIER)%'
    environment_key: '%env(string:FEATURIT_ENVIRONMENT_KEY)%'
    enable_analytics: '%env(bool:FEATURIT_ENABLE_ANALYTICS)%'
    enable_tracking: '%env(bool:FEATURIT_ENABLE_TRACKING)%'
    cache_ttl_minutes: '%env(int:FEATURIT_CACHE_TTL_MINUTES)%'
    send_analytics_interval_minutes: '%env(int:FEATURIT_SEND_ANALYTICS_INTERVAL_MINUTES)%'
    featurit_user_context_provider: '@my_service_implementing_featurit_user_context_provider'
    cache: '@my_custom_psr_cache_interface_implementation'
Authors
FeaturIT