Xavier Bruhiere

Domain Driven Design and event sourcing

April 18, 2018 (6y ago)2 views

Abstract: I didn’t think about it but actually I believe the perspective of “cloud” on this matter makes it very interesting : how to apply progressively DDD ideas to cloud architectures and why. This is a new and trendy paradigm and I met many CTOs whos concerns were how to make time for integrating it into their legacy codebase. Having microservices and an event-based design makes it possible and safer so I can develop on that : 1. What is DDD applicable to the cloud and why we want it in our stack 2. Strategy and micro services to the rescue 3. Events and event sourcing to go further


TDD, Test Driven Development is quite a thing among developers. On one hand most of us acknowledge the beauty of the idea and thrive to use it here and there when new projects pop up with good intentions. But on the other hand this is one of the first thing we cut when business requirements, lazyness, or prototyping spirit take over the initial plan. I guess we all have some kind of partially adapted version of TDD at some place in our repository.

So what are we going to talk about here ? What does DDD stand for ? A 15th évolution of some kind of software pattern that will enforce rigorous code while keeping developers happy and productive ? After all DDD is Domain Driven Development, quite the same look and feel.

I understand DDD like a set of strategies and thinking tools that can help designing system and manage complexities. You can learn a lot more online and in this article we won't dive too deep into the theory and roots, although it will remain mostly a high level reflexion. Instead I want to develop how interesting it is when architecting cloud applications. We will try to keep it concrete, talk about integration in existing stacks and especially how it can help building powerful event sourcing systems.

Why it's worth it

I'm no expert in Domain Driven Development but I tinkered enough with it to make my own idea and take the points I thought interesting to include into my system design toolbox. Basically DDD tries to manage technical complexity by:

  • Minimizing technical debt - the accidental complexity various factors brought into the project. It is often a necessary evil that rushing to MVP (Minimum Viable Product) or cutting corners to win market share grow projects into big monolith of mysterious mechanisms (what DDD calls _Big Balls Of Mud).
  • Controlling inherent problem complexity - the inherent difficulty of solving your specific business case. Like Paul Graham says, you are probably (hopefully) trying to solve something hard enough that your competitors didn't figure out how to outrun you on the market.

How ? Martin Fowler sums it well enough :

DDD is about designing software based on models of the underlying domain. A model acts as a UbiquitousLanguage to help communication between software developers and domain experts. It also acts as the conceptual foundation for the design of the software itself - how it's broken down into objects or functions.

Thinking about software this way brings a lot of advantages indeed. You can (you need) to explicitely bound context so that teams can solve specific problems with specific solutions.

  • It becomes simpler : big problems are cut into more manageable one you can focus on.
  • Everyone go faster, especially on the long term since they can go parallel and progress with a clearer understanding of the problems' context.

Ideally of course but we will discuss nuances later. All those details let us recognise the link with cloud applciations. As we said DDD provides us with strategies for modelising problems and design architectural solutions that we can implement with code and functions. But I do believe the principles hold for service architectures. Bringing the right people together to think about what domains need to be modelled and how they interface with each other are a great playbook for designing services topology.

It is also a good proxy to confirm during development that we serve real business needs. All too often engineers fall for beautiful architectures where quicker and less exciting approaches would have been enough. Defining applications context and usage with other characters and in the perspective of real life models is certainly a good reality check when we dive too deep into the technical details.

Micro-services can help

Unfortunately I met a lot of engineering leaders or DDD enthousiasts that acknowledge the potential but didn't know how to bring it to their stack. As Eric Evans stated, it is a costly and invasive process and one really need to both justify the need and make it right. I will let you judge if your domains need this level of complexity management and if your software needs those patterns to avoid being quickly brittle. And of course there are probably other strategies but I did witnessed a common pattern among those who took the step to make it part of their stack : one service at the time.

It may sound obvious but it just makes sense. First follow DDD advises : gather people and define your domains. You will probably find opportunities for new services and then you can design those services in a DDD fashion. Simple as that. Yet powerful because you can recognise all the agile benefits. If you start with only one service it makes the process of integration easier to learn for newcomers and easier to iterate on and tweak to the way you already work. This is also easier to reverse. You still take the risk of calling it off and regret having lost quite a lot of time, but chances are most of the code will still be useable and the knowledge gained to identify domains still useful.

What's more, all those patterns and process details will be hidden behind some kind of remote interface. Being a CRUD/REST API or a GraphQL endpoint, the other parts of your stack should be light to modify. And as you design code following DDD practice and decouple clients from core logic, it will become even simpler.

This is especially true if micro-services are already in place since you can recognize the same advantages : smaller and better bounded components that we can easily modify as long as the other components know how to interface with them. This notion of interface is key. It is directly linked to the importance of good bounded contexts in DDD and to good decoupling of services in micro-services. That is one of the great reason I think it makes sense to integrate them together : they help each other.

It is worth noting that having already in place such architecture doesn't necessarily mean it will be easier to bring DDD on the table. I can think of three cases :

  • Micro services in place. The good news is you probably don't need to develop all the bolts and details needed for micro-services to work : service discovery, log management, deployment process, … But bad services topology is a great technical debt and may be harder to fix since you might be forced to move around features between pieces of code that didn't even run on the same servers.
  • Monolith in place : probably one of the most classic startup-case in 2017. You are certainly dying from moving brittle code into it's own service and DDD will help you a lot. But keep in mind that micro-services come at a price. Google develops on only one repository because the coordination between teams and the consistency between their applications are really different. Yet engineering is about trade-off and DDD gives good reasons (technical and business wise) for switching.
  • Monolith in place (bis) : new features are probably still coming in so get a team together and let them design the new service as a squad. All the isolation will also force to clearly define the contexts, the scope, the models and everything needed to get at the end a new service that they can actually present following the principles we mentionned. This is an excellent opportunity to assess what they gained doing so, if it worths it and how to improve it.

It might be a good transition to talk about teams and business. In real life all of this comes to serve a business and since DDD tries to conciliate languages of all department we can't skip this detail of course. What makes the idea really interesting, in my own opinion, is that it could solve a lot of satellite problems. Communication in companies, like culture, is hard to get right. Having a common framework to make it easier between departments to talk about the business issues is really valuable. Engineering can better understand why their code make sense and how to prioritise. Product owners have an easier time doing the interface. Sales and executives also better understand where the IT is on the roadmap and what are the challenges, the expectations.

So spending time to get developers to embrace this philosophy goes further than just following a new software pattern or strategy. It could help them better communicate in the company. And it will also give them a great set of tools to think about solving problems. All the trade offs of micro-services and how to model problems to solve come into a new light and a new vocabulary under DDD.

We insisted a lot on why we need to use it. Let's get a little fancy and push the idea a little further, in a less obvious and generic area : event sourcing.

Match in heaven with Event Sourcing

I will fly faster over this one so I can just reach your curiosity and may be encourage you to investigate further the opportunity.

Event sourcing is a strategic pattern that enforce thinking your application as a series of events you need to store any time something changes the state of your world. This way one can review, understand, replay, debug and do whatever is required on the history of events. It is more and more discussed today as data streaming and state management become relevant to modern development. Read this if you want to spend well the next hour and get a good understanding of all the implication and benefits. But this is the kind of idea that powers relational databases or companies like Linkedin.

This is hard to implement but the kind of architecture and tools possible on top of it makes it worth the effort. And I found out that DDD and everything we said can help. If so far you implemented micro-services that solve a specific problem by defining clear interfaces, vocabularies and bounded contexts, you are in a very good spot to look into Event sourcing.

The decoupling let you reason about data flow in a clear manner and you can probably organize it as events triggering actions. This is often implemented as task queues so in this case you can get started by ensuring you are storing all the original events and may be for example stop polling for changes and instead waiting for inputs. As a side note it fits perfectly the kind of values lambda architectures offer.

Then the consistent vocabulary defined will help you define what those events actually are. Technical constraints will come into play (like caching database queries or limiting connections bottleneck) but you will think about the problem with the right business goal and context in mind. Then it will be much more future proof. Basing your reasonning on your core value that remain the same throughout the product development makes for a consistent evolution.

The integration is also safer for the same reasons micro-services decouple components. There are a lot of benefits and you will have an easier time managing teams and building tools on top on architectures that are close to business knowledge and natural to think about for everyone involved in the project.

Conclusion

This was a very theoric post where I essentially reflected on the past months spent at my current company Kpler working on this kind of approaches. We were very successful at implementing pieces and bits, here and there, of DDD and Event Sourcing where we needed to meet strong requirements in term of data quality and load. It wasn't easy of course. We had to educate the team on the new projects, justify the additional time spent on experimentations and failures. But out of it came stronger services with great flexibility for integration in our stack and future features.

So I hope it gives you the curiousity to investigate and experiment on your own. Being a side project or a new service, you will certainely gain great insights for your future projects by tinkering with those advanced patterns. And given the time we spent at meetups and events talking with other companies, I am confident your teams and business will share this output.