Static Site Feature Flags

Note: Turn off any ad-blockers! Ad-blockers can prevent this demo from working.

Feature flags enable a powerful continuous delivery process and provide a platform for progressive delivery with phased rollouts and A/B tests.

However, feature flags installed in a static site can be difficult because there's usually not a backing database tracking users in a static site.

In this step-by-step lab, you'll see how to integrate feature flags into a static site with a free version of Optimizely, Optimizely Rollouts

Pre-requisites

  • A modern web browser

Steps

Note: You can also follow along this video with the instructions below

  1. Create an Optimizely Rollouts Account
  2. Open up a blank codepen
  3. Create a static site by pasting in the following html:
<html>
  <head>
  </head>
  <body>
    <h1>Hello World</h1>
    <p id="subtitle">This is a subtitle</p>
  </body>
</html>
  1. Install the Optimizely SDK via the html script tag by placing the below code within the <head></head> tags of the html. This makes the SDK available at window.optimizelySdk:
    <script src="https://unpkg.com/@optimizely/optimizely-sdk@3.5/dist/optimizely.browser.umd.min.js"></script>
  1. Install the datafile for your project using the html script tag by placing the below code within the <head></head> tags of the html. Be sure to replace Your_SDK_Key with your SDK key from the Optimizely Application. This makes the SDK available at window.optimizelyDatafile:
    <script src="https://cdn.optimizely.com/datafiles/<Your_SDK_KEY>.json/tag.js"></script>

You can find your SDK Key in the Optimizely application by navigating to the far left 'Settings' > 'Environments' and copy the Development SDK Key value.

Screenshot

  1. Place the following script tag near the bottom of the html within the <html> tag, which will initialize the SDK:
  <script>
    var optimizelyClientInstance = window.optimizelySdk.createInstance({
      datafile: window.optimizelyDatafile,
    });
  </script>
  1. Create a feature flag named hello_world in the Optimizely UI. Navigate to 'Features > Create New Feature' and create a feature flag called 'hello_world':

Screenshot

  1. Implement the hello_world feature flag by placing the following code just below our initialization code above:
    var userId = 'user123';
    var enabled = optimizelyClientInstance.isFeatureEnabled('hello_world', userId);
    if (enabled) {
      document.querySelector("#subtitle").innerText = "Feature flag is ON!"
    } else {
      document.querySelector("#subtitle").innerText = "Feature flag is off"
    }
  1. Define a function to get or generate userIds and replace user123 with getOrGenerateUserId():
    function getOrGenerateUserId() {
      var userId;
      try {
        var storageKey = 'optimizely-userId'
        userId = window.localStorage.getItem(storageKey);
        if (!userId) {
          userId = String(Math.random());
          localStorage.setItem(storageKey, userId);
        }
      } catch {
        console.warn('[OPTIMIZELY] LocalStorage not available to store userId. Generating random userId each page load');
        userId = String(Math.random());
      }
      return userId;
    }

    var userId = getOrGenerateUserId();
  1. Now you can turn the feature flag on and off in the Optimizely UI and see the changes in your static site! 🎉

To turn on your feature flag:

  1. Navigate to Features
  2. Click on the 'hello_world' feature
  3. Change to the 'Development' environment to match the SDK Key we used
  4. Roll the feature out to ensure it is set to 100% for everyone
  5. Click Save to save your changes

Screenshot

After at least 10 seconds you can try and reload the codepen page to see the feature flag update has propagated to your application.

If you did not see the feature flag turn on, double check that you are not running any ad-blockers and check your code against a full code sample below (being sure to update the SDK key to your own SDK key):

<html>
  <head>
    <script src="https://unpkg.com/@optimizely/optimizely-sdk@3.5/dist/optimizely.browser.umd.min.js"></script>
    <script src="https://cdn.optimizely.com/datafiles/V7kE2tbrJaYGmtRF5W5QXf.json/tag.js"></script>
  </head>
  <body>
    <h1>Hello World</h1>
    <p id="subtitle">This is a subtitle</p>
  </body>
  <script>
    var optimizely = optimizelySdk.createInstance({
      datafile: window.optimizelyDatafile,
    });

    function getOrGenerateUserId() {
      var userId;
      try {
        var storageKey = 'optimizely-userId'
        userId = window.localStorage.getItem(storageKey);
        if (!userId) {
          userId = String(Math.random());
          localStorage.setItem(storageKey, userId);
        }
      } catch {
        console.warn('[OPTIMIZELY] LocalStorage not available to store userId. Generating random userId each page load');
        userId = String(Math.random());
      }
      return userId;
    }

    var userId = getOrGenerateUserId();
    var enabled = optimizelyClientInstance.isFeatureEnabled('hello_world', userId);
    if (enabled) {
      document.querySelector("#subtitle").innerText = "Feature flag is ON!"
    } else {
      document.querySelector("#subtitle").innerText = "Feature flag is off"
    }
  </script>
</html>
Additional links