name: domain description: Create a new domain bounded context with aggregates, commands, events, and handlers
Domain Builder
When to use
Use this skill when asked to create a new domain module (bounded context) in the domains/ directory, or to add new aggregates/commands/events to an existing domain.
Reference
Two patterns exist in the codebase:
- Simple (todo): Everything in a single file
lib/{domain}/todo.rb - Structured (crm, ordering, etc.): Separate files for commands, events, aggregates, and handlers
Use the structured pattern for any non-trivial domain. The simple pattern is only for demos.
Step-by-step process
1. Gather requirements
Before writing any code, clarify:
- The domain name (snake_case, e.g.
project_management,ticketing) - The aggregates — what are the core entities? (e.g.
Customer,Deal,Contact) - For each aggregate: what commands can be issued and what events do they produce?
- What business rules does the aggregate enforce? (invariants, state guards)
2. Scaffold the domain directory
Create the directory structure:
domains/{domain_name}/
├── Gemfile
├── Makefile
├── .mutant.yml
├── lib/
│ ├── {domain_name}.rb # Module entry point + Configuration
│ └── {domain_name}/
│ ├── commands/
│ │ └── {command_name}.rb # One file per command
│ ├── events/
│ │ └── {event_name}.rb # One file per event
│ ├── {aggregate_name}.rb # Aggregate root
│ └── {aggregate_name}_service.rb # Command handlers
└── test/
├── test_helper.rb
└── {test_name}_test.rb # One file per test concern
3. Create boilerplate files
Gemfile:
source "https://rubygems.org"
eval_gemfile "../../infra/Gemfile.test"
gem "infra", path: "../../infra"
Makefile:
install:
@bundle install
test:
@bundle exec ruby -e "require \"rake/rake_test_loader\"" test/*_test.rb
mutate:
@RAILS_ENV=test bundle exec mutant run
.PHONY: install test mutate
.mutant.yml:
requires:
- ./test/test_helper
integration: minitest
usage: opensource
coverage_criteria:
process_abort: true
matcher:
subjects:
- {DomainModule}*
ignore:
- {DomainModule}::Configuration#call
4. Write tests first (TDD)
Create test/test_helper.rb:
require "minitest/autorun"
require "mutant/minitest/coverage"
require_relative "../lib/{domain_name}"
module DomainModule
class Test < Infra::InMemoryTest
def before_setup
super
Configuration.new.call(event_store, command_bus)
end
private
def some_helper(id, attribute)
run_command(SomeCommand.new(entity_id: id, attribute: attribute))
end
end
end
Create test files — one per logical concern (e.g. test/registration_test.rb, test/assignment_test.rb):
require_relative "test_helper"
module DomainModule
class SomeTest < Test
cover "DomainModule*"
def test_entity_can_be_created
id = SecureRandom.uuid
create_entity(id, "some value")
expected_event = EntityCreated.new(data: { entity_id: id, attribute: "some value" })
assert_events("DomainModule::Entity$#{id}", expected_event) do
create_entity(id, "some value")
end
end
def test_cannot_create_same_entity_twice
id = SecureRandom.uuid
create_entity(id, "value")
assert_raises(Entity::AlreadyExists) do
create_entity(id, "value")
end
end
private
def create_entity(id, value)
run_command(CreateEntity.new(entity_id: id, attribute: value))
end
end
end
Test conventions:
- Use
cover "DomainModule*"for mutation testing - Use
run_command(...)to dispatch commands (provided byInfra::InMemoryTest) - Use
assert_events(stream, expected_event) { block }to verify events - Use
assert_raises(ErrorClass) { block }for invariant violations - Extract command-calling helpers as private methods
- Test both happy paths and invariant violations (double-creation, invalid state transitions)
5. Create commands
One file per command in lib/{domain_name}/commands/:
module DomainModule
class CreateEntity < Infra::Command
attribute :entity_id, Infra::Types::UUID
attribute :name, Infra::Types::String
alias aggregate_id entity_id
end
end
Command conventions:
- Inherit from
Infra::Command - Use
Infra::Types::UUIDfor UUIDs,Infra::Types::Stringfor strings - Add
alias aggregate_id {entity_id_field}so the handler can usecommand.aggregate_id
6. Create events
One file per event in lib/{domain_name}/events/:
module DomainModule
class EntityCreated < Infra::Event
attribute :entity_id, Infra::Types::UUID
attribute :name, Infra::Types::String
end
end
Event conventions:
- Inherit from
Infra::Event - Events are named in past tense (
Created,Updated,Assigned,Promoted) - Include the aggregate ID and any relevant data
7. Create aggregate root
module DomainModule
class Entity
include AggregateRoot
AlreadyExists = Class.new(StandardError)
NotFound = Class.new(StandardError)
def initialize(id)
@id = id
end
def create(name)
raise AlreadyExists if @created
apply EntityCreated.new(data: { entity_id: @id, name: name })
end
def update_name(name)
raise NotFound unless @created
apply EntityNameUpdated.new(data: { entity_id: @id, name: name })
end
private
on EntityCreated do |event|
@created = true
end
on EntityNameUpdated do |event|
end
end
end
Aggregate conventions:
include AggregateRoot- State tracked via instance variables (
@created,@completed, etc.) - Business rules enforced via
raisebeforeapply on EventClass do |event|blocks update internal state- Events that don't change state still need empty
onblocks - Custom error classes as
Class.new(StandardError)
8. Create command handlers
One handler class per command, grouped in a service file:
module DomainModule
class OnCreateEntity
def initialize(event_store)
@repository = Infra::AggregateRootRepository.new(event_store)
end
def call(command)
@repository.with_aggregate(Entity, command.aggregate_id) do |entity|
entity.create(command.name)
end
end
end
class OnUpdateEntityName
def initialize(event_store)
@repository = Infra::AggregateRootRepository.new(event_store)
end
def call(command)
@repository.with_aggregate(Entity, command.aggregate_id) do |entity|
entity.update_name(command.name)
end
end
end
end
Handler conventions:
- Each handler wraps
@repository.with_aggregate(AggregateClass, id) { |agg| ... } - Handler names describe the action:
OnRegistration,OnSetCustomer,OnPromoteCustomerToVip - All in one
_service.rbfile per aggregate, or separate files for clarity
9. Create module entry point with Configuration
Create lib/{domain_name}.rb:
require "infra"
require_relative "{domain_name}/commands/create_entity"
require_relative "{domain_name}/commands/update_entity_name"
require_relative "{domain_name}/events/entity_created"
require_relative "{domain_name}/events/entity_name_updated"
require_relative "{domain_name}/entity_service"
require_relative "{domain_name}/entity"
module DomainModule
class Configuration
def call(event_store, command_bus)
command_bus.register(CreateEntity, OnCreateEntity.new(event_store))
command_bus.register(UpdateEntityName, OnUpdateEntityName.new(event_store))
end
end
end
Configuration conventions:
require "infra"first, then all domain files- Each command registered with its handler
- Handlers receive
event_storein constructor
10. Run verification
cd domains/{domain_name}
bundle install
make test # All tests pass
make mutate # 100% mutation score
Then from the project root:
make test # Ensure nothing is broken globally
Aggregate design principles
Keep aggregates as small as possible. An aggregate is often a small state machine with two or three states. Adding new methods to an existing aggregate is a smell — it signals a cohesion problem. Before adding a method, ask: "Is this really the same concept, or is it a new aggregate?"
Associations between entities are often aggregates on their own. For example, "Contact assigned to Company" is not a method on Contact — it's a separate ContactCompanyAssignment aggregate with its own lifecycle. The Contact aggregate handles contact registration and attributes. The assignment is a different concern.
Passing strings or "data" into aggregate methods is a smell. Aggregate methods should ideally receive only IDs (UUIDs). If you're passing names, descriptions, or other data, consider whether the aggregate is doing too much. Small smells are acceptable — the goal is to avoid ActiveRecord-like bloat where aggregates accumulate dozens of setter methods.
Don't validate other aggregates from command handlers. Avoid patterns like @event_store.read.stream("Crm::Entity$#{id}").last or raise NotFound in command handlers. This couples the handler to stream naming conventions and makes it responsible for validating external state. Instead, trust the caller (controllers provide valid IDs from read model dropdowns) or model the relationship as its own aggregate that can enforce its own invariants.
Signs an aggregate is too big:
- More than 4-5 command methods
- Instance variables tracking unrelated concerns (e.g.,
@name,@email,@company_id) - Command handlers needing to check other aggregate streams
- The
onblocks are growing in number
Key conventions
- Commands are imperative:
RegisterCustomer,AddTodo,AssignDeal - Events are past tense:
CustomerRegistered,TodoAdded,DealAssigned - Aggregates enforce business rules (invariants) before applying events
- UUIDs for all entity IDs
Infra::Types::UUIDandInfra::Types::Stringfor typed attributes- Test-first TDD, 100% mutation score
- Each domain is fully independent — no cross-domain imports
- Domains only communicate via events (consumed by read models and process managers in apps)