<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Krishan Sharma</title>
    <description>The latest articles on DEV Community by Krishan Sharma (@krishan_sharma_561a52817e).</description>
    <link>https://dev.to/krishan_sharma_561a52817e</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3956537%2F877d2c42-8569-4852-a8cf-d7891425c664.png</url>
      <title>DEV Community: Krishan Sharma</title>
      <link>https://dev.to/krishan_sharma_561a52817e</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/krishan_sharma_561a52817e"/>
    <language>en</language>
    <item>
      <title>Your Node.js Codebase Has Flag Debt. Here's How to Find It in 30 Seconds.</title>
      <dc:creator>Krishan Sharma</dc:creator>
      <pubDate>Tue, 02 Jun 2026 15:54:10 +0000</pubDate>
      <link>https://dev.to/krishan_sharma_561a52817e/your-nodejs-codebase-has-flag-debt-heres-how-to-find-it-in-30-seconds-1pg1</link>
      <guid>https://dev.to/krishan_sharma_561a52817e/your-nodejs-codebase-has-flag-debt-heres-how-to-find-it-in-30-seconds-1pg1</guid>
      <description>&lt;p&gt;Most teams don't know how many feature flags are in their codebase.&lt;/p&gt;

&lt;p&gt;They know they have &lt;em&gt;some&lt;/em&gt;. They think they cleaned up &lt;em&gt;most&lt;/em&gt;. They're not sure about the rest.&lt;/p&gt;

&lt;p&gt;One command changes that:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint audit ./src
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;No API key. No credentials. No dashboard to sign up for. Just your source code and an honest answer.&lt;/p&gt;




&lt;h2&gt;
  
  
  The Problem Nobody Talks About
&lt;/h2&gt;

&lt;p&gt;According to LaunchDarkly's best practices, most release flags should live for only days to weeks — yet many remain in codebases for months or years.&lt;/p&gt;

&lt;p&gt;That's not a LaunchDarkly problem. It's a universal one.&lt;/p&gt;

&lt;p&gt;Flags accumulate because adding one is fast and removing one is work. You ship the feature, move on, and the flag stays. Six months later a new engineer asks "is this safe to delete?" and nobody knows. So it stays another six months.&lt;/p&gt;

&lt;p&gt;Stale flags make code "more complex and harder to maintain," as developers spend extra time navigating obsolete conditionals. Unused toggles may degrade performance or even inadvertently expose features or data.&lt;/p&gt;

&lt;p&gt;And here's the part that stings: developers spend 33–42% of their time dealing with technical debt and maintenance. Feature flag debt is a quiet contributor to that number.&lt;/p&gt;




&lt;h2&gt;
  
  
  What "Flag Debt" Actually Looks Like
&lt;/h2&gt;

&lt;p&gt;Here's a real checkout service. Nothing exotic — a Node.js backend with LaunchDarkly calls spread across five files.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// checkout.ts&lt;/span&gt;
&lt;span class="k"&gt;export&lt;/span&gt; &lt;span class="k"&gt;async&lt;/span&gt; &lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;isCheckoutV2Enabled&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;user&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;User&lt;/span&gt;&lt;span class="p"&gt;):&lt;/span&gt; &lt;span class="nb"&gt;Promise&lt;/span&gt;&lt;span class="o"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nx"&gt;boolean&lt;/span&gt;&lt;span class="o"&gt;&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;ctx&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;targetingKey&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;user&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;email&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;user&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;email&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;plan&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;user&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;plan&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;boolVariation&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;checkout-v2&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ctx&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;

&lt;span class="c1"&gt;// discounts.ts&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;flagKey&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s2"&gt;`discount-&lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;experimentName&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;enabled&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;boolVariation&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;flagKey&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ctx&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;

&lt;span class="c1"&gt;// analytics.ts&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;state&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;allFlagsState&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;ctx&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Three different patterns. Three very different levels of risk.&lt;/p&gt;

&lt;p&gt;The first one is fine — static key, known type, safely removable.&lt;/p&gt;

&lt;p&gt;The second one is a problem — the key is a template literal. You can't statically know which flag it evaluates at runtime.&lt;/p&gt;

&lt;p&gt;The third one is a migration blocker — &lt;code&gt;allFlagsState&lt;/code&gt; has no OpenFeature equivalent. It requires an architecture decision before you touch it.&lt;/p&gt;

&lt;p&gt;Most teams treat all three the same. They shouldn't.&lt;/p&gt;




&lt;h2&gt;
  
  
  The Audit Command
&lt;/h2&gt;

&lt;p&gt;FlagLint v0.6.0 ships with a new command: &lt;code&gt;flaglint audit&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;It scans your codebase, classifies every flag call by risk level, and tells you exactly what you're dealing with — before you touch anything.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint@latest audit ./src
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Running it against the enterprise checkout service above produces:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;✓ Audit complete: 13 flags — 3 high risk, 10 medium risk, 0 low risk
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;With the full table:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Flag Key&lt;/th&gt;
&lt;th&gt;Risk&lt;/th&gt;
&lt;th&gt;Usages&lt;/th&gt;
&lt;th&gt;Reasons&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;dynamic&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;🔴 High&lt;/td&gt;
&lt;td&gt;7&lt;/td&gt;
&lt;td&gt;dynamic key&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;checkout-experiment&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;🔴 High&lt;/td&gt;
&lt;td&gt;1&lt;/td&gt;
&lt;td&gt;detail evaluation&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;*&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;🔴 High&lt;/td&gt;
&lt;td&gt;1&lt;/td&gt;
&lt;td&gt;bulk call&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;checkout-v2&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;🟡 Medium&lt;/td&gt;
&lt;td&gt;1&lt;/td&gt;
&lt;td&gt;safely automatable&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;payment-provider&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;🟡 Medium&lt;/td&gt;
&lt;td&gt;1&lt;/td&gt;
&lt;td&gt;safely automatable&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;discount-config&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;🟡 Medium&lt;/td&gt;
&lt;td&gt;1&lt;/td&gt;
&lt;td&gt;safely automatable, json variation&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;...&lt;/td&gt;
&lt;td&gt;&lt;/td&gt;
&lt;td&gt;&lt;/td&gt;
&lt;td&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;&lt;strong&gt;High risk&lt;/strong&gt; means the call needs manual review before anything happens:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Dynamic key&lt;/strong&gt; — the flag key is a variable or template literal. FlagLint can't tell which flag you're evaluating.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Detail evaluation&lt;/strong&gt; — &lt;code&gt;boolVariationDetail&lt;/code&gt; returns metadata with no direct OpenFeature equivalent.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Bulk call&lt;/strong&gt; — &lt;code&gt;allFlagsState&lt;/code&gt; is an architecture decision, not a line-by-line migration.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;Medium risk&lt;/strong&gt; means FlagLint can automate the migration safely, but the flag is still a direct vendor call that will need to move eventually.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Low risk&lt;/strong&gt; means the flag is already migrated or has no debt signals.&lt;/p&gt;




&lt;h2&gt;
  
  
  Three Output Formats
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="c"&gt;# Markdown — good for PRs and docs&lt;/span&gt;
npx flaglint audit ./src &lt;span class="nt"&gt;--format&lt;/span&gt; markdown

&lt;span class="c"&gt;# JSON — good for CI pipelines and dashboards&lt;/span&gt;
npx flaglint audit ./src &lt;span class="nt"&gt;--format&lt;/span&gt; json &lt;span class="nt"&gt;--output&lt;/span&gt; audit.json

&lt;span class="c"&gt;# HTML — shareable report for engineering reviews&lt;/span&gt;
npx flaglint audit ./src &lt;span class="nt"&gt;--format&lt;/span&gt; html &lt;span class="nt"&gt;--output&lt;/span&gt; flag-debt.html
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The HTML report is the one worth sharing with your team or your manager. It's a single self-contained file — no server needed, just open it in a browser. Drop it in a PR, a Jira ticket, or an email. It shows exactly which flags need attention and why.&lt;/p&gt;




&lt;h2&gt;
  
  
  From Audit to Action
&lt;/h2&gt;

&lt;p&gt;The audit is informational. It doesn't touch your code.&lt;/p&gt;

&lt;p&gt;Once you've seen your report, FlagLint has two more commands for when you're ready to act:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Preview the migration:&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint migrate ./src &lt;span class="nt"&gt;--dry-run&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This shows before/after diffs for every safely automatable call — the Medium risk flags from your audit. Nothing is written to disk.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight diff"&gt;&lt;code&gt;&lt;span class="gd"&gt;--- checkout.ts
&lt;/span&gt;&lt;span class="gi"&gt;+++ checkout.ts
&lt;/span&gt;&lt;span class="gd"&gt;-  return ldClient.boolVariation("checkout-v2", ctx, false);
&lt;/span&gt;&lt;span class="gi"&gt;+  return openFeatureClient.getBooleanValue("checkout-v2", false, ctx);
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Note the argument order. LaunchDarkly is &lt;code&gt;(flagKey, context, fallback)&lt;/code&gt;. OpenFeature is &lt;code&gt;(flagKey, fallback, context)&lt;/code&gt;. A manual find-and-replace migration silently swaps them — that's a production bug waiting to happen. FlagLint gets it right because it uses AST analysis, not text matching.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Apply the migration:&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint migrate ./src &lt;span class="nt"&gt;--apply&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Rewrites only the calls that are proven safe. Dynamic keys, detail methods, and bulk calls are skipped and reported for manual review. Won't run on a dirty git tree. Idempotent — safe to run twice.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Lock the boundary in CI:&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint validate ./src &lt;span class="nt"&gt;--no-direct-launchdarkly&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Exits 1 if any direct LaunchDarkly evaluation call appears in your codebase. Add this to your GitHub Actions workflow and the migration can't silently reverse.&lt;/p&gt;




&lt;h2&gt;
  
  
  Why No API Key?
&lt;/h2&gt;

&lt;p&gt;Every major feature flag platform has a tool that connects to their API to show you flag health. Those tools are useful, but they only see flags that exist in their own system. They won't show you a LaunchDarkly flag if you're asking a Statsig dashboard.&lt;/p&gt;

&lt;p&gt;More importantly — they're built to keep you in their platform. No vendor builds the exit ramp for their own tool.&lt;/p&gt;

&lt;p&gt;FlagLint only looks at your source code. It doesn't know what's in your LaunchDarkly dashboard. It doesn't need to. The question it answers is: &lt;em&gt;what is your code actually doing right now?&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;That's a different question, and it has a different answer.&lt;/p&gt;




&lt;h2&gt;
  
  
  The Full Workflow
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="c"&gt;# Step 1: Understand what you have&lt;/span&gt;
npx flaglint audit ./src &lt;span class="nt"&gt;--format&lt;/span&gt; html &lt;span class="nt"&gt;--output&lt;/span&gt; flag-debt.html

&lt;span class="c"&gt;# Step 2: Preview safe migrations&lt;/span&gt;
npx flaglint migrate ./src &lt;span class="nt"&gt;--dry-run&lt;/span&gt;

&lt;span class="c"&gt;# Step 3: Apply the safe ones&lt;/span&gt;
npx flaglint migrate ./src &lt;span class="nt"&gt;--apply&lt;/span&gt;

&lt;span class="c"&gt;# Step 4: Lock it in CI&lt;/span&gt;
npx flaglint validate ./src &lt;span class="nt"&gt;--no-direct-launchdarkly&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Four commands. Covers the full lifecycle from "what do we have?" to "this can never regress."&lt;/p&gt;




&lt;h2&gt;
  
  
  Try It
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint@latest audit ./src
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;No install required. Works on any Node.js 20+ project using the LaunchDarkly Node.js server SDK (both &lt;code&gt;launchdarkly-node-server-sdk&lt;/code&gt; and &lt;code&gt;@launchdarkly/node-server-sdk&lt;/code&gt;).&lt;/p&gt;

&lt;p&gt;→ &lt;a href="https://github.com/flaglint/flaglint" rel="noopener noreferrer"&gt;GitHub&lt;/a&gt; — MIT licensed, open source&lt;br&gt;&lt;br&gt;
→ &lt;a href="https://www.npmjs.com/package/flaglint" rel="noopener noreferrer"&gt;npm&lt;/a&gt;&lt;br&gt;&lt;br&gt;
→ &lt;a href="https://flaglint.dev/docs/cli/audit" rel="noopener noreferrer"&gt;Docs&lt;/a&gt;&lt;/p&gt;




&lt;p&gt;&lt;em&gt;FlagLint is a vendor-neutral CLI for LaunchDarkly → OpenFeature migrations. v0.6.0 adds flag debt auditing. &lt;a href="https://github.com/flaglint/flaglint/blob/main/CHANGELOG.md" rel="noopener noreferrer"&gt;Read the changelog →&lt;/a&gt;&lt;/em&gt;&lt;/p&gt;




&lt;p&gt;&lt;strong&gt;Tags:&lt;/strong&gt; &lt;code&gt;node&lt;/code&gt; &lt;code&gt;devops&lt;/code&gt; &lt;code&gt;javascript&lt;/code&gt; &lt;code&gt;typescript&lt;/code&gt; &lt;code&gt;openfeature&lt;/code&gt; &lt;code&gt;launchdarkly&lt;/code&gt; &lt;code&gt;featureflags&lt;/code&gt; &lt;code&gt;technicaldebt&lt;/code&gt; &lt;code&gt;opensource&lt;/code&gt; &lt;code&gt;featureflags&lt;/code&gt;&lt;/p&gt;

</description>
    </item>
    <item>
      <title>Standardizing Feature Flags Is Easy to Agree On. Migrating Safely Is the Hard Part.</title>
      <dc:creator>Krishan Sharma</dc:creator>
      <pubDate>Thu, 28 May 2026 12:02:08 +0000</pubDate>
      <link>https://dev.to/krishan_sharma_561a52817e/standardizing-feature-flags-is-easy-to-agree-on-migrating-safely-is-the-hard-part-1k39</link>
      <guid>https://dev.to/krishan_sharma_561a52817e/standardizing-feature-flags-is-easy-to-agree-on-migrating-safely-is-the-hard-part-1k39</guid>
      <description>&lt;h2&gt;
  
  
  Why I built FlagLint, an open-source CLI for moving direct LaunchDarkly Node.js usage behind an OpenFeature boundary
&lt;/h2&gt;

&lt;p&gt;Feature flags usually begin as a simple engineering decision.&lt;/p&gt;

&lt;p&gt;A team needs to release gradually. A developer adds a flag. The application evaluates it through the provider SDK. The rollout succeeds.&lt;/p&gt;

&lt;p&gt;Then the pattern repeats.&lt;/p&gt;

&lt;p&gt;One feature becomes ten. One service becomes dozens. Over time, application code starts to accumulate direct calls to a provider-specific API:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;enabled&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;boolVariation&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
  &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;checkout-v2&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="nx"&gt;context&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="kc"&gt;false&lt;/span&gt;
&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;There is nothing wrong with that call by itself. LaunchDarkly is doing exactly what it is supposed to do: providing feature-flag management and evaluation.&lt;/p&gt;

&lt;p&gt;The problem appears later, when a platform team wants a consistent application-facing abstraction across services.&lt;/p&gt;

&lt;p&gt;Maybe the organization wants to standardize on &lt;a href="https://openfeature.dev/" rel="noopener noreferrer"&gt;OpenFeature&lt;/a&gt;, the CNCF-incubating, vendor-agnostic feature-flag API. Maybe teams want feature-flag instrumentation, governance, or shared patterns implemented once at a platform boundary instead of separately inside each service.&lt;/p&gt;

&lt;p&gt;At that point, the difficult question is no longer:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Should we standardize feature-flag evaluation?&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;The difficult questions are:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Where are all the direct SDK calls? Which ones can be migrated safely? Which ones require human review? And how do we stop new direct calls from returning after migration?&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;That is the problem I built &lt;strong&gt;&lt;a href="https://flaglint.dev/" rel="noopener noreferrer"&gt;FlagLint&lt;/a&gt;&lt;/strong&gt; to solve.&lt;/p&gt;




&lt;h2&gt;
  
  
  OpenFeature Does Not Mean Replacing LaunchDarkly
&lt;/h2&gt;

&lt;p&gt;A common misconception in migration conversations is that introducing OpenFeature means replacing the current feature-flag provider.&lt;/p&gt;

&lt;p&gt;It does not.&lt;/p&gt;

&lt;p&gt;OpenFeature standardizes the evaluation API that application code uses. A provider still performs the actual evaluation. LaunchDarkly offers an official OpenFeature provider for its Node.js server-side SDK, so a Node.js service can continue using LaunchDarkly as its feature-flag provider while application code evaluates flags through OpenFeature.&lt;/p&gt;

&lt;p&gt;Conceptually, the change looks like this:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Direct provider-specific application usage&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;boolVariation&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;checkout-v2&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;context&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;becoming:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Standard application-facing evaluation API&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;openFeatureClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;getBooleanValue&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;checkout-v2&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;context&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;LaunchDarkly remains behind the provider boundary. The application code moves toward a standard interface.&lt;/p&gt;

&lt;p&gt;That distinction matters. Platform standardization should not require an unnecessary provider replacement project.&lt;/p&gt;




&lt;h2&gt;
  
  
  Why This Migration Is More Dangerous Than It Looks
&lt;/h2&gt;

&lt;p&gt;At first glance, this appears to be a simple codemod problem: find a method call and rename it.&lt;/p&gt;

&lt;p&gt;But feature-flag migrations carry runtime behavior. An incorrect transformation can change what users see in production.&lt;/p&gt;

&lt;p&gt;For example, the LaunchDarkly and OpenFeature method signatures differ in argument order:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// LaunchDarkly&lt;/span&gt;
&lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;boolVariation&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;key&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;context&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;fallback&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

&lt;span class="c1"&gt;// OpenFeature&lt;/span&gt;
&lt;span class="nx"&gt;openFeatureClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;getBooleanValue&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;key&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;fallback&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;context&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;A careless rewrite can silently swap the fallback value and context.&lt;/p&gt;

&lt;p&gt;There are additional complications:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;a flag key may be a dynamic expression rather than a static string;&lt;/li&gt;
&lt;li&gt;a call may request evaluation details rather than only a flag value;&lt;/li&gt;
&lt;li&gt;code may use bulk flag-state APIs;&lt;/li&gt;
&lt;li&gt;a fallback type may not be statically clear;&lt;/li&gt;
&lt;li&gt;a service may import a shared platform-owned OpenFeature client using a local alias;&lt;/li&gt;
&lt;li&gt;the provider bootstrap file may legitimately reference LaunchDarkly directly;&lt;/li&gt;
&lt;li&gt;asynchronous behavior must be preserved exactly.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;In a migration involving feature flags, “we rewrote most of it automatically” is not enough. The important question is whether the automation knows when &lt;strong&gt;not&lt;/strong&gt; to rewrite code.&lt;/p&gt;




&lt;h2&gt;
  
  
  The Design Principle Behind FlagLint: Conservative Automation
&lt;/h2&gt;

&lt;p&gt;FlagLint is an open-source CLI focused on a specific workflow:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Help Node.js teams inventory direct LaunchDarkly server SDK evaluations, migrate only provably safe call sites to OpenFeature, and enforce the new boundary in CI.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;Its scope is deliberately narrow today. FlagLint supports JavaScript and TypeScript code using the LaunchDarkly Node.js server-side SDK package forms:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;code&gt;@launchdarkly/node-server-sdk&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;code&gt;launchdarkly-node-server-sdk&lt;/code&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;It is not trying to claim every language, every SDK, or every feature-flag lifecycle problem.&lt;/p&gt;

&lt;p&gt;FlagLint uses AST-based analysis to identify direct LaunchDarkly Node.js evaluation calls. It can automatically transform typed static evaluations only when the important parts are explicit:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;the flag key;&lt;/li&gt;
&lt;li&gt;the fallback value and type;&lt;/li&gt;
&lt;li&gt;the evaluation context;&lt;/li&gt;
&lt;li&gt;a proven OpenFeature client binding.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;For example:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Before&lt;/span&gt;
&lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;ldClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;boolVariation&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;checkout-v2&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ctx&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;can safely become:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// After&lt;/span&gt;
&lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;openFeatureClient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;getBooleanValue&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;checkout-v2&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ctx&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;But FlagLint does &lt;strong&gt;not&lt;/strong&gt; automatically rewrite uncertain patterns. Dynamic keys, detail evaluation methods, bulk calls, unknown fallbacks, browser or React SDK usage, and ambiguous bindings remain visible for human review.&lt;/p&gt;

&lt;p&gt;This is not a limitation to hide. It is the safety model.&lt;/p&gt;

&lt;p&gt;A migration tool should automate the obvious cases and make uncertainty impossible to ignore.&lt;/p&gt;




&lt;h2&gt;
  
  
  The Workflow: Scan, Migrate, Enforce
&lt;/h2&gt;

&lt;p&gt;FlagLint is designed around three steps.&lt;/p&gt;

&lt;h3&gt;
  
  
  1. Inventory direct SDK coupling
&lt;/h3&gt;

&lt;p&gt;Before a migration starts, teams need to understand what exists in the codebase:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint scan ./src
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The scan identifies direct LaunchDarkly Node.js server SDK evaluation calls, their files, flag keys, call types, and patterns that need manual review. Reports can be emitted in formats including Markdown, JSON, HTML, and SARIF.&lt;/p&gt;

&lt;p&gt;This is useful for both developers and platform teams. Before changing code, you can see whether a service is mostly straightforward or full of patterns that require careful migration planning.&lt;/p&gt;

&lt;h3&gt;
  
  
  2. Preview and apply safe transformations
&lt;/h3&gt;

&lt;p&gt;Next, teams can generate a migration plan and inspect diffs before touching source code:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint migrate ./src &lt;span class="nt"&gt;--dry-run&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;FlagLint generates reviewable before/after diffs. When a file already contains a proven OpenFeature client binding, including an approved imported shared client binding, the preview uses that exact binding.&lt;/p&gt;

&lt;p&gt;For example, a service may already import a platform-owned client with an alias:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;openFeatureClient&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="nx"&gt;flags&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;../platform/feature-flags.js&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;FlagLint preserves that architecture and previews the migration using &lt;code&gt;flags.getBooleanValue(...)&lt;/code&gt; rather than inventing a new local client.&lt;/p&gt;

&lt;p&gt;When ready, safe transformations can be applied:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint migrate ./src &lt;span class="nt"&gt;--apply&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The apply mode is guarded. It requires a proven OpenFeature client binding, refuses to write into a dirty Git working tree unless explicitly overridden, does not insert provider bootstrap setup automatically, and does not rewrite uncertain patterns.&lt;/p&gt;

&lt;h3&gt;
  
  
  3. Prevent the migration from reversing
&lt;/h3&gt;

&lt;p&gt;Migration is not complete if new direct provider SDK calls can quietly appear in future pull requests.&lt;/p&gt;

&lt;p&gt;Once a service has completed the boundary migration, FlagLint can enforce it in CI:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint validate ./src &lt;span class="nt"&gt;--no-direct-launchdarkly&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Validation can emit SARIF findings so direct LaunchDarkly policy violations can appear as annotations in code scanning and pull-request review flows.&lt;/p&gt;

&lt;p&gt;This turns a one-time refactor into an enforceable engineering standard.&lt;/p&gt;




&lt;h2&gt;
  
  
  An Important Boundary: FlagLint Is Not a Stale-Flag Platform
&lt;/h2&gt;

&lt;p&gt;There is a broader category of feature-flag debt: flags that are fully rolled out, inactive, unused at runtime, or ready for deletion in the provider platform.&lt;/p&gt;

&lt;p&gt;That is an important problem, but it requires lifecycle and runtime information that source code alone cannot prove.&lt;/p&gt;

&lt;p&gt;FlagLint does not currently claim to delete stale flags, inspect production evaluations, replace LaunchDarkly lifecycle tooling, or reduce feature-flag billing.&lt;/p&gt;

&lt;p&gt;Its current problem is more precise:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Where is application code directly coupled to the LaunchDarkly Node.js SDK, what can move safely behind OpenFeature, and can we enforce that boundary afterward?&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;That focus is intentional. Tools earn trust by being accurate about what they know and what they do not know.&lt;/p&gt;




&lt;h2&gt;
  
  
  What I Learned Building the First Release
&lt;/h2&gt;

&lt;p&gt;The technical work was only one part of the project. The harder part was defining safe behavior.&lt;/p&gt;

&lt;p&gt;A migration CLI for application infrastructure cannot be aggressive by default. It needs to be explainable. Every automated rewrite should be reviewable. Every skipped pattern should make sense to the engineer reading the report.&lt;/p&gt;

&lt;p&gt;During post-release validation of FlagLint v0.5.0, I found a good example of why that matters. A dry-run using a proven aliased OpenFeature client binding correctly previewed a safe transformation, but the CLI still printed global guidance implying provider setup was required. The transformation itself was correct, but the message was contradictory.&lt;/p&gt;

&lt;p&gt;That mattered because trust in migration tools is not only about whether they edit code correctly. It is also about whether their explanations match their behavior.&lt;/p&gt;

&lt;p&gt;In v0.5.1, I fixed that messaging so proven bindings are described accurately, missing bindings still receive setup guidance, and mixed cases clearly scope guidance only to diffs that need it.&lt;/p&gt;

&lt;p&gt;No safety boundary was weakened to make the tool appear more capable.&lt;/p&gt;

&lt;p&gt;That is the standard I want FlagLint to maintain as it grows.&lt;/p&gt;




&lt;h2&gt;
  
  
  What Comes Next
&lt;/h2&gt;

&lt;p&gt;The next challenge is not just transforming code. It is helping real teams adopt the boundary incrementally.&lt;/p&gt;

&lt;p&gt;In a small service, a team may be able to migrate all direct SDK calls and enable strict CI enforcement immediately.&lt;/p&gt;

&lt;p&gt;In an established codebase, that may not be realistic. A platform team may find hundreds of existing direct evaluations spread across services. They still need a way to prevent &lt;em&gt;new&lt;/em&gt; vendor-coupled access while reducing existing usage over time.&lt;/p&gt;

&lt;p&gt;That leads to the next direction for FlagLint: team adoption and governance workflows, including the ability to measure migration readiness across services and support gradual CI rollout without requiring a big-bang refactor.&lt;/p&gt;

&lt;p&gt;The goal remains simple:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Make the safer architecture easier to adopt than the shortcut.&lt;/p&gt;
&lt;/blockquote&gt;




&lt;h2&gt;
  
  
  Try FlagLint
&lt;/h2&gt;

&lt;p&gt;FlagLint is open source and available today as v0.5.1.&lt;/p&gt;

&lt;p&gt;Run it against a Node.js service using the LaunchDarkly server-side SDK:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint scan &lt;span class="nb"&gt;.&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Then explore a reviewable migration preview:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx flaglint migrate &lt;span class="nt"&gt;--dry-run&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Resources:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Website: &lt;a href="https://flaglint.dev/" rel="noopener noreferrer"&gt;flaglint.dev&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;GitHub: &lt;a href="https://github.com/flaglint/flaglint" rel="noopener noreferrer"&gt;github.com/flaglint/flaglint&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;npm: &lt;a href="https://www.npmjs.com/package/flaglint" rel="noopener noreferrer"&gt;npmjs.com/package/flaglint&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;OpenFeature: &lt;a href="https://openfeature.dev/" rel="noopener noreferrer"&gt;openfeature.dev&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;LaunchDarkly OpenFeature Node.js provider documentation: &lt;a href="https://launchdarkly.com/docs/sdk/openfeature/node-js" rel="noopener noreferrer"&gt;LaunchDarkly docs&lt;/a&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;I would especially value feedback from Node.js backend engineers and platform teams already using LaunchDarkly:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;What direct SDK patterns exist in your repositories?&lt;/li&gt;
&lt;li&gt;Do you use internal wrappers or shared feature-flag clients?&lt;/li&gt;
&lt;li&gt;What would make an OpenFeature migration safe enough to evaluate in a real service?&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The best developer tools are shaped by the real codebases they have to survive.&lt;/p&gt;

&lt;h1&gt;
  
  
  devops #typescript #javascript #opensource #featureflags
&lt;/h1&gt;

</description>
    </item>
  </channel>
</rss>
