In my work, I draw a lot of inspiration from Systems Thinking and Complexity Theory. In this post, I'd like to describe a concept of enabling constraints and how to use them in software development.
A constraint is something that limits what you can do. We can differentiate between two types of constraints: governing constraints and enabling constraints.
Governing constraints are strict and don’t allow for much flexibility. Here are some examples:
Governing constraints lead everyone to follow the same, well-defined path.
Enabling constraints are different. They don't show a single path, but enable you to explore new ways of solving problems. To understand why, let's take a look at some examples.
Pushing all the code directly to master, without branches, pull requests and blocking code review might seem weird.
I'm sure you can list many problems with this approach. What are they?
Branches and pull requests provide a safety net for our codebase. What if the safety net was off? Well, we'd have to try different ideas. Here are a few:
Here's a great enabling constraint: every Friday, you get the team and stakeholders together and do a demo of what you were able to accomplish, ideally on a live product.
What's the problem here? The most obvious one is that it's not always possible to ship an entire feature during one week. To solve this problem, we have to change the way we work. Again, some ideas:
In addition, we have a point in time when the entire team can reflect on the progress, learn, make sure that the work contributes to the long-term plan and goals. This creates alignment and provides the crucial feedback we need when dealing with complex problems.
A similar example of an enabling constraint is having daily status updates.
Code:
Deployment / operations:
Work organisation / teamwork:
Can you think what these constraints enable? How do they affect the people who do the work? What behavior emerges when these constraints are there?
You can notice a pattern here. The goal of governing constraints is to ensure a single proven (and clear) way of doing something. Governing constraints are like processes - they can give you a lot of safety and efficiency when applied correctly.
Governing constraints correspond to top-down decision making. They work well for complicated problems - when the rules are known, the goal is clear and things change slowly. If you are dealing with a well-defined problem and there’s an established, repeatable solution, setting governing constraints might be a good option.
The goal of enabling constraints is to encourage novelty and innovation. To generate new ideas, create many possible solutions, and experiment with them. The goal of this is to constantly learn and improve. That’s why it’s crucial for enabling constraint to provide constant and fast feedback.
Enabling constraints to support bottom-up decision making. They are more preferable for complex problems - when the rules are vague, uncertainty is high, the environment is rapidly changing, goals cannot be clearly defined and measured, things are unpredictable, and there are no best practices.
Enabling constraints influence how elements of the system behave and collaborate. They force alignment and provide feedback, while leaving a lot of freedom.
Constraints can lead to innovation. Preventing you from doing something doesn't necessarily mean that this thing is bad - it can be a great way to encourage trying new ideas in an uncertain and unstable environment.
Also - I’d encourage you to read more about Cynefin and differences between complex and complicated domains. Software development is full of both, so it is crucial to be able to recognize which kind of problem you're dealing with and act accordingly. Understanding what kind of problem you’re facing, will help you set the proper constraints.
If your environment is stable and you want to maximize safety and efficiency, think about governing constraints. If the environment is rapidly changing and unpredictable, you might want to optimize for novelty and innovation instead. In this case, setting the right enabling constraints is probably a good thing to do.
What is one practice that you think is absolutely crucial for the work you do? Is it code review? Is it TDD? Is it the relational database? Is it going to the office every day?
Can you imagine setting an enabling constraint that forbids that practice? How would your work look if you did that?
Are you willing to experiment a bit, try that for some time and see what happens? If you do, I'd love to hear about this!