Defining bounded contexts
Historical context
Until May 2024 the GitLab codebase didn't have a clear domain structure. We have forced the creation of some modules as a first step but we didn't have a well defined strategy for doing it consistently.
The majority of the code was not properly namespaced and organized:
- Ruby namespaces used didn't always represent the SSoT. We had overlapping concepts spread across multiple
namespaces. For example:
Abuse::
andSpam::
orSecurity::Orchestration::
andSecurity::SecurityOrchestration
. - Domain code related to the same bounded context was scattered across multiple directories.
- Domain code was present in
lib/
directory under namespaces that differed from the same domain underapp/
. - Some namespaces were very shallow, containing a few classes while other namespaces were very deep and large.
- A lot of the old code was not namespaced, making it difficult to understand the context where it was used.
In May 2024 we defined and enforced bounded contexts.
Goal
- Define a list of characteristics that bounded contexts should have. For example: must relate to at least 1 product category.
- Have a list of top-level bounded contexts where all domain code is broken down into.
- Engineers can clearly see the list of available bounded contexts and can make an easy decision where to add new classes and modules.
- Define a process for adding a new bounded context to the application. This should occur quite infrequently and new bounded contexts need to adhere to the characteristics defined previously.
- Enforce the list of bounded contexts so that no new top-level namespaces can be used aside from the authorized ones.
Iterations
-
Extract libraries out of the
lib/
directory.- This step is non blocking to modularization but the less generic code exists in
lib/
the easier will be to identify and separate bounded context. - Isolate code that could live in a separate project, to prevent it from depending on domain code.
- This step is non blocking to modularization but the less generic code exists in
-
ADR-001: Modularize application domain? Start with modularizing
-
ADR-002: Define bounded context around feature categories as a SSoT in the code.
-
Publish the list of bounded contexts.
- Define a SSoT list of bounded contexts.
- Enforce enforce it using RuboCop static analyzer.
- Autoload non-standard Rails directories based on the given list.