Slide 1

Slide 1 text

From CRUD to DDD How did we end up here? Arnout Boks - Dutch PHP Conference - 16 March 2024

Slide 2

Slide 2 text

CRUD Operations on an entity: ● Create ● Read ● Update ● Delete

Slide 3

Slide 3 text

CRUD The database has a central role. The interface (both backend API and UI) are just a thin wrapper.

Slide 4

Slide 4 text

CRUD User Interface edit_recipe.php view_recipe.php Database SQL SQL id: 123 id: 123 title: “foo” ingredients: [...] published: true

Slide 5

Slide 5 text

CRUD with a modern twist User Interface Recipe:edit controller Recipe:view controller Database SQL GET /recipes/123 POST /recipes/123 title: “foo” ingredients: [...] published: true app.php front controller ORM

Slide 6

Slide 6 text

Domain Driven Design (DDD) The domain (the subject area where the software is used) and the business logic therein gets a central role in the application. The rest of the software (the infrastructure) is built around it.

Slide 7

Slide 7 text

Domain Driven Design: ‘the full monty’ User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository RecipePublished event Event store Event bus SQL Recipe event projector ORM Recipe read model repository SQL load & save publish

Slide 8

Slide 8 text

How did we end up here?

Slide 9

Slide 9 text

Before we start ● Steps may be taken in different order ● Steps may be executed differently ○ Interpretation of principles ○ Implementation choices ○ Chosen technology (language, framework, ORM)

Slide 10

Slide 10 text

Before we start New! New! New! Generic, system-wide component Component specific to this feature/use case New possibility, optional addition

Slide 11

Slide 11 text

DDD as an umbrella term ● Ubiquitous language ● Event Sourcing ● Command pattern ● Entity ● CQRS ● Repository ● Aggregate ● Value Object ● Event-driven architecture ● Bounded context ● Command bus ● Anti-corruption layer

Slide 12

Slide 12 text

Recap: our CRUD starting point User Interface Recipe:edit controller Database app.php front controller Recipe:view controller ORM SQL

Slide 13

Slide 13 text

Aggregates An aggregate is a cluster of domain objects that can be treated as a single unit. An aggregate is responsible for maintaining the integrity and business rules within the cluster.

Slide 14

Slide 14 text

Aggregates User Interface Recipe:edit controller Database app.php front controller Recipe:view controller Recipe aggregate Recipe repository ORM SQL load & save edit title: “foo” ingredients: [...] published: true

Slide 15

Slide 15 text

Aggregates User Interface Recipe:edit controller Database app.php front controller Recipe:view controller Recipe aggregate Recipe repository ORM SQL Unit tests load & save edit title: “foo” ingredients: [...] published: true

Slide 16

Slide 16 text

Command pattern Models the possible actions/changes on a domain aggregate, and separates the intent from its execution.

Slide 17

Slide 17 text

Command pattern (1) User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Recipe aggregate Recipe repository ORM SQL load & save publish

Slide 18

Slide 18 text

Command pattern (1) User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Recipe aggregate Recipe repository ORM SQL Recipe:addIngredient controller load & save publish

Slide 19

Slide 19 text

Command pattern (2) User Interface Recipe:publish controller Database app.php front controller Recipe:view controller PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe repository ORM SQL publish load & save

Slide 20

Slide 20 text

Command pattern (2) User Interface Recipe:publish controller Database app.php front controller Recipe:view controller PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe repository ORM SQL recipe:publish console command publish load & save

Slide 21

Slide 21 text

Command bus A ‘one stop shop’ to execute commands. The command bus will find the correct handler for a command and deliver the command to it.

Slide 22

Slide 22 text

Command bus User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe repository ORM SQL

Slide 23

Slide 23 text

Command bus decoration Command bus command handler 2 command handler 1 command handler 3 controller 2 controller 1 console command 1

Slide 24

Slide 24 text

Command bus decoration Transactional command bus Command bus command handler 2 command handler 1 command handler 3 controller 2 controller 1 console command 1

Slide 25

Slide 25 text

Queueing command bus Logging command bus Command bus decoration Transactional command bus Command bus command handler 2 command handler 1 command handler 3 controller 2 controller 1 console command 1

Slide 26

Slide 26 text

Command bus User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe repository ORM SQL

Slide 27

Slide 27 text

Command-Query Responsibility Separation CQRS distinguishes commands (which change the state of the system) from queries (which retrieve information, but make no changes) and treats these separately. In particular, this means that both responsibilities can have separate (optimized/specialized) models.

Slide 28

Slide 28 text

Command-Query Responsibility Separation User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository ORM Recipe read model repository SQL ?

Slide 29

Slide 29 text

Command-Query Responsibility Separation User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository ORM Recipe read model repository SQL Recipe:dashboard controller Recipe dashboard read model repository ? Redis cluster

Slide 30

Slide 30 text

Event-Driven Architecture An Event-Driven Architecture uses events (models of significant things that happened in the past) for the communication between, and decoupling of, components in the system.

Slide 31

Slide 31 text

Event-Driven Architecture User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository RecipePublished event Event bus Recipe event projector ORM Recipe read model repository SQL

Slide 32

Slide 32 text

Event-Driven Architecture User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository RecipePublished event Event bus Recipe event projector ORM Recipe read model repository SQL Notifications event listener ResetRatings process manager

Slide 33

Slide 33 text

Event Sourcing Events are stored, and are the single source of truth in the system. All other information is derived from the events.

Slide 34

Slide 34 text

Event Sourcing: role of the aggregate PublishRecipe command Recipe aggregate RecipePublished event RecipeIngredientAdded event ingredient: Onion quantity: 1 RecipeIngredientAdded event ingredient: Garlic quantity: 2 cloves

Slide 35

Slide 35 text

Event Sourcing: role of the aggregate PublishRecipe command Recipe aggregate HasNoIngredients exception RecipeIngredientAdded event ingredient: Onion quantity: 1 RecipeIngredientRemoved event ingredient: Onion

Slide 36

Slide 36 text

Event Sourcing: role of the aggregate PublishRecipe command Recipe aggregate AlreadyPublished exception RecipeIngredientAdded event ingredient: Onion quantity: 1 RecipeIngredientAdded event ingredient: Garlic quantity: 2 cloves RecipePublished event

Slide 37

Slide 37 text

Event Sourcing User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository RecipePublished event Event store Event bus SQL Recipe event projector ORM Recipe read model repository SQL load & save events

Slide 38

Slide 38 text

Reprojection User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository RecipePublished event Event store Event bus SQL Recipe event projector ORM Recipe read model repository SQL Reprojection system Audit dashboard load & save events

Slide 39

Slide 39 text

Domain Driven Design: ‘the full monty’ User Interface Recipe:publish controller Database app.php front controller Recipe:view controller Command bus PublishRecipe command PublishRecipe command handler Recipe aggregate Recipe write model repository RecipePublished event Event store Event bus SQL Recipe event projector ORM Recipe read model repository SQL publish load & save

Slide 40

Slide 40 text

What did we gain? ● Business domain logic at the core of our code ● Decoupling ● Testability ● Commands conveying clear intention ● Command transactionality ● Command logging ● Command queueing ● Specialized read models ● Specialized data stores ● Managers for distributed business processes ● Notifications ● Auditability ● Reprojections for data migrations and new read models

Slide 41

Slide 41 text

Make informed choices for your architecture, based on your needs.

Slide 42

Slide 42 text

Thank you! - Any questions? Arnout Boks @aboks@phpc.social @arnoutboks https://aboks.github.io https://www.moxio.com Feedback https://joind.in/talk/565f8 We’re hiring!