Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Specification Pattern in Domain Driven Design

I've been struggling around a DDD-related issue with Specifications and I've read much into DDD and specifications and repositories.

However, there is an issue if trying to combine all 3 of these without breaking the domain-driven design. It boils down to how to apply filters with performance in mind.

First a few obvious facts:

  1. Repositories to got DataAccess/Infrastructure Layer
  2. Domain Models represent Business Logic and go to the Domain layer
  3. Data Access Models represent the Persistence layer and go to the Persistence/Infrastructure/DataAccess layer
  4. Business Logic goes to Domain Layer
  5. Specifications are Business Logic, so they belong to the Domain layer too.
  6. In all these examples, an ORM Framework and SQL Server is used inside the Repository
  7. Persistance Models may not leak into Domain Layer

So far, so easy. The problem arises when/if we try to apply Specifications to the Repository and not breaking the DDD pattern or having performance issues.

The possible ways to apply Specifications:

1) Classic way: Specifications using Domain Model in Domain Layer

Apply the traditional Specification Pattern, with a IsSatisfiedBy method, returning a bool and Composite Specifications to combine multiple Specifications.

This let us keep specifications in Domain Layer, but...

  1. It has to work with Domain Models, while the repository uses Persistence Models which represent the data structure of the persistence layer. This one is easy to fix with the usage of mappers such as AutoMapper.
  2. However, the problem which can't be solved: All the specifications would have to be performed in memory. In a big table/database this means a huge impact if you have to iterate through ALL Entities only to filter out the one which meets your specifications

2) Specifications using Persistence Model

This is similar to 1), but using Persistence Models in the specification. This allows direct use of the Specification as part of our .Where predicate which will be translated into a query (i.e. TSQL) and the filtering will be performed on the Persistence storage (i.e. SQL Server).

  1. While this gives us good performance, it clearly violates the DDD pattern. Our Persistence model leaks into the Domain layer, making the Domain Layer depend on the Persistence Layer instead of the other way around.

3) Like 2), but make Specifications Part of the Persistence Layer

  1. This doesn't work, because Domain Layer needs to reference the Specifications. It would still depend on the persistence layer.
  2. We would have business logic inside the Persistence layer. Which also violates the DDD pattern

4) Like 3, but use abstract the Specifications as Interfaces

We would have Specification interfaces in our Domain layer, our concrete implementations of the Specifications in the Persistence Layer. Now our Domain Layer would only interact with the interfaces and not depend on the Persistence layer.

  1. This still violates #2 from 3). We would have business logic in the persistence layer, which is bad.

5) Translate the Expression Tree from Domain Model into Persistence Model

This certainly solves the problem, but it's a non-trivial task but it would keep the Specifications inside our Domain Layer while still benefiting from SQL optimization because the Specifications becomes part of the Repositories Where clause and translates into TSQL

I tried going this approach and there are several issues (from the implementation side):

  1. We would need to know the Configuration from the Mapper (if we use one) or keep our own mapping system. This can be partly done (reading Mapper configuration) with i.e. AutoMapper, but further issues exist
  2. It's acceptable for one where one Property of Model A maps to one Property of Model B. It becomes more difficult if the types are different (i.e. due to persistence types, for example, Enums being saved as strings or key/value pairs in another table and we need to do conversions inside the resolver.
  3. It gets pretty complicated if multiple fields get mapped into one destination field. I believe this is non an issue for Domain Model -> Persistence Model mappings

6) Query Builder like API

The last one is making some kind of query API which is passed into the specification and from whom the Repository/Persistence layer would generate an Expression Tree to be passed to .Where clause and which uses an Interface to declare all filterable fields.

I did a few attempts in that direction too but wasn't too happy about the results. Something like

public interface IQuery<T> {     IQuery<T> Where(Expression<Func<T, T>> predicate); } public interface IQueryFilter<TFilter> {     TFilter And(TFilter other);     TFilter Or(TFilter other);     TFilter Not(TFilter other); }  public interface IQueryField<TSource, IQueryFilter> {     IQueryFilter Equal(TSource other);     IQueryFilter GreaterThan(TSource other);     IQueryFilter Greater(TSource other);     IQueryFilter LesserThan(TSource other);     IQueryFilter Lesser(TSource other); } public interface IPersonQueryFilter : IQueryFilter<IPersonQueryFilter> {     IQueryField<int, IPersonQueryFilter> ID { get; }     IQueryField<string, IPersonQueryFilter> Name { get; }     IQueryField<int, IPersonQueryFilter> Age { get; } } 

and in the specification, we would pass an IQuery<IPersonQueryFilter> query to the specifications constructor and then apply the specifications to it when using or combining it.

IQuery<IGridQueryFilter> query = null;  query.Where(f => f.Name.Equal("Bob") ); 

I don't like this approach much, as it makes handling complex specifications somewhat hard (like and or if chaining) and I don't like the way the And/Or/Not would work, especially creating expression trees from this "API".

I have been looking for weeks all over the Internet read dozens of articles on DDD and Specification, but they always only handle simple cases and don't take the performance into consideration or they violate the DDD pattern.

How do you solve this in a real-world application without doing in-memory filtering or leaking Persistence into Domain Layer??

Are there any frameworks that solve the issues above with one of the two ways (Query Builder like syntax to Expression Trees or an Expression Tree translator)?

like image 769
Tseng Avatar asked Sep 20 '14 22:09

Tseng


People also ask

What is domain design pattern?

Domain-Driven Design(DDD) is a collection of principles and patterns that help developers craft elegant object systems. Properly applied it can lead to software abstractions called domain models. These models encapsulate complex business logic, closing the gap between business reality and code.

Is DDD a design pattern?

Where to draw the boundaries is the key task when designing and defining a microservice. DDD patterns help you understand the complexity in the domain. For the domain model for each Bounded Context, you identify and define the entities, value objects, and aggregates that model your domain.

What is the domain model in domain-driven design?

The Domain Model is your organised and structured knowledge of the problem. The Domain Model should represent the vocabulary and key concepts of the problem domain and it should identify the relationships among all of the entities within the scope of the domain.

What is domain-driven design example?

An aggregate is a domain-driven design pattern. It's a cluster of domain objects (e.g. entity, value object), treated as one single unit. A car is a good example. It consists of wheels, lights and an engine.


2 Answers

I think Specification pattern is not designed for query criteria. Actually, the whole concept of DDD is not, either. Consider CQRS if there are plethora of query requirements.

Specification pattern helps develop ubiquitous language, I think it's like kind of a DSL. It declares what to do rather than how to do it. For example, in a ordering context, orders are considered as overdue if it was placed but not paid within 30 minutes. With Specification pattern, your team can talk with a short but unique term: OverdueOrderSpecification. Imagine the discussion below:

case -1

Business people: I want to find out all overdue orders and ...   Developer: I can do that, it is easy to find all satisfying orders with an overdue order specification and.. 

case -2

Business people: I want to find out all orders which were placed before 30 minutes and still unpaid...   Developer: I can do that, it is easy to filter order from tbl_order where placed_at is less that 30minutes before sysdate.... 

Which one do you prefer?

Usually, we need a DSL handler to parse the dsl, in this case, it may be in the persistence adapter, translates the specification to a query criteria. This dependence (infrastrructure.persistence => domain) does not violates the architecture principal.

class OrderMonitorApplication {     public void alarm() {        // The specification pattern keeps the overdue order ubiquitous language in domain        List<Order> overdueOrders = orderRepository.findBy(new OverdueSpecification());        for (Order order: overdueOrders) {            //notify admin        }     } }  class HibernateOrderRepository implements orderRepository {     public List<Order> findBy(OrderSpecification spec) {         criteria.le("whenPlaced", spec.placedBefore())//returns sysdate - 30         criteria.eq("status", spec.status());//returns WAIT_PAYMENT         return ...     } } 
like image 131
Yugang Zhou Avatar answered Sep 19 '22 15:09

Yugang Zhou


Once I implemented Specification but...

  1. It was based on LINQ and IQueryable.
  2. It used single unified Repository (but as for me it's not bad and I think that it's main reason to use Specification).
  3. It used single model for domain and persistant needs (which I think to be bad).

Repository:

public interface IRepository<TEntity> where TEntity : Entity, IAggregateRoot {     TEntity Get<TKey>(TKey id);      TEntity TryGet<TKey>(TKey id);      void DeleteByKey<TKey>(TKey id);      void Delete(TEntity entity);      void Delete(IEnumerable<TEntity> entities);      IEnumerable<TEntity> List(FilterSpecification<TEntity> specification);      TEntity Single(FilterSpecification<TEntity> specification);              TEntity First(FilterSpecification<TEntity> specification);      TResult Compute<TResult>(ComputationSpecification<TEntity, TResult> specification);      IEnumerable<TEntity> ListAll();      //and some other methods } 

Filter specification:

public abstract class FilterSpecification<TAggregateRoot> where TAggregateRoot : Entity, IAggregateRoot {       public abstract IQueryable<TAggregateRoot> Filter(IQueryable<TAggregateRoot> aggregateRoots);       public static FilterSpecification<TAggregateRoot> CreateByPredicate(Expression<Func<TAggregateRoot, bool>> predicate)      {          return new PredicateFilterSpecification<TAggregateRoot>(predicate);      }             public static FilterSpecification<TAggregateRoot> operator &(FilterSpecification<TAggregateRoot> op1, FilterSpecification<TAggregateRoot> op2)      {          return new CompositeFilterSpecification<TAggregateRoot>(op1, op2);      }               public static FilterSpecification<TAggregateRoot> CreateDummy()      {          return new DummyFilterSpecification<TAggregateRoot>();      }  }   public class CompositeFilterSpecification<TAggregateRoot> : FilterSpecification<TAggregateRoot> where TAggregateRoot : Entity, IAggregateRoot {      private readonly FilterSpecification<TAggregateRoot> _firstOperand;     private readonly FilterSpecification<TAggregateRoot> _secondOperand;      public CompositeFilterSpecification(FilterSpecification<TAggregateRoot> firstOperand, FilterSpecification<TAggregateRoot> secondOperand)     {         _firstOperand = firstOperand;         _secondOperand = secondOperand;     }      public override IQueryable<TAggregateRoot> Filter(IQueryable<TAggregateRoot> aggregateRoots)     {         var operand1Results = _firstOperand.Filter(aggregateRoots);         return _secondOperand.Filter(operand1Results);     } }  public class PredicateFilterSpecification<TAggregateRoot> : FilterSpecification<TAggregateRoot> where TAggregateRoot : Entity, IAggregateRoot {      private readonly Expression<Func<TAggregateRoot, bool>> _predicate;      public PredicateFilterSpecification(Expression<Func<TAggregateRoot, bool>> predicate)     {         _predicate = predicate;     }      public override IQueryable<TAggregateRoot> Filter(IQueryable<TAggregateRoot> aggregateRoots)     {         return aggregateRoots.Where(_predicate);     } } 

Another kind of specification:

public abstract class ComputationSpecification<TAggregateRoot, TResult> where TAggregateRoot : Entity, IAggregateRoot {      public abstract TResult Compute(IQueryable<TAggregateRoot> aggregateRoots);      public static CompositeComputationSpecification<TAggregateRoot, TResult> operator &(FilterSpecification<TAggregateRoot> op1, ComputationSpecification<TAggregateRoot, TResult> op2)     {         return new CompositeComputationSpecification<TAggregateRoot, TResult>(op1, op2);     }  } 

and usages:

OrderRepository.Compute(new MaxInvoiceNumberComputationSpecification()) + 1 PlaceRepository.Single(FilterSpecification<Place>.CreateByPredicate(p => p.Name == placeName)); UnitRepository.Compute(new UnitsAreAvailableForPickingFilterSpecification() & new CheckStockContainsEnoughUnitsOfGivenProductComputatonSpecification(count, product)); 

Custom implementations may look like

public class CheckUnitsOfGivenProductExistOnPlaceComputationSpecification : ComputationSpecification<Unit, bool> {     private readonly Product _product;     private readonly Place _place;      public CheckUnitsOfGivenProductExistOnPlaceComputationSpecification(         Place place,         Product product)     {         _place = place;         _product = product;     }      public override bool Compute(IQueryable<Unit> aggregateRoots)     {         return aggregateRoots.Any(unit => unit.Product == _product && unit.Place == _place);     } } 

Finally, I'm forced to tell that simple Specficiation implementation fits bad according to DDD. You have done great research in this area and it's unlikely that someone proposes something new :). Also, take a look at http://www.sapiensworks.com/blog/ blog.

like image 21
Valentin P. Avatar answered Sep 22 '22 15:09

Valentin P.