Feature Flags Done My Way
When I started developing software many years ago I started the self-taught, hacky way. I improved my craft throughout the years, but when I came across the concept of feature flags I didn’t pay much attention to it. First I thought it was one of those fancy names we give to pretty common things, you know, to make money out of it. I’ve had my fair share of geeky names that were fairly meaningless by then, or that meant different things to different people and everyone was too scared to ask what the real meaning behind it was. I thought I was doing feature flags already, using variables in the code to turn things on and off, and later I moved those things to environment variables, so that I wouldn’t even have to “change code” to action my flags. It worked well for me, or so I thought.
Years later I came across a software that was doing feature flagging. I tested it, and I thought to myself: why would you want to turn things on and off through a UI? Why would I want my system to depend on a third party service to be able to function? Clearly that was an inferior design choice, even though it was indeed a prettier UI. By that point I already had something similar, a software/system that I used to configure other systems, a “dashboard” so to speak or control center. The problem with this control plane is that every time I had to develop a new feature I had to do some work. That’s a problem. The second problem was the coupling between the two systems, and yet a third was that each system was different to the other, thus abstracting that logic away and standardizing that implementation wouldn’t be so great for me. Yes, I could do easy on/off switches like that, but anything beyond was not really that feasible. This feature flagging thing was not all that great after all.
Feature Flags Again
Serendipitiously I was going through GitLab’s docs and I found the feature flagging documentation. I don’t quite recall now what they were using, but I thought to myself: hey, if I can configure my software in the same platform I use to keep track of my code, that would be pretty cool! I digged a bit at the documentation, but what I found was disappointing. Yes, you could do it, but it was a mere on/off switch. I went down a rabbit hole and found a software called Growthbook. That’s when things changed.
GrowthBook - Feature Flags Done Right
Things really did change that day for me. I tested their platform out which was free, and I was like: this is pretty cool! Not only could you turn things on/off, but you had multiple environments from the start, integration with the code was super simple due to their neat SDK, and you could return more than a boolean, you could return a JSON object! Now that opened up doors. The platform is basically a rules engine that gives you back a value. That value can be a simple on/off, true/false, or something as complex as a JSON, whatever JSON you want. Now, I knew a lot of places where I could use something like that. Matteroffactly, I had a lot of places in my code where I held configuration information that was just a long list of if/else conditions, and that ruled engine looked pretty sleek. One thing led to another, and collaborating with a colleague we moved about 40 rules or more to a “feature flag” in GrowthBook. The thinking was simple: if I give you an input and you give me a single output (much like a function), I can move all my configuration to you and you’ll just spit it back. I remove logic from my code that is just holding basic configuration, and my code becomes all the clearer! Well, that was just the tip of the iceberg. Our POC went extremely well, the code became much lighter, our configuration moved away from the codebase (many times we were deploying just configuration changes), but we gained more than pure technical points. Our configuration became transparent. All of a sudden we had a system that had all the rules written in plain English (or plain enough), that any business user could read and understand, and even make changes themselves if we so allowed. The system was version controlled, independently testable through the environments, and you could just add one config after the other and move stuff away one microservice at the time. Soon all our “configuration” was moved to this platform, secrets and environmental variables notwithstanding.
Thinking on the terms of business logic: do I want to do X if Y happens? We had already experimenting building Low-code solutions, we had tried doing something like that in a way business users could understand. That was a fun experience that didn’t give us much. We tried it. There, done. I’ll write an article on that later. GrowthBook feature flags were different though. They were just a set of rules, executed from top to bottom, that gave our app certain configuration we needed. Think about routing decisions. We had a ton of routing decisions, all of that living in code. Well, this took care of that. We had the previous dashboard we had build that was holding some of these “configuration” values. Well, we moved that too to GrowthBook. Why build something manually when you can just build just another flag to hold this configuration. A flag that holds on/off is no fun. A flag taht holds JSON is a game changer, for it can hold whatever you want.
Benefits We Experienced
Some key benefits were not obvious at first. With a simple feature flagging system you can turn things on/off. If that doesn’t happen often, well, who cares, you don’t win much. However, the benefits go much deeper than this.
- Exposing core logic to business users.
This was our first big win. The logic that was before hidden, and everyone always had questions about, even our own developers, was now publicly accessible. Just the fact that was somewhere people could see it made a difference psychologically. I don’t think anybody pays it any attention, but they know it is there, somewhere.
- Simplifying our code.
This was huge, well beyond expectations. Our code got a lot simpler because all of this logic was out the window. We had a lot of if/else conditions that were just abstracted away.
- Deprecated systems.
Well, we just removed some functionality we had that was sort-of doing the same thing. You shouldn’t compete at building something that won’t differentiate you, and a control panel is just that, a control panel. Move on, let others build it for you.
- No trivial deployments.
I’m all for deploying often, but many deployments before were just us changing some configuration that had nothing to do with actual code. These were all gone in one swoop.
- Faster development.
No, really. Have you been in the situation where you are done, but you can’t deploy just yet because you are coordinating with someone? Have you been coordinating for two weeks? Well, I have. With a feature flag you can just turn the new code off until that dependency clears, and then you just “turn it on”. That gives you speed, because you can get to that deployment and that QA (assuming you can QA the feature) all the quicker, while you still remember what it is that could break. Even better, you can turn things on quickly, test them, and turn them back off, or you can create conditions that only you can reach, so that you can test, and then clean that up once you are good. This gives you speed. No more coordinating to deploy. What’s even better, your business users can turn the thing on themselves, if they know where to look.
- Nimble testing and fine-grained control
Well, GrowthBook has morphed into a testing platform now, but I’m speaking here from a pure development perspective. You can just test things quicker with a good platform like this. Imagine creating rules that would A/B test two microservices, one legacy and one newer version. With a good feature flag, you can control the rollout from one system to the next and back again within seconds. You can do that with code too, but it takes a lot of deployments of trivial code, and that takes time. That simple UI comes in handy at these junctions. You just need to make a change, add a commit message, and publish your change. And rinse and repeat. We’ve been able to migrate from one system to the next in a very controlled manner this way. We’ve also been able to deploy critical functionality in a controlled fashion, and roll it all back within minutes. It doesn’t happen that way if you don’t develop code that way, but with a good tool you can start small and chip at it.
Conclusion
I’m sure there are more good benefits to feature flags, but the above is plenty to make a good case for using them often. More often than not we don’t use them for wrapping new features on them. We use them when we need to, for specific purposes. We don’t have 1000 flags controlling everything the code does. Every IF condition doesn’t become a feature flag. We control what’s strategic for us. Places of uncertainly. Functionality that requires modification, routing decisions, or coordination. We use if often to migrate from one API to another in a controlled way, and for removing logic that just shouldn’t exist in the codebase. Now, a word of warning: if the use case doesn’t fit the tool, don’t force it. We’ve seen rules that are just too complex for a simple rules engine to handle, and we’ve had to force the rule to fit the system. Don’t do that. Start small, and expand from there. A good feature flag is much more than a simple “semaphore”. Think bigger. I know I had to. I also had to find the right tool. You don’t need a whole book about it. There’s my two cents on the topic. Happy coding!