Most developers have never seen a domain model, only a data model.
DDD EU 2017
Most developers that we talk to about architecture have a nagging sense that things could be better. They’re often trying to rescue a system that has gone wrong somehow, and trying to put some structure back into a ball of mud. They know that their business logic shouldn’t be spread all over the place, but they’ve no idea how to fix it.
We’ve found that many developers, when asked to design a new system, will immediately start to build a database schema, with the object model treated as an afterthought. This is where it all starts to go wrong. Instead, behavior should come first, and drive our storage requirements.
After all, our customers don’t care about the data model. They care about what the system does, otherwise they’d just use a spreadsheet.
In this first part we’ll look at how to build a rich object model through TDD (in Chapter 1), and then we’ll see how to keep that model decoupled from technical concerns. We’ll see how to build persistence-agnostic code and how to create stable APIs around our domain so that we can refactor aggressively.
To do that, we’ll look at four key design patterns:
-
The Repository pattern, an abstraction over the idea of persistent storage
-
A Service Layer that clearly defines where our use-cases begin and end
-
The Unit of Work pattern to provide atomic operations
-
And the Aggregate pattern to enforce the integrity of our data.
If you’d like a picture of where we’re going, take a look at A component diagram for our app at the end of Part 1, but don’t worry if none of it makes any sense yet! We’ll introduce each box one by one.
We’ll also take a little time out to talk about coupling and abstractions, illustrating the discussion with a simple example that shows how and why we choose our abstractions.
Several of the appendices are further explorations of the content from Part 1:
-
[appendix_project_structure] is a write-up of the infrastructure for our example code: how we build and run the docker images, where we manage configuration info, how we run different types of tests.
-
[appendix_csvs] is a "the proof is in the pudding" kind of chapter, showing how easy it is to swap out our entire infrastructure — the flask API, the ORM and postgres, for a totally different I/O model involving a CLI and CSVs.
-
Finally, [appendix_django] may be of interest if you’re wondering how these patterns might look if using Django, instead of Flask+SQLAlchemy