Introduction
When a feature flag becomes stale, developers need to manually remove it from the codebase. This process involves more than just replacing the feature flag SDK call with a static value; it also requires a thorough refactoring of the code to ensure it remains clean after the change. The task can be intricate, especially in larger codebases where feature flags might be deeply integrated into various components and logic flows.
To illustrate this, let's look at an example of such a refactoring process. The code snippet from the before
state, which includes conditional logic based on the feature flag, can be significantly simplified in the after
state once the flag is removed and the code is refactored.
Before
1const myFlag = useVariableValue('feature-name', false);23if(myFlag) {4 console.log('flag is true');5}67if(!myFlag) {8 console.log('flag is false');9}
After
1console.log('flag is false');
To automate this process, we built codemods that autonomously handles both the removal of feature flags and the subsequent refactoring of the code. These codemods significantly streamline the cleanup process by automating the identification and replacement of feature flag SDK calls with static values, as well as performing the necessary code refactoring to maintain code quality.
Currently we added codemods for multiple popular feature flag providers, such as DevCycle and Statsig, but in future we can easily add support for many other providers.
Architecture overview
To make it easy to support multiple feature flag providers, we split the code into two main components:
- Feature Flag SDK Specific Code (Provider): This component encapsulates SDK-specific knowledge, such as SDK method names, import names, and method signatures. Essentially, it understands how to match SDK calls and determine the appropriate values to replace them with. For each SDK, we implement two methods:
getMatcher
andgetReplacer
. ThegetMatcher
method is responsible for identifying the correctCallExpression
, while thegetReplacer
method returns the appropriate replacement node. - Unused Code Removal: This component handles the removal of code that is no longer needed due to the feature flag cleanup. It is designed to be generic, allowing easily add new providers. Unused code removal includes multiple atomic steps that are executed one by one. This diagram shows the top level architecture of the codemod.
Modular & customizable design for feature flag cleanup codemod
Unused code removal
Let's take a closer look at each step of code removal component.
Simplify Object Properties
This step involves removing redundant member expressions that remain after replacing the feature flag SDK call with an object literal.
Before
1({ value: 'theValue' }).value
After
1'theValue'
Update References
If flag is used as the part of variable declaration, we want to replace the variable associated with the feature flag as well.
Before
1const var1 = useFlag(user, 'simple-case', true);2const var2 = var1;34console.log(var1);56if(var1) {7 console.log('var2 is true');8}
After
1const var3 = true;23console.log(true);45if(true) {6 console.log('var2 is true');7}
Simplify Unary Expressions
This stepis aiming for simplifying unary expression with exclamation mark:
This transformation is only applied to the literals.
Before
1!!!false2!!"string"3""
After
1true2true3false
Simplify Binary Expressions
This step replaces useless binary expressions. When left
and right
values are literals we can evaluate such expressions and replace the expression with its result.
Before
1true === true21 === true3```
After
1true2false
Simplify Logical Expressions
This step is similar with previous, here we try to simplify useless logical expressions. In javascript logical expression with ampersand returns the first falsy value, if we see expression that starts with true
or truthy literal, we can just remove the literal. When expression starts with the falsy literal, replace the expression with this literal.
Before
1true && x2false && x
After
1x2false
Remove Unnecessary Parenthesis
Sometimes after simplifying nested expression we have useless parenthesis, this step removes them.
Simplify If Statements
This step simply unwraps the if statement if condition is truthy, or removes it if its falsy. On this step we eliminate useless branching.
Before
1if(true) {2 console.log('truthy')3}45if(false) {6 console.log('truthy')7}
After
1console.log('truthy')
Simplify Conditional Expressions
This step is similar as previous, but for ternary expressions:
Before
1true ? 'truthy' : 'falsy'
After
1'truthy'
To handle nesting, we run all steps multiple times.
Adding support for custom feature flag provider
Adding support for a new provider is quite straightforward. We simply need to define the getMatcher
and getReplacer
methods.
getMatcher
accepts the feature flag key name and return the matcher function that accepts the CallExpression
and should return the SDK method name or undefined
if we want to skip the node.
getReplacer
accepts the key
, type
and the value
that should be used for replacement of the feature flag. Additionally it accepts the SDK method name, because we can have different replacements for different SDK method calls.
getReplacer
should return the node that will be stringified and used as replacement for feature flag call in the main part of the codemod.
Here is simple example of the custom provider that matches method with the name useFlag
and given feature flag key name and replaces it with the literal.
1const SimpleProvider = {2 getMatcher: (keyName: string) => (ce: CallExpression) => {3 const name = getCEExpressionName(ce);4 // only match useFlag call expression5 if (name !== "useFlag") {6 return null;7 }89 const args = ce.getArguments();1011 const keyArg = args[0];1213 // only match the useFlag with given keyName14 if(!Node.isStringLiteral(keyArg) || keyArg.getLiteralText() !== keyName) {15 return;16 }1718 return { name };19 },20 // replacer accepts key, type (because we need to cast the value that is passed as string)21 // value and the matched callExpressions name22 getReplacer: (23 key: string,24 type: VariableType,25 value: VariableValue,26 { name }: string,27 ) => {28 return buildLiteral(type, value);29 },30};
Conclusion
Cleaning up feature flags is a major hassle; developers in large teams waste months on manual work. Fortunately, this can be solved with a customized codemod, a practice established in big tech for years. We've developed a modular and customizable codemod so that anyone can easily adapt it to their needs. To further customize our codemods, you can use Codemod Studio for AI assistance or contact us to build professional-grade codemods tailored to your specific requirements. Codemod automates repetitive code maintenance tasks for feature flags, reducing friction for experimentation and leading to more and better features for end users.