If your company is small and your product is incipient it is likely that there will be very little specialization inside your engineering organization. Things are easy to understand, and everyone has a very good view of everything else that is happening.
However, as your engineering organization grows and your product surface area expands people will start losing track of what other people are doing.
A long tail of small tasks, bugs and requests for improvements will start to accumulate, and things will start falling through cracks. Items with higher urgency or higher impact will take over all the prioritization meetings.
People will either not know these long tail tasks exist, or, if they know, they may ignore them since no one up the chain seems to care.
At the same time, not all product surfaces are created equal. Some less visited corners, such as configuration pages, might start getting very little attention. No one is really looking into how to improve them.
These are all natural consequences of increased scope and limited time to pay attention to everything. What ends up happening after this is an overall degradation of product quality.
This degradation might manifest, for example, as a larger number of small bugs, usability issues or missing features. These might not hurt the main product functionality, but it may make your product look unprofessional, amateurish and unfinished.
While these small problems might be overlooked by some initial users, it will start to hurt you whenever your product is in a more competitive setup, such as when being compared side by side against a competitor. Larger deals will also come with larger scrutiny.
And unfortunately bugs are often non linear in how they manifest. A small overlooked bug might develop into a full fledged outage, with much larger and obvious consequences for the business.
Break it Down
This all indicates the need for better mechanisms to control the entropy as your company grows. One necessity that naturally arises is to subdivide your engineering organization into components.
Issues, tasks and roadmap items can be tracked against a component. Each component has an owner, so that smaller long tail tasks can be delegated. This ensures no issue or part of the product is overlooked and that responsibility can be properly attributed.
Components, however, come with their own pitfalls.
Human nature is to create walls around whatever they own. Dispute for resources and politics can easily evolve from components. This creates inefficiencies that are not in the best interest of the customer.
Also, components come with some red tape. Tickets need to be delegated to specific components. There has to exist control mechanisms to ensure that the owner will prioritize the work according to a common guideline. Metrics will be produced against a component, and they also need to be tracked.
Additionally, components might cause engineers to over-specialize too early and stop paying attention and contributing to other areas of the product. An engineering organization needs to be plastic in order to adapt quickly to new customer demands.
This often means that a large enough number of engineers need to be able to quickly switch to other areas of the product that demand more attention. Also, different components might start to adapt to diverging tech stacks and processes, which makes it harder for one engineer to migrate from one component into another.
Finally, component boundaries that are not well thought can cause confusion on who owns what and where to attribute an issue. This causes delay and back and forth on bug attribution, which is clearly waste from the customer point of view, and completely infuriating to watch as a support engineer.
Components as Code
One natural way to subdivide an engineering organization into components is to follow code interfaces and abstractions. There are many reasons for that:
- Code interfaces are intentionally written to allow good abstractions, which is exactly what delegating product areas into components means to do;
- Supposedly engineers should be thinking hard already on how to break the product code into modules; why not just leverage that;
- The cost to maintain code over time trumps the cost to first write it (specially in AI age); in the long term, the work will likely come proportionally to the amount of code and its complexity;
- Code is often organized into layers of abstraction, which is also a very good way to organize the components of an organization; an issue assigned to a higher layer can percolate to lower layers the same way function calls can percolate across the product stack;
- Abstractions should map upwards to the product user visible modules; this makes issue attribution easier for someone that does not understand the details of the product stack;
- Well thought abstractions are unlikely to change over time as the product changes; this makes sure that the same components will still remain valid over time, instead of changing constantly with new projects or product directions.
Component != Team
While each component should have one and only one owner at any specific moment in time, teams do not need to map directly to component borders.
Teams should obviously somewhat align in terms of the theme they work on. However, a healthy organization should make sure there is mobility of people across teams, and mobility of teams across components. This ensures that silos are not built around components, and that people can be allocated to the components that need more work at any point in time, without much red tape.
Aligning teams 1-1 in component borders would be counterproductive, since not all components require the same amount of work at any given moment of time. Some components are more mature or don’t really have much urgent or important work at a given point in time.
As a rule of thumb, engineers should not only be allowed to make changes in other components not owned by their teams, but they also should be incentivized. This again increases the plasticity of the organization in face of new customer demands.
What is NOT a Component
Following the component as code principle introduced above, non-functional areas, such as performance, quality or tests do not make good components.
These areas do not have clear interfaces and abstractions, and they are not easy to attribute responsibility. Arguably, these areas should be treated as horizontal areas that should apply to all components.
Non-functional components can cause functional component owners to overlook non-functional requirements – because they have someone else to blame. On the other side, owners of non-functional areas often do not have enough context on a specific functional component to be able to be effective.
Some caveats apply, however.
While these non-functional areas should not be treated as their own component, sometimes non-functional areas are hard to staff due to the prioritization of more easily visible functional areas. Also, it is very difficult to coordinate improvements and changes across several components under the same guidelines.
In these situations, a multi-functional team horizontal team, either temporarily assembled or permanent, might be the best solution. Remember that teams do not need to align to component boundaries, but where work needs to be done. An engineer can be part of a functional team and part of a cross-functional team for a specific share of its time.
Also, sometimes there is common code associated with non-functional areas, such as common test infrastructure, or common performance tooling. Depending on how mature an organization is it might make sense to build components around these code abstractions. These are support areas only, however – the owner of the performance tooling team cannot be responsible for every performance issue in the product.
Keys
- Organizations need components as they grow.
- However, they can slow down your organization when incorrectly implemented.
- Components should follow code. Code abstractions map naturally to component borders.
- Components != Teams. Teams should be dynamic. Components should be static.
- Non-functional areas are not good components.