Domain Driven Design
A good code base creates value for the organization and to the client. The better the code reflects the problem domain the more Money it is going to create for the organization. In his book (Domain-Driven Design: Tackling Complexity in the Heart of Software), Eric Evans first introduced the Philosophy of the Domain Driven design (DDD). While the full explanation would take a couple of 500-page books, the essence of DDD is profoundly simple: capture the domain model in domain terms, embed the model in the code, and protect it from corruption.
Although let me warn you DDD is not
It always feels good when you look at the code which is written by you sometime back, and you are still able to understand it and modify it without affecting anything else in the system, it feels much better if someone else does it with your code without having your guidance. Is it because the person who is doing it is genius? Or is it because the chunk of the code written in a way which is easier to understand and modify? Well both the cases can be true. In the former case, if the code base is rotten, even the genius developers will not be able to do anything at certain point but to apply patches here and there. In such cases usually management people takes control of the situations and load more developers in the team. What these new developers do? they write more code, and as we already know it results in a nightmare. Adding simple features become way more difficult than to re-write a certain module and then add new features to it.
We are aware of these problems and we don’t want to repeat the same mistake again in our next assignment. In such situations Developers generally look out for new fancy coding practices and trendy design pattern. Well these things can save you for some time but it doesn’t show you how to model your domain in to the code, how to architect a model which is
Below is a high-level diagram of how DDD solution looks.
Below are the building blocks of DDD
Layered Architecture:
Creating a complex software which can handle very complex tasks calls for separation of concerns, allowing focusing on various parts of the software in isolation. There are all sorts of ways and practices a software system is divided, but through experience and convention the industry has settled on Layered Architectures. The value of layering is that each layer specializes into a aspect of the system. Also, the layers work in a way that the elements of a layer depends only on other elements of the same layer or on the elements of the layer beneath it.
User Interface (or presentation layer): Responsible for showing information to the user of the software. It interprets the user commands.
Application Layer: Manages transactions, translate DTO’s, coordinate application activities, creates and access domain objects. This layer is kept thin, it does not contain any business rules or knowledge. It does not have state reflecting the business situation.
Domain Layer: This layer is the heart of the business software. It represents the Business concepts, business Rules. This layer should not have any technical details like, database code, network code, this should be delegated to infrastructure layer.
Infrastructure Layer: The technical capabilities that support the layers above, i.e. persistence or messaging. This layer provides generic technical capabilities to the higher layers.
The Ubiquitous Language
You might have heard this term before, it emphasizes on using the Domain terminologies everywhere. It simply means call a domain term with its name, the rule is use that name everywhere, while talking to the client, discussing within the team and off course while writing the code. Your focus should be on creating a language which is understandable by the client and the developers. Your code should scream this language and the domain model should
reflect this language. This language should keep evolving with the better understanding of the domain, so is the domain model.
So, how do you create this ubiquitous language? the answer is simple, talk to domain experts, creating ubiquitous language is an evolving process, one simply can’t discover all the terminologies within a given timeframe. Another problem we face is domain experts doesn’t understand the technical jargon we developers use. Having a ubiquitous language to communicate greatly solves.
The Design
Entities: An Object primarily defined by its identity is called an Entity. It is quite easy to identify the entities; the domain experts refer to it as a single Identifiable item. In many cases you get the hint directly, the domain experts will mention a unique number or a ID or a serial number etc., as being part of the concept. If business care about tracking each instance of that concept, we are dealing with entities.
For example, consider a Person Concept, if we have two-person objects with the same name, are they the same person? “Raj Mehta” from Mumbai and “Raj Mehta” from Indore might not agree. So, if name is not the person’s identity, what it is? Address? In this case the person is not identifiable by its attributes, like Name, Address, age etc. It can be identifiable by its unique Identifier, which can be generated by the system, or by some algorithm, but to access that entity we should always use this identifier.
If the two instances of the same objects have different attribute values, but same identity values, are they the same entity? If the answer is yes, then the class is indeed an Entity.
Value Objects: Unlike Entities value objects don’t have any special identity also they are immutable.
“An object that represents a descriptive aspect of the domain with no conceptual identity is called a value object.” -Eric Evans
Value objects are not very hard to identify, it’s a concept which doesn’t have any identity. Value objects not only have values, but they incorporate business constraints. Value objects don’t have any meaning on their own, they are basically a part of some Entities.
Let’s take an example, A price is a value objects which is represented by an Amount and the currency, it doesn’t have any explicit identity. We generally do not differentiate between one $10 and another $10.
Another example is “Address” instead of adding attributes like Street, City, State, Pin etc. to the Person Object, it should be part of an “Address” object, which again doesn’t have any meaning without the Person object.
We don’t care which instance of a value object we have. This lack of constraint gives us design freedom we can use to simplify design or optimize performance. They can be passed as a parameter between two objects. They are frequently transient, created for an operation and then discarded.
Services: A Service is an operation offered as an interface that stands alone in the model. Without encapsulating any state like Entities or Value Objects. When a model concept doesn’t naturally fit in to Entity or Value object, a service is an appropriate. While creating a new service, define the interface and operations in terms of model language. Make sure the operation name is part of the Ubiquitous Language. Make the service stateless.
A good service has below characteristics
The life cycle of a domain Object (controlling the dance of the Entities)
Every Object has a lifecycle, an object is born, it then goes through various states and eventually dies. Some of the objects live a short life while others have longer lifespan. The objects with longer lifespan doesn’t necessarily spent all the time in the active memory. They have complicated lifecycle and dependencies with other objects. They go through changes of state,]
Managing these objects present challenges that can easily derail an attempt at Model Driven Design. These challenges fall in to two categories
Below are the patterns which will address these challenges.
Aggregates: Aggregates tighten up the model itself by defining the clear ownership and boundaries, avoiding the chaotic tangled web of objects. It is a cluster of domain objects that can be treated as a single unit. An Aggregate will have one of its component object act as the aggregate root, any reference from outside should only go to the aggregate root. The root can thus ensure the integrity of the Aggregate. If say you want to access the Address of a Person, then it should always be accessed by using the Person object as an aggregate root. The Root (Person) then can give the safe access to the required object, in this case it is Address. In almost all the cases the Root is always going to be an Entity.
Below are the set of rules while defining the Aggregates.
A good code base creates value for the organization and to the client. The better the code reflects the problem domain the more Money it is going to create for the organization. In his book (Domain-Driven Design: Tackling Complexity in the Heart of Software), Eric Evans first introduced the Philosophy of the Domain Driven design (DDD). While the full explanation would take a couple of 500-page books, the essence of DDD is profoundly simple: capture the domain model in domain terms, embed the model in the code, and protect it from corruption.
Although let me warn you DDD is not
- A Silver bullet for every problem
- It is not the new Shiny/Trendy design pattern
- it is not a new template in the latest and greatest IDE’s
- And certainly not some coding standard/guidelines for the rookie software developers.
It always feels good when you look at the code which is written by you sometime back, and you are still able to understand it and modify it without affecting anything else in the system, it feels much better if someone else does it with your code without having your guidance. Is it because the person who is doing it is genius? Or is it because the chunk of the code written in a way which is easier to understand and modify? Well both the cases can be true. In the former case, if the code base is rotten, even the genius developers will not be able to do anything at certain point but to apply patches here and there. In such cases usually management people takes control of the situations and load more developers in the team. What these new developers do? they write more code, and as we already know it results in a nightmare. Adding simple features become way more difficult than to re-write a certain module and then add new features to it.
We are aware of these problems and we don’t want to repeat the same mistake again in our next assignment. In such situations Developers generally look out for new fancy coding practices and trendy design pattern. Well these things can save you for some time but it doesn’t show you how to model your domain in to the code, how to architect a model which is
- testable (you are free to write unit tests on it),
- which is independent of the framework on which it is going to run (A web framework? A mobile application or a console application.)
- can adopt changes easily
- can scale with the changing requirements
- it is protected from corruption
- which is independent of the database being used (the team can decide which database they are going to use in the later phase of the development, no more databasish code).
Below is a high-level diagram of how DDD solution looks.
We developers love to write code, we always thrive to write code which is scalable and easy to understand by others. To achieve this, we keep haunting for the best practices. As I already stated, a good code base creates value for the organization. The more the scalable code is, the easier and faster it is to add features in it.
Below are the building blocks of DDD
Layered Architecture:
Creating a complex software which can handle very complex tasks calls for separation of concerns, allowing focusing on various parts of the software in isolation. There are all sorts of ways and practices a software system is divided, but through experience and convention the industry has settled on Layered Architectures. The value of layering is that each layer specializes into a aspect of the system. Also, the layers work in a way that the elements of a layer depends only on other elements of the same layer or on the elements of the layer beneath it.
User Interface (or presentation layer): Responsible for showing information to the user of the software. It interprets the user commands.
Application Layer: Manages transactions, translate DTO’s, coordinate application activities, creates and access domain objects. This layer is kept thin, it does not contain any business rules or knowledge. It does not have state reflecting the business situation.
Domain Layer: This layer is the heart of the business software. It represents the Business concepts, business Rules. This layer should not have any technical details like, database code, network code, this should be delegated to infrastructure layer.
Infrastructure Layer: The technical capabilities that support the layers above, i.e. persistence or messaging. This layer provides generic technical capabilities to the higher layers.
The Ubiquitous Language
You might have heard this term before, it emphasizes on using the Domain terminologies everywhere. It simply means call a domain term with its name, the rule is use that name everywhere, while talking to the client, discussing within the team and off course while writing the code. Your focus should be on creating a language which is understandable by the client and the developers. Your code should scream this language and the domain model should
reflect this language. This language should keep evolving with the better understanding of the domain, so is the domain model.
So, how do you create this ubiquitous language? the answer is simple, talk to domain experts, creating ubiquitous language is an evolving process, one simply can’t discover all the terminologies within a given timeframe. Another problem we face is domain experts doesn’t understand the technical jargon we developers use. Having a ubiquitous language to communicate greatly solves.
Business experts have their own Jargon that might be difficult for developers to digest, similarly developers have their own technical jargon which business people don’t understand. Having a ubiquitous language in place, it becomes easier to communicate with the Business people, if the Domain Model is speaking this language without any ambiguity the business people would be able to participate and sometimes they can suggest changes to the model (off course for this to happen the Model should be simpler). If there are ambiguities in the terminologies used in the Model and the Business Expert Jargon, it results in a fractured Model.
Entities: An Object primarily defined by its identity is called an Entity. It is quite easy to identify the entities; the domain experts refer to it as a single Identifiable item. In many cases you get the hint directly, the domain experts will mention a unique number or a ID or a serial number etc., as being part of the concept. If business care about tracking each instance of that concept, we are dealing with entities.
For example, consider a Person Concept, if we have two-person objects with the same name, are they the same person? “Raj Mehta” from Mumbai and “Raj Mehta” from Indore might not agree. So, if name is not the person’s identity, what it is? Address? In this case the person is not identifiable by its attributes, like Name, Address, age etc. It can be identifiable by its unique Identifier, which can be generated by the system, or by some algorithm, but to access that entity we should always use this identifier.
If the two instances of the same objects have different attribute values, but same identity values, are they the same entity? If the answer is yes, then the class is indeed an Entity.
Value Objects: Unlike Entities value objects don’t have any special identity also they are immutable.
“An object that represents a descriptive aspect of the domain with no conceptual identity is called a value object.” -Eric Evans
Value objects are not very hard to identify, it’s a concept which doesn’t have any identity. Value objects not only have values, but they incorporate business constraints. Value objects don’t have any meaning on their own, they are basically a part of some Entities.
Let’s take an example, A price is a value objects which is represented by an Amount and the currency, it doesn’t have any explicit identity. We generally do not differentiate between one $10 and another $10.
Another example is “Address” instead of adding attributes like Street, City, State, Pin etc. to the Person Object, it should be part of an “Address” object, which again doesn’t have any meaning without the Person object.
We don’t care which instance of a value object we have. This lack of constraint gives us design freedom we can use to simplify design or optimize performance. They can be passed as a parameter between two objects. They are frequently transient, created for an operation and then discarded.
Services: A Service is an operation offered as an interface that stands alone in the model. Without encapsulating any state like Entities or Value Objects. When a model concept doesn’t naturally fit in to Entity or Value object, a service is an appropriate. While creating a new service, define the interface and operations in terms of model language. Make sure the operation name is part of the Ubiquitous Language. Make the service stateless.
A good service has below characteristics
- The operation relates to a domain concept that is not a natural part of an Entity or Value Object
- The interface is defined in terms of other elements in the domain model.
- The Service is stateless.
- Domain Services: Domain Services generally hold domain concepts which doesn’t naturally fits in to Entities or Value Objects.
- Application Services: They Don’t hold any Domain concepts. e.g., JSON Parser, Cache, Web Services etc.
- Infrastructure Services: Infrastructure Services generally are the external dependencies that need to be implemented by the system, they are not part of the Domain model, although domain model might be dependent on them. e.g. Email Service, File Explorer etc.
The life cycle of a domain Object (controlling the dance of the Entities)
Every Object has a lifecycle, an object is born, it then goes through various states and eventually dies. Some of the objects live a short life while others have longer lifespan. The objects with longer lifespan doesn’t necessarily spent all the time in the active memory. They have complicated lifecycle and dependencies with other objects. They go through changes of state,]
- Maintaining integrity throughout the lifecycle
- Preventing the model from getting swamped by the complexity of managing the lifecycle.
Below are the patterns which will address these challenges.
Aggregates: Aggregates tighten up the model itself by defining the clear ownership and boundaries, avoiding the chaotic tangled web of objects. It is a cluster of domain objects that can be treated as a single unit. An Aggregate will have one of its component object act as the aggregate root, any reference from outside should only go to the aggregate root. The root can thus ensure the integrity of the Aggregate. If say you want to access the Address of a Person, then it should always be accessed by using the Person object as an aggregate root. The Root (Person) then can give the safe access to the required object, in this case it is Address. In almost all the cases the Root is always going to be an Entity.
Below are the set of rules while defining the Aggregates.
- The root Entity has global identity and is ultimately responsible for checking invariants Root Entities have global identity.
- Entities inside the boundary have local identity, unique only within the Aggregate.
- Nothing outside the Aggregate boundary can hold a reference to anything inside, except to the root Entity. The root Entity can hand references to the internal Entities to other objects, but they can only use them transiently (within a single method or block).
- Only Aggregate Roots can be obtained directly with database queries. Everything else must be done through traversal.
- Objects within the Aggregate can hold references to other Aggregate roots.
- A delete operation must remove everything within the Aggregate boundary all at once.
- When a change to any object within the Aggregate boundary is committed, all invariants of the whole Aggregate must be satisfied.
Factories: When creation of an object or an entire Aggregate, becomes complicated, or reveals too much of the internal structure, Factories provide encapsulation. Complex object creation is the responsibility of the domain layer, yet that task does not belong to the objects that express the model.
▪Centralize the creation of an object
Factories provide a consistent interface for creating complex objects within an application. As an application grows there are chances that the rules might get lost. Factories provide a standard way to instantiate objects so that the creation of objects not lost in ambiguity.
▪ Factories provide Encapsulation
A factory encapsulates the details which are required to create the complex object or an Aggregate. The factory must deal with this complexity by knowing all the necessary details required to instantiate the object or an Aggregate, but it will encapsulate all these details from outside world by providing a simple interface.
Because of the complex creation process, it is sometimes natural that the internals of an object or an Aggregate leak in to the factory. This is an inevitable situation. Objects in their lifecycle will be moved around throughout the application process, it is rare that they will entirely live in memory, instead objects are moved in and out of the Databases or they are transferred across network. Factories can be used to reconstitute the object as well as create the new object. However, factories should not assign a new ID to an existing Entity.
Repositories: Repositories are the classes which encapsulates the Data access logic from the Domain model. They generally centralize the common data access functionality, providing better maintainability and decoupling the infrastructure or technology used to access objects from the database or from a remote server in the form of JSON or XML.
for the application repositories generally acts as in memory domain object collection. Objects can be added or removed from the repositories as they represent them as the collection of objects. Conceptually Repositories encapsulates the set of objects persisted in data store and the operations performed over them, providing a more object-oriented view of the persistence layer.
Repositories have many advantages including following,
1. They present clients with a simple model for obtaining persistent object and managing their lifecycle.
2. They decouple application and the domain design from persistence technology.
3. They allow easy substitution of a dummy implementation, for use in testing (typically using in memory collections).
And there is much more!!!
Domain Driven Design is a collection of practices that illustrate how to design and architect the model by using the information in the form of Ubiquitous language. A strong ubiquitous language and the model driven design at the core is a foundation to have a stable software. Teams find themselves accomplishing their work with enhanced precision and clarity. This small writeup is just a tip of the iceberg. To explore this topic in depth I would recommend you have a look at Eric Evan’s Book “Domain-Driven Design: Tackling Complexity in the Heart of Software”.
Comments
Post a Comment