Skip to content

Latest commit

 

History

History
256 lines (187 loc) · 7.26 KB

README.md

File metadata and controls

256 lines (187 loc) · 7.26 KB

Banderole

Build Status codecov

Banderole is a versatile Javascript feature-[flags / toggle / bits / flippers] library designed with simplicity in mind. There's no black magic inside !

Install

npm install banderole 
or
yarn add banderole

Usage

All feature flags are described in a json configuration file, under a root features key.

Configure & boot banderole

const banderole = require('banderole');

//From a file
const featureFlags = require('./your-own-feature-flags-definition-file.json');
/// Content of /your-own-feature-flags-definition-file.json :
///
/// {
///    "features": {
///        "switchboard": true,
///        "clock": false,
///        "send-slack-message-on-error": {
///            "enabled": false
///        }
///    }
/// }
///

//Or from a plain javascript object
const featureFlags = {
    "features": {
        "switchboard": false,
        "clock": true,
        "send-slack-message-on-error": {
            "enabled": false
        }
    }
};

banderole.boot(featureFlags);

API

banderole.boot(flags: object, [context: object]): void

  • Boot the router. Optionally pass a context object as second argument.

const isEnabled = banderole.isEnabled(featureName: string): boolean

  • Return whether the specified feature is enabled or disabled.

banderole.addCustomRule(ruleName: string, lambda: Function<context, ...args>): void

  • Register a new rule with a custom lambda to fit your own decision logic. Let's say you want to open a send-slack-message-on-error feature only on some runtime configuration :
const currentEnv = process.env.NODE_ENV;
banderole.addCustomRule('env', (context, ...envs) => envs.includes(currentEnv));

In feature flag configuration, env refer to the lambda above ; if the currentEnv is "DEV" or "STAGING", the flag will be enabled.

"features": {
    "send-slack-message-on-error": {
        "env": ["DEV", "STAGING"]  
    }
}

For more informations on writing your own custom rules : Create your own custom rules

Rules

bandeole, the toggle router, is responsible of providing the state of a feature. There's two ways to describe your feature configuration.

Flag definition

Short expression syntax

<feature>:<flag> pattern, as in "switchboard": true is the sample below. The flag value meant to be a boolean value, true|false. It's static nature make it useful as a release toggles for separating the feature release from code deployment.

const flags = {
    "features": {
        "switchboard": true,
        "clock": false,
    }
};

banderole.boot(flags);

banderole.isEnabled('switchboard'); // true
banderole.isEnabled('clock');       // false

Rules syntax

Rules syntax are a more versatile way to decide whether the feature is enabled or disabled. They can be built-in or user-defined to fit your own logic.

Take care that the toggle router only supports one rule per feature ; if you need more control, read the strategy:* built-in rules or create your own !

Built-in rules

enabled : true|false

  • whether the feature is enabled or not ; it exists only for learning purpose : Short expression syntax is far more expressive.

strategy:affirmative

  • take as argument an hash of rules which will be evaluated one by one. With this strategy, a feature is considered as enabled as soon as one rule decide the feature is enabled.
{
    "features": {
        "shopping-cart-v2": {
            "strategy:affirmative": {
                "env": ["DEV", "QA"],
                "between-hours": ["08", "22"]
            }
        },

    },
}

In this exemple, the shopping-cart-v2 feature will be enabled if the application is running in DEV or QA environment OR between 8AM and 10PM.

strategy:unanimous

  • Arguments similar to strategy:affirmative. But unlike strategy:affirmative, the feature will be considered as enabled if all rules decide that the feature should be enabled.
{
    "features": {
        "shopping-cart-v2": {
            "strategy:unanimous": {
                "env": ["DEV", "QA"],
                "between-hours": ["08", "22"]
            }
        },

    },
}

In this exemple, the shopping-cart-v2 feature will be enabled if the application is running in DEV or QA environment AND between 8AM and 10PM.

Create your own custom rules

It's easy for banderole to fit your stack and your needs.

Just write and register your own rules !

By the exemple: the runing env

Let's say we need to enable or disable a given dazzleling feature according to the running environment, i.e. PROD, STAGING, DEV, QA...

To achieve that, you'll first register a feature called new-dazzle-feature, an env rule, taking an array of arguments : "DEV" and "QA".

const flags = {
    "features": {
        "new-dazzle-feature": {
            "env": ["DEV", "QA"]
        },
    },
};
banderole.boot(flags);

const currentEnv = process.env.NODE_ENV;
banderole.addCustomRule('new-dazzle-feature', (context, ...envs) => envs.includes(currentEnv));

banderole.isEnabled('new-dazzle-feature'); //Will return true if currentEnv is DEV or QA envs, else it will return false

By the exemple: enable a feature between 8AM and 10PM.

const flags = {
    "features": {
        "light-theme": {
            "between-hours": ["08", "22"]
        },
    },
};

banderole.boot(flags);
banderole.addCustomRule('between-hours', (context, startHour, endHour) => {
    const now = new Date();
    const hour = now.getHours();

    if (hour > startHour && hour < endHour) return true;
    return false;
};

//Between 08AM and 22PM
banderole.isEnabled('light-theme'); // true

//Between 22PM and 08AM
banderole.isEnabled('light-theme'); // false

Context object

The context object is nothing more that an user-defined databag. It is acessible to all custom rules to :

  • help keeping configuration things outside your custom rules logic
  • help keeping the feature-flags system unaware of your application logic

A common use case would be to save the current runtime-environment inside context object and read that value fom a custom-rule :

    const context = {
        env: process.env.RUNTIME_ENVIRONMENT || process.env.NODE_ENV, 
    };

    const flags = {
        features: {
            "debbug-logger": {
                "runtime-env": ["DEV"],
            }
        }
    };

    banderole.boot(flags, context);
    banderole.addCustomRule('runtime-env', (context, env) => {
        return context.env === env;
    });

You can pass anykind of data inside the context object, it's yours, feel free to also add some useful functions too !

Requesting a non-existing flag

Whenever you try to request an non-existing flag, it'll be considered as disabled.

const flags = {
    "features": {
        "switchboard": true,
    }
}

banderole.boot(flags);

banderole.isEnabled('api-v2'); // false