6 min read

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);
2
3if(myFlag) {
4 console.log('flag is true');
5}
6
7if(!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.

Architecture overview

To make it easy to support multiple feature flag providers, we split the code into two main components:

  1. 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 and getReplacer. The getMatcher method is responsible for identifying the correct CallExpression, while the getReplacer method returns the appropriate replacement node.
  2. 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

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;
3
4console.log(var1);
5
6if(var1) {
7 console.log('var2 is true');
8}

After

1const var3 = true;
2
3console.log(true);
4
5if(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!!!false
2!!"string"
3""

After

1true
2true
3false

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 === true
21 === true
3```

After

1true
2false

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 && x
2false && x

After

1x
2false

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}
4
5if(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 expression
5 if (name !== "useFlag") {
6 return null;
7 }
8
9 const args = ce.getArguments();
10
11 const keyArg = args[0];
12
13 // only match the useFlag with given keyName
14 if(!Node.isStringLiteral(keyArg) || keyArg.getLiteralText() !== keyName) {
15 return;
16 }
17
18 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 name
22 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.

Join our community of code migration experts.

Slack community

Stay in the loop, subscribe to our newsletter.