I live in one of the suburbs of Tel-Aviv. One day, I woke up to insane drilling sounds. It started! the construction work to build the purple line of the light rail. “How bad is it going to be?” I thought to myself…
A few weeks in, sporadically waking up at 5am depending on when the construction work starts, my wife told me that it’s time for us to move out. Desperately trying to build my case for staying at the apartment (I love the area!), I tried to find the construction plans, assuming that the drilling part won’t take so long.
When I viewed the construction plans, as a software engineer, I was pretty amazed. They had detailed, concrete plans for the next 3 years. They separated it into 3 phases, with exact measurements and road structures for each phase. They had a solid timeline. And the most interesting part — I was able to easily read it, and it was clear to me, although I have 0 knowledge in civil engineering.
In fact, this expands to other types of engineering/architecture as well. Think about diagrams of house architecture, electrical engineering, or mechanical engineering — there are consistent ways to write these so all other engineers in the field can read.
But this is not the case for software engineering. While it’s easy to create architecture diagrams in software (using boxes and arrows), reading and understanding them might not be straightforward.
The Difference with Software Engineering
During my career as an engineer I’ve created, read and reviewed numerous software architecture diagrams and design docs, and conducted many system design interviews. I came to realize that people describe software very differently.
Differently, means:
- Different level of abstractions
- Different level of details
- Different illustration/diagramming techniques
- Different focus on non-functional aspects such as reliability, security, privacy and scale
It’s common to find such communication gaps even in the same team, not to mention different teams, orgs or companies. These gaps manifest, for instance, in architecture diagrams that are not self-explanatory. They either come with significant wording (think 10-page design doc) or require significant time spent on explaining them.
These communication gaps have even more impact due to a recent trend in the software engineering world.
Decentralizing the Architect Role
“Architecture as a team sport:
Architects no longer work alone, and architects can no longer think only about technical issues. The role of an architect varies greatly across the industry, and some companies have eliminated the title entirely…”InfoQ Software Architecture and Design Trends Report — April 2023
In the past, most companies and organizations implemented a top-down approach for software design and architecture — employing an architect for each big enough group, that acts as the centralized authority for the product/system design. It was common for engineers to execute coding tasks based on given architecture, specification and APIs.
The world has shifted to a more decentralized approach, where engineers at all levels now participate in designing and architecting, rather than just coding. What varies between the different levels and skillset of engineers is the scope of their work — for instance, junior engineers could design a specific component, while more senior engineers can design a system, or a multi-system environment.
Basically, this is a cultural and mindset shift from the top-down, centralized design process to a bottom-up, decentralized and “owned by everyone” process.
The Cost
This change comes with many advantages: empowering and growing engineers of all levels, increased sense of ownership and engagement, eliminating the single point of failure, and better spread of knowledge across teams.
However, it also comes with a cost:
- Quality: an architect is usually qualified as an experienced engineer with mileage in building and maintaining software systems. Spreading this work to everyone in the team means there is no further qualification aside of being part of the team. In order to keep an high quality bar, teams need to invest in training, education and mentoring, build a good design process and incorporate an open feedback culture.
- Overhead: having a solid design process in the org helps to increase the quality bar, but introduces a significant overhead. Usually such processes involve creating a design document, system diagrams, flow charts, and include a feedback process that can require multiple iterations and passes on the design.
Implementing a healthy design process in an organization is challenging. There is a tension between the overhead it introduces and the quality gain achieved by the process. If the process is too light, or does not exist at all, there is no quality gain. If the process is too heavy, it can become burdensome for engineers, who may not fully adhere to it or might even seek ways to circumvent it — ultimately leading to reduced efficiency.
The sweet spot is something in between, as always.
A Practical Takeaway: The C4 Model
There’s much to discuss on how a healthy design process should look like, but one practical concept I want to call out is the C4 model.
Visualizations are key part in software design. As mentioned above, people describe software very differently, especially when it comes to software diagrams.
Diagramming → Modeling
The C4 model aims to evolve the short-lived, freestyle diagramming techniques into a long-lasting, canonical, and standardized modeling technique. You can think of it as a set of rules and conventions on how software architecture should be visualized and described (similar to the standards that exist in other engineering fields).
The C4 model consists of 4 layers of abstractions (hence the name):
- System Context: the highest level of abstraction. This layer contains internal systems, external systems, users (if there are multiple types of users, they should be specified) and the relationships between the 3 of them.
- Container: once you have your systems in place, you can zoom into a specific system. A system is composed of one or more containers. A container represents an application or a data store, such as: backend service, frontend application, mobile application, relational DB, key-value store, message bus, object storage, etc.
An easy way to think about it, at least from a backend perspective, it to think about separate processes/executables. Each executable maps to its own container.
If you’ve had a system design interview before, it usually focuses on the container layer. - Component: next, a container can be broken into several components, usually by functionality and areas of responsibility. For instance, if we take a backend payment service, it could include components such as: payment executor, 3rd party client, recovery mechanism, payment recorder, notification sender, etc.
It’s ok that the components of a container will interact with other containers from layer 2 — for instance, the recovery mechanism could submit a message to the message bus to retry a failed payment in a certain delay. - Code: this is the mapping of components to actual code classes/modules/etc. This layer of implementation details is usually less interesting and is very hard to maintain, and in general not recommended for most use cases.
However, when writing code, it can be helpful to think “In which component my code fits?”; if the answer is not clear, either you’re creating a new component, or you’re violating the existing boundaries of components.
A few notes on the C4 model:
- Container and Component are the critical layers. Putting the essence of these layers clearly — your design choices on the Container layer impact scalability, reliability, deployment and evolvability.
Your design choices on the Component layer impact developer velocity, efficiency, maintainability and quality. - The C4 model adds depth to software architecture diagrams. Think of it as an online map — you start with the high level view, and can zoom in to places of interest. In case you got too much into the details, you can zoom out again and find yourself in the overarching architecture.
- The C4 model is a collaborative, long-term model. Instead of each engineer creating their own ad-hoc diagrams, the C4 model can be shared and worked on by multiple engineers or teams, expanding as the org and the product grow.
- There are plenty of tools that support the C4 model. Nowadays, even common enterprise diagramming software feature a C4 template.
Conclusion
Software design is a crucial part of the software development lifecycle, and has significant impact on the end result of the product or system, execution velocity, maintainability and evolvability.
With the recent trends of decentralizing the architect role and embracing “architecture as a team sport”, the structure of software teams is changing and organizations need to put careful thought on how their design process should look like and find the right balance between introducing overhead to maintaining a high quality bar.
Another interesting question is how GenAI will affect software development and software design in the long-term. We already see changes in engineers’ day to day with the usage of AI tooling, co-pilots and agents.
How does the design process in your organization look like? do you feel it’s well-structured and impactful? or is there a place for improvement? would love to hear your thoughts via the comments below.