Entities & Aggregate Roots
Entities
Entities are one of the core concepts of DDD (Domain Driven Design). Eric Evans describes it as "An object that is not fundamentally defined by its attributes, but rather by a thread of continuity and identity".
An entity is generally mapped to a table in a relational database.
Entity Class
Entities are derived from the Entity<TKey>
class as shown below:
C#Copy
If you do not want to derive your entity from the base
Entity<TKey>
class, you can directly implementIEntity<TKey>
interface.
Entity<TKey>
class just defines an Id
property with the given primary key type, which is Guid
in the example above. It can be other types like string
, int
, long
, or whatever you need.
Entities with GUID Keys
If your entity's Id type is Guid
, there are some good practices to implement:
Create a constructor that gets the Id as a parameter and passes to the base class.
If you don't set a GUID Id, ABP sets it on save, but it is good to have a valid Id on the entity even before saving it to the database.
If you create an entity with a constructor that takes parameters, also create a
private
orprotected
empty constructor. This is used while your database provider reads your entity from the database (on deserialization).
An example entity:
Example usage in an application service:
BookAppService
injects the default repository for the book entity and uses itsInsertAsync
method to insert aBook
to the database.GuidGenerator
is type ofIGuidGenerator
which is a property defined in theApplicationService
base class. ABP defines such frequently used base properties as pre-injected for you, so you don't need to manually inject them.If you want to follow the DDD best practices, see the Aggregate Example section below.
Entities with Composite Keys
Some entities may need to have composite keys. In that case, you can derive your entity from the non-generic Entity
class. Example:
For the example above, the composite key is composed of UserId
and RoleId
. For a relational database, it is the composite primary key of the related table. Entities with composite keys should implement the GetKeys()
method as shown above.
EntityEquals
Entity.EntityEquals(...)
method is used to check if two Entity Objects are equals.
Example:
AggregateRoot Class
AggregateRoot<TKey>
class extends the Entity<TKey>
class. So, it also has an Id
property by default.
An aggregate root is responsible for preserving its own integrity. This is also true for all entities, but the aggregate root has responsibility for its sub-entities too. So, the aggregate root must always be in a valid state.
An aggregate root can be referenced by its
Id
. Do not reference it by its navigation property.An aggregate root is treated as a single unit. It's retrieved and updated as a single unit. It's generally considered as a transaction boundary.
Work with sub-entities over the aggregate root- do not modify them independently.
Aggregate Example
This is a full sample of an aggregate root with a related sub-entity collection:
If you do not want to derive your aggregate root from the base
AggregateRoot<TKey>
class, you can directly implement theIAggregateRoot<TKey>
interface.
Order
is an aggregate root with Guid
type Id
property. It has a collection of OrderLine
entities. OrderLine
is another entity with a composite primary key (OrderId
and ProductId
).
While this example may not implement all the best practices of an aggregate root, it still follows some good practices:
Order
has a public constructor that takes minimal requirements to construct anOrder
instance. So, it's not possible to create an order without an id and reference number. The protected/private constructor is only necessary to deserialize the object while reading from a data source.OrderLine
constructor is internal, so it is only allowed to be created by the domain layer. It's used inside of theOrder.AddProduct
method.Order.AddProduct
implements the business rule to add a product to an order.All properties have
protected
setters. This is to prevent the entity from arbitrary changes from outside of the entity. For example, it would be dangerous to setTotalItemCount
without adding a new product to the order. Its value is maintained by theAddProduct
method.
ABP does not force you to apply any DDD rule or patterns. However, it tries to make it possible and easier when you do want to apply them. The documentation also follows the same principle.
Aggregate Roots with Composite Keys
While it's not common (and not suggested) for aggregate roots, it is in fact possible to define composite keys in the same way as defined for the mentioned entities above. Use non-generic AggregateRoot
base class in that case.
BasicAggregateRoot Class
However, if you don't need these features, you can inherit from the BasicAggregateRoot<TKey>
(or BasicAggregateRoot
) for your aggregate root.
Base Classes & Interfaces for Audit Properties
There are some properties like CreationTime
, CreatorId
, LastModificationTime
... which are very common in all applications. ABP provides some interfaces and base classes to standardize these properties and also sets their values automatically.
Auditing Interfaces
There are a lot of auditing interfaces, so you can implement the one that you need.
While you can manually implement these interfaces, you can use the base classes defined in the next section to simplify it.
IHasCreationTime
defines the following properties:CreationTime
IMayHaveCreator
defines the following properties:CreatorId
ICreationAuditedObject
inherits from theIHasCreationTime
and theIMayHaveCreator
, so it defines the following properties:CreationTime
CreatorId
IHasModificationTime
defines the following properties:LastModificationTime
IModificationAuditedObject
extends theIHasModificationTime
and adds theLastModifierId
property. So, it defines the following properties:LastModificationTime
LastModifierId
IAuditedObject
extends theICreationAuditedObject
and theIModificationAuditedObject
, so it defines the following properties:CreationTime
CreatorId
LastModificationTime
LastModifierId
IsDeleted
IHasDeletionTime
extends theISoftDelete
and adds theDeletionTime
property. So, it defines the following properties:IsDeleted
DeletionTime
IDeletionAuditedObject
extends theIHasDeletionTime
and adds theDeleterId
property. So, it defines the following properties:IsDeleted
DeletionTime
DeleterId
IFullAuditedObject
inherits from theIAuditedObject
and theIDeletionAuditedObject
, so it defines the following properties:CreationTime
CreatorId
LastModificationTime
LastModifierId
IsDeleted
DeletionTime
DeleterId
Once you implement any of the interfaces, or derive from a class defined in the next section, ABP automatically manages these properties wherever possible.
Auditing Base Classes
While you can manually implement any of the interfaces defined above, it is suggested to inherit from the base classes defined here:
CreationAuditedEntity<TKey>
andCreationAuditedAggregateRoot<TKey>
implement theICreationAuditedObject
interface.AuditedEntity<TKey>
andAuditedAggregateRoot<TKey>
implement theIAuditedObject
interface.FullAuditedEntity<TKey>
andFullAuditedAggregateRoot<TKey>
implement theIFullAuditedObject
interface.
All these base classes also have non-generic versions to take AuditedEntity
and FullAuditedAggregateRoot
to support the composite primary keys.
Caching Entities
It's designed as read-only and automatically invalidates a cached entity if the entity is updated or deleted.
Versioning Entities
ABP defines the IHasEntityVersion
interface for automatic versioning of your entities. It only provides a single EntityVersion
property, as shown in the following code block:
If you implement the IHasEntityVersion
interface, ABP automatically increases the EntityVersion
value whenever you update your entity. The initial EntityVersion
value will be 0
, when you first create an entity and save to the database.
ABP can not increase the version if you directly execute SQL
UPDATE
commands in the database. It is your responsibility to increase theEntityVersion
value in that case. Also, if you are using the aggregate pattern and change sub-collections of an aggregate root, it is your responsibility if you want to increase the version of the aggregate root object.
Extra Properties
ABP defines the IHasExtraProperties
interface that can be implemented by an entity to be able to dynamically set and get properties for the entity. AggregateRoot
base class already implements the IHasExtraProperties
interface. If you've derived from this class (or one of the related audit class defined above), you can directly use the API.
GetProperty & SetProperty Extension Methods
These extension methods are the recommended way to get and set data for an entity. Example:
Property's value is object and can be any type of object (string, int, bool... etc).
GetProperty
returnsnull
if given property was not set before.You can store more than one property at the same time by using different property names (like
Title
here).
It would be a good practice to define a constant for the property name to prevent typo errors. It would be even a better practice to define extension methods to take the advantage of the intellisense. Example:
Then you can directly use user.SetTitle("...")
and user.GetTitle()
for an IdentityUser
object.
HasProperty & RemoveProperty Extension Methods
HasProperty
is used to check if the object has a property set before.RemoveProperty
is used to remove a property from the object. You can use this instead of setting anull
value.
How it is Implemented?
IHasExtraProperties
interface requires to define a Dictionary<string, object>
property, named ExtraProperties
, for the implemented class.
So, you can directly use the ExtraProperties
property to use the dictionary API, if you like. However, SetProperty
and GetProperty
methods are the recommended ways since they also check for null
s.
How is it Stored?
The way to store this dictionary in the database depends on the database provider you're using.
Discussion for the Extra Properties
Extra Properties system is especially useful if you are using a re-usable module that defines an entity inside and you want to get/set some data related to this entity in an easy way.
You typically don't need to use this system for your own entities, because it has the following drawbacks:
It is not fully type safe since it works with strings as property names.
Extra Properties Behind Entities
IHasExtraProperties
is not restricted to be used with entities. You can implement this interface for any kind of class and use the GetProperty
, SetProperty
and other related methods.
Last updated