Feature
Toggle
Driven
Development
Igal Steklov
data:image/s3,"s3://crabby-images/a3f2e/a3f2eaab13199515abe447296af514e44e049bce" alt=""
CEO & FED
Cheers
🍻
Good Morning
☕
data:image/s3,"s3://crabby-images/21f4b/21f4b5ae1e0f231b4b577b24ada289c34683a9ab" alt=""
data:image/s3,"s3://crabby-images/9d223/9d2233646c2f0042df448cc940fc2564fc1a8773" alt=""
data:image/s3,"s3://crabby-images/1f09e/1f09e1a6c1dec57ee60709b8c9dd4428c687b631" alt=""
data:image/s3,"s3://crabby-images/35525/3552576199e9d99eb35764256119b507cf8c57fa" alt=""
data:image/s3,"s3://crabby-images/b0b23/b0b23f68f02003559c8accd7885a2bf5c3619292" alt=""
data:image/s3,"s3://crabby-images/9163e/9163e41637b8056cc3594d692ba8d7ba03886b19" alt=""
data:image/s3,"s3://crabby-images/95285/952850d8f43f350349e1fa2fe08844c6b0858ccf" alt=""
data:image/s3,"s3://crabby-images/1f09e/1f09e1a6c1dec57ee60709b8c9dd4428c687b631" alt=""
data:image/s3,"s3://crabby-images/e2cb3/e2cb3fd3db8f6aabb49ecc82e67f3346ad252572" alt=""
data:image/s3,"s3://crabby-images/39c29/39c29c6c0cbdca112b48f4fb7504aa581850d629" alt=""
Deploy to Production?
How Often Do You
How stable is it?
TDD
Test Driven Development
CI/CD
Continuous Integration & Deployment
How do you control when code is exposed to users?
data:image/s3,"s3://crabby-images/f8a1d/f8a1d49aaae2492cbee474f90bd9829afa3a6040" alt=""
Feature Toggles
you need...
data:image/s3,"s3://crabby-images/f235a/f235a7385ea1645090a21ecff085cf5b5053c566" alt=""
Feature Toggles?
What are
data:image/s3,"s3://crabby-images/c7b99/c7b999b57c491deb4c8b04a8a676c481502d8f0a" alt=""
data:image/s3,"s3://crabby-images/6df8a/6df8a3d57fd1a262a8cf5342d0e2f31d00af8cf2" alt=""
Feature Toggles
(or Feature Flags)
import React, { Component } from 'react';
import featureToggles from 'services/featureToggles';
class ExampleComp extends Component {
render () {
return featureToggles.isEnabled('my_new_feature')
? <a href="...">Click me for new feature</a>
: <p>New feature is coming soon</p>
}
}
services/featureToggles.js
import bulletTrain from 'bullet-train-client';
featureToggles.isEnabled('my_feature') // true | false
import bulletTrain from 'bullet-train-client';
const config = {
environmentID: process.env.BULLET_TRAIN_ENV_ID
};
class FeatureToggles {
// Called once app loaded and user data retrieved
init(user) {
bulletTrain.init(config);
bulletTrain.identify(`${user.id}_${user.fullName}`);
}
// Called on logout from our app
destroy() {
bulletTrain.logout();
}
isEnabled(featureToggle) {
return bulletTrain.hasFeature(featureToggle);
}
}
export default new FeatureToggles();
[
{
"id":3000,
"feature":{
"id":818,
"name":"notes_list",
"created_date":"2019-05-25T15:22:26.058231Z",
"initial_value":null,
"description":null,
"default_enabled":true,
"type":"FLAG",
"project":325
},
"feature_state_value":null,
"enabled":true,
"environment":814,
"identity":null
},
{ ... },
{ ... }
]
~ curl 'https://api.bullet-train.io/api/v1/flags/' -H 'X-Environment-Key: BgwB3E6'
It's just a simple JSON
<body data-ft-new_design>
...
</body>
.button {
font-size: 12px;
}
body[data-ft-new_design] .button {
font-size: 18px;
}
Also in your CSS
.button {
font-size: 12px;
@include when-feature-enabled("new_design") {
// Compiles to body[data-ft-new_design] .button
font-size: 18px;
}
}
Or SASS...
New Dev
Workflow
Driven by Feature Toggles
1.
Start with creating a FEATURE TOGGLE
2.
Block entry points with conditions
import React, { Component } from 'react';
import { NavLink } from 'react-router-dom';
import featureToggles from 'services/featureToggles';
class Navigation extends Component {
render () {
return (
<nav>
<ul>
<li><NavLink to="/">Homepage</NavLink></li>
<li><NavLink to="/reports">Reports</NavLink></li>
{featureToggles.isEnabled('notes_list') &&
<li>
<NavLink to="/notes">
Notes
</NavLink>
</li>
}
</ul>
</nav>
);
}
}
import React, { Component } from 'react';
import history from 'utils/history';
import featureToggles from 'services/featureToggles';
class NotesList extends Component {
//UNSAFE_componentWillMount
constructor(props) {
super(props);
if (!featureToggles.isEnabled('notes_list')) {
history.push('/error-page/404');
return;
}
}
render () {
return null;
}
}
3.
Work on your feature,
deploy to Production
import React, { Component } from 'react';
import history from 'utils/history';
import featureToggles from 'services/featureToggles';
class NotesList extends Component {
//UNSAFE_componentWillMount
constructor(props) {
super(props);
if (!featureToggles.isEnabled('notes_list')) {
history.push('/error-page/404');
return;
}
}
render () {
return (
<div>
Notes list component is under development,
so no user reach current component, yay 🎉.
</div>
);
}
}
data:image/s3,"s3://crabby-images/7c5fb/7c5fb42039e2e567820de1ec2867764694e4bb89" alt=""
4.
Share work progress with PMs & QA
data:image/s3,"s3://crabby-images/6a42c/6a42c544e9806b97389ffa3463f57f2b32e891d5" alt=""
5.
Controlled Release of Features,
Bug Fixes
or Infra Refactoring
AB Test or just Release
6.
Monitor Production: Biz & Tech KPIs
data:image/s3,"s3://crabby-images/240d2/240d2d20c2673ea0a69c914c643c55996874051d" alt=""
data:image/s3,"s3://crabby-images/240d2/240d2d20c2673ea0a69c914c643c55996874051d" alt=""
data:image/s3,"s3://crabby-images/5f049/5f0495d2f8c19a82184c9c7c46093500580a1295" alt=""
data:image/s3,"s3://crabby-images/7b92c/7b92cd1886ddbc1935229726f3d0c57a3e33cc78" alt=""
data:image/s3,"s3://crabby-images/fe74a/fe74a67aa926d4abc3f60da4cb6ad8b65125b282" alt=""
It's OK to...
data:image/s3,"s3://crabby-images/006db/006dbcd17d9b4fde9e88dd9295d32d5cff68bcaa" alt=""
What's so Great About FTs?
-
Better collaboration with PMs, QA and other stakeholders.
-
Ability to gradually rollout sensitive changes.
- Quickly resolve Production issues by closing FT (aka Kill Switch).
- Verify code works on Production, daily.
- No need for staging if you have tests – auto-deploy to Prod when CI passes.
- Better code – everything is "Production Code"
Tips
1.
Feature Toggle
Any Non-trivial
Code Change
- Tiny feature? YES!
- Bug fix? YES!
- Internal infra change? YES!
2.
It's OK to FT
Entire Components
import React, { Component, Fragment } from 'react';
import { Switch, Route } from 'react-router-dom';
import featureToggles from 'services/featureToggles';
import DashboardPage from 'containers/DashboardPage';
import ListPage from 'containers/ListPage';
import ListPageV2 from 'containers/ListPageV2';
export default class App extends Component {
render() {
return (
<main>
<Switch>
<Route path="/" component={DashboardPage}/>
<Route path="/list" render={() => featureToggles.isEnabled('list_page_v2')
? <ListPageV2/>
: <ListPage/>
}/>
</Switch>
</main>
);
}
}
3.
Remember the Backend APIs
import axios from 'axios';
import featureToggles from 'services/featureToggles';
export function listNotes() {
const apiVersion = featureToggles.isEnabled('notes_api_v2')
? 'v2'
: 'v1';
return axios.get(`api/${apiVersion}/notes/`);
}
4.
Plan Feature Toggle so that they can be turned off if needed
class NotesList extends Component {
render () {
const { notes = [] } = this.props;
if (featureToggles.isEnabled('notes_list')) {
return (
<ul>
{notes.map(note =>
<li key={note.id}>{note.text}</li>
)}
</ul>
);
}
return (<p>Notes are coming soon</p>);
}
}
class NotesList extends Component {
render () {
const { notes = [] } = this.props;
const isNotesExist = notes.length > 0;
if (isNotesExist || featureToggles.isEnabled('notes_list')) {
return (
<ul>
{notes.map(note =>
<li key={note.id}>{note.text}</li>
)}
</ul>
);
}
return (<p>Notes are coming soon</p>);
}
}
5.
Cleanup your Feature Toggles after release
class NotesList extends Component {
render () {
const { notes = [] } = this.props;
const isNotesExist = notes.length > 0;
if (isNotesExist || featureToggles.isEnabled('notes_list')) {
return (
<ul>
{notes.map(note =>
<li key={note.id}>{note.text}</li>
)}
</ul>
);
}
return (<p>Notes are coming soon</p>);
}
}
class NotesList extends Component {
render () {
const { notes = [] } = this.props;
return (
<ul>
{notes.map(note =>
<li key={note.id}>{note.text}</li>
)}
</ul>
);
}
}
Step
1
Step
2
Step
3
-
Easy to Start
-
Safer Deploys
-
Stable Production
-
PMs & QA ❤️ You
-
Infra for A/B Tests
Thank You :)
Igal Steklov
igal@webiya.co.il
+972-54-6490123
data:image/s3,"s3://crabby-images/a3f2e/a3f2eaab13199515abe447296af514e44e049bce" alt=""
Slides at: tiny.cc/FTDD
Questions?
data:image/s3,"s3://crabby-images/8bb60/8bb606f3bad0459ca40fee95734760ec6ce895fd" alt=""
FTDD - Feature Toggle Driven Development
By Igal Steklov