Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
Avoiding the mud - Symfony Live London
Search
Sponsored
·
Ship Features Fearlessly
Turn features on and off without deploys. Used by thousands of Ruby developers.
→
Richard Miller
September 21, 2013
Technology
7
780
Avoiding the mud - Symfony Live London
Richard Miller
September 21, 2013
Tweet
Share
More Decks by Richard Miller
See All by Richard Miller
Introduction to Symfony2 - DrupalCamp London 2014
richardmiller
0
300
Avoiding the mud
richardmiller
22
11k
better searching with elasticsearch - PHPConfPL
richardmiller
2
520
better searching with elasticsearch
richardmiller
2
300
What do I get from the full stack framework?
richardmiller
1
1.6k
Advanced Service Container Utilization
richardmiller
4
1.2k
Dependency Injection and the Symfony2 Service Container
richardmiller
9
13k
Other Decks in Technology
See All in Technology
セキュリティについて学ぶ会 / 2026 01 25 Takamatsu WordPress Meetup
rocketmartue
1
310
日本の85%が使う公共SaaSは、どう育ったのか
taketakekaho
1
230
Embedded SREの終わりを設計する 「なんとなく」から計画的な自立支援へ
sansantech
PRO
3
2.6k
茨城の思い出を振り返る ~CDKのセキュリティを添えて~ / 20260201 Mitsutoshi Matsuo
shift_evolve
PRO
1
360
顧客との商談議事録をみんなで読んで顧客解像度を上げよう
shibayu36
0
270
【Oracle Cloud ウェビナー】[Oracle AI Database + AWS] Oracle Database@AWSで広がるクラウドの新たな選択肢とAI時代のデータ戦略
oracle4engineer
PRO
2
180
What happened to RubyGems and what can we learn?
mikemcquaid
0
310
Agile Leadership Summit Keynote 2026
m_seki
1
650
20260208_第66回 コンピュータビジョン勉強会
keiichiito1978
0
190
AIエージェントを開発しよう!-AgentCore活用の勘所-
yukiogawa
0
180
顧客の言葉を、そのまま信じない勇気
yamatai1212
1
360
2026年、サーバーレスの現在地 -「制約と戦う技術」から「当たり前の実行基盤」へ- /serverless2026
slsops
2
260
Featured
See All Featured
SEOcharity - Dark patterns in SEO and UX: How to avoid them and build a more ethical web
sarafernandez
0
120
Heart Work Chapter 1 - Part 1
lfama
PRO
5
35k
How To Speak Unicorn (iThemes Webinar)
marktimemedia
1
380
Data-driven link building: lessons from a $708K investment (BrightonSEO talk)
szymonslowik
1
920
Scaling GitHub
holman
464
140k
Efficient Content Optimization with Google Search Console & Apps Script
katarinadahlin
PRO
1
330
Agile Actions for Facilitating Distributed Teams - ADO2019
mkilby
0
120
Refactoring Trust on Your Teams (GOTO; Chicago 2020)
rmw
35
3.4k
Exploring anti-patterns in Rails
aemeredith
2
250
The Hidden Cost of Media on the Web [PixelPalooza 2025]
tammyeverts
2
190
A Tale of Four Properties
chriscoyier
162
24k
Building the Perfect Custom Keyboard
takai
2
690
Transcript
AVOIDING THE MUD by Richard Miller http://www.flickr.com/photos/burge5000/22568539/
@mr_r_miller http://www.richardmiller.co.uk
http://www.flickr.com/photos/brostad/8583990904
http://www.flickr.com/photos/dominik99/384027019
http://www.flickr.com/photos/jasmeet/2158002974
Bundles For Everything? http://www.flickr.com/photos/jarkel/2288176
Separate Domain Model From Bundles http://www.flickr.com/photos/30666161@N05/3705814710
HumanResourcesBundle
HumanResourcesBundle HumanResources
HumanResourcesBundle HumanResources
http://www.flickr.com/photos/26582481@N08/3040936355 Enough?
Still Tightly Coupled
AbsenceEntity Request Absence Form Cancel Absence Form Approve Absence Form
Change Absence Type Form
Sensio\HumanResources\Absence\AbsenceEntity: properties: reason: - NotBlank: { groups: [cancellation] } -
Length: { max: 300, groups: [cancellation] }
AbsenceEntity Request Absence Form Cancel Absence Form Approve Absence Form
Change Absence Type Form Request Absence Command Cancel Absence Command Approve Absence Command Change Absence Type Command
What About Templates?
AbsenceEntity upcoming .html .twig details .html .twig calendar .html .twig
unapproved .html .twig
{% if absence.endDate < date() %} past {% else %}
absence.status {% endif %} getDisplayStatus() AbsenceEntity getEndDate() getStatus()
{% if absence.endDate < date() %} past {% else %}
absence.status {% endif %} getDisplayStatus() AbsenceEntity getEndDate() getStatus()
{{ absence.displayStatus }} getDisplayStatus() AbsenceEntity getEndDate() getStatus() getDisplayStatus()
{{ absence.displayStatus }} getDisplayStatus() AbsenceEntity getEndDate() getStatus() getDisplayStatus()
AbsenceEntity Request Absence Form Cancel Absence Form Approve Absence Form
Change Absence Type Form Absence View upcoming .html .twig details .html .twig calendar .html .twig unapproved .html .twig
AbsenceEntity Request Absence Form Cancel Absence Form Approve Absence Form
Change Absence Type Form Absence View upcoming .html .twig details .html .twig calendar .html .twig unapproved .html .twig getStatus()
Unapproved Absence View Calendar View AbsenceEntity Request Absence Form Cancel
Absence Form Approve Absence Form Change Absence Type Form Upcoming Absences View Absence Details View upcoming .html .twig details .html .twig calendar .html .twig unapproved .html .twig
SRP
Behaviour Not State
Request Absence Controller Absence Entity setReason($reason) setStatus(AbsenceEntity::CANCELLED) setCancellationDate(new DateTime()) Cancel
Absence Command getReason()
Request Absence Controller Absence Entity cancel($reason) Cancel Absence Command getReason()
Cancel Absence Command Absence Entity Validate internal consistency Validate user
input
CancelAbsenceComm andHandler Cancel Absence Command Handler AbsenceEntity cancel($reason) handle($command) Request
Absence Controller Absence Entity cancel($reason) Cancel Absence Command
Infrastructure?
Absence Repository (Interface) findById($id) findUpcomingByMember($member) findAllByMember($member) Doctrine ORM Absence Repository
Doctrine ODM Absence Repository Guzzle Client Absence Repository Infrastructure Domain
AbsenceEntity Absence Repository Domain Bundle Service Config Infrastructure Doctrine ORM
Absence Repository Mapping Config
Absence Repository (Interface) findById($id) findUpcomingByMember($member) findAllByMember($member) add(AbsenceEntity $absence)
Events
OnUpdate Event AbsenceEntity Dispatches
OnUpdate Event AbsenceEntity Dispatches
Absence Cancelled Event Cancel Absence Command Handler Dispatches
Absence Cancelled Event In Memory Listener Immediate Action Place In
Messaging Queue Listens
Absence Cancelled Event Flush Listener Listens Absence Approved Event Domain
Infrastructure Listens
Separating Commands From Queries
Command Command Handler Entity Datastore
Datastore View Datastore View Datastore Command Command Handler Entity Event
Datastore View Datastore View Datastore Command Command Handler Entity View
Template Event
Separating Bundles Horizontally
Non-application Specific Code
HumanResources CustomLoggingBundle HumanResourcesBundle
HumanResourcesBundle HumanResources CustomLoggingBundle CustomLogging CustomLoggingBundle CustomLoggingBundle CustomLogging HumanResourcesBundle
HumanResources CustomLoggingBundle CustomLogging CustomLoggingBundle CustomLoggingBundle CustomLogging If it must! HumanResourcesBundle
HumanResources CustomLoggingBundle CustomLogging CustomLoggingBundle CustomLoggingBundle CustomLogging HumanResourcesBundle
Separating By Context
AbsencePlannerBundle AbsencePlanner CustomLoggingBundle CustomLogging CustomLoggingBundle RecruitmentBundle Recruitment
Message Queue/REST AbsencePlannerBundle AbsencePlanner CustomLoggingBundle CustomLogging CustomLoggingBundle RecruitmentBundle Recruitment
CustomLoggingBundle CustomLogging CustomLoggingBundle RecruitmentBundle Recruitment Off the shelf Package Message
Queue/REST Message Queue/REST AbsencePlannerBundle AbsencePlanner
Separate Commonality
AbsencePlannerBundle AbsencePlanner RecruitmentBundle Recruitment AbsencePlannerBundle AbsencePlanner CustomLoggingBundle CustomLogging CustomLoggingBundle RecruitmentBundle
Recruitment User Management
!= Staff Member Entity User Entity
AbsencePlannerBundle AbsencePlanner AbsencePlannerBundle AbsencePlanner CustomLoggingBundle CustomLogging CustomLoggingBundle RecruitmentBundle Recruitment UserManagementBundle
Now?
http://www.flickr.com/photos/mtsofan/8313606885
https://github.com/beberlei/litecqrs-php http://behat.org/
http://williamdurand.fr/ http://verraes.net/#blog http://www.whitewashing.de/
QUESTIONS?