MCP HubMCP Hub
스킬 목록으로 돌아가기

domain-driven-design

wondelai
업데이트됨 2 days ago
7 조회
1,096
121
1,096
GitHub에서 보기
디자인aidesign

정보

이 스킬은 개발자들이 경계 컨텍스트, 애그리거트, 유비쿼터스 언어와 같은 DDD 개념을 사용하여 비즈니스 도메인을 중심으로 소프트웨어를 모델링하도록 돕습니다. 도메인 모델링, 마이크로서비스 경계 설정, 또는 코드를 비즈니스 프로세스에 맞추는 논의가 있을 때 발동됩니다. 이 스킬은 엔티티 대 값 객체, 도메인 이벤트, 컨텍스트 매핑 전략을 포함한 핵심 DDD 패턴을 다룹니다.

빠른 설치

Claude Code

추천
기본
npx skills add wondelai/skills -a claude-code
플러그인 명령대체
/plugin add https://github.com/wondelai/skills
Git 클론대체
git clone https://github.com/wondelai/skills.git ~/.claude/skills/domain-driven-design

Claude Code에서 이 명령을 복사하여 붙여넣어 스킬을 설치하세요

문서

Domain-Driven Design Framework

Framework for tackling software complexity by modeling code around the business domain. Based on a fundamental truth: the greatest risk in software is not technical failure -- it is building a model that does not reflect how the business actually works.

Core Principle

The model is the code; the code is the model. Software should embody a deep, shared understanding of the business domain. When domain experts and developers speak the same language and that language is directly expressed in the codebase, complexity becomes manageable, requirements are captured precisely, and the system evolves gracefully as the business changes.

Scoring

Goal: 10/10. When reviewing or creating domain models, rate them 0-10 based on adherence to the principles below. A 10/10 means full alignment with all guidelines; lower scores indicate gaps to address. Always provide the current score and specific improvements needed to reach 10/10.

Framework

1. Ubiquitous Language

Core concept: A shared, rigorous language between developers and domain experts that is used consistently in conversation, documentation, and code. When the language changes, the code changes. When the code reveals awkward naming, the language is refined.

Why it works: Ambiguity is the root cause of most modeling failures. When a developer says "order" and a domain expert means "purchase request," bugs are inevitable. A ubiquitous language forces alignment so that every class, method, and variable name maps to a concept the business recognizes and validates.

Key insights:

  • The language is not a glossary bolted on after the fact -- it emerges from deep collaboration
  • If a concept is hard to name, the model is likely wrong; naming difficulty is a design signal
  • Code that uses technical jargon instead of domain terms (e.g., DataProcessor vs. ClaimAdjudicator) hides domain logic
  • Language must be enforced in code: class names, method names, event names, module names
  • Different bounded contexts may use the same word with different meanings -- and that is fine

Code applications:

ContextPatternExample
Class namingName classes after domain conceptsLoanApplication, not RequestHandler
Method namingUse verbs the business usespolicy.underwrite(), not policy.process()
Event namingPast-tense domain actionsClaimSubmitted, not DataSaved
Module structureOrganize by domain conceptshipping/, billing/, not controllers/, services/
Code reviewReject technical-only namesFlag Manager, Helper, Processor, Utils as naming smells

See: references/ubiquitous-language.md

2. Bounded Contexts and Context Mapping

Core concept: A bounded context is an explicit boundary within which a particular domain model is defined and applicable. The same word (e.g., "Customer") can mean different things in different contexts. Context maps define the relationships and translation strategies between bounded contexts.

Why it works: Large systems that try to maintain a single unified model inevitably collapse into inconsistency. Bounded contexts accept that different parts of the business have different models and make the boundaries explicit. Context maps then manage integration so that each context preserves its internal consistency.

Key insights:

  • A bounded context is not a microservice -- it is a linguistic and model boundary that may contain multiple services
  • Context boundaries often align with team boundaries (Conway's Law)
  • The nine context mapping patterns describe political and technical relationships between teams
  • Anti-Corruption Layer is the most important defensive pattern -- never let a foreign model leak into your core domain
  • Shared Kernel is dangerous: it couples two teams and should be small and explicitly governed
  • Start by mapping what exists (Big Ball of Mud), then define target boundaries

Code applications:

ContextPatternExample
Service integrationAnti-Corruption LayerTranslate external API responses into your domain objects at the boundary
Team collaborationShared KernelTwo teams co-own a small Money value object library
Legacy migrationConformist / ACLWrap legacy system behind an adapter that speaks your domain language
API designOpen Host Service + Published LanguageExpose a well-documented REST API with a canonical schema
Module boundariesSeparate packages per contextmyapp.shipping and myapp.billing packages with explicit translation

See: references/bounded-contexts.md

3. Entities, Value Objects, and Aggregates

Core concept: Entities have identity that persists across state changes. Value Objects are defined entirely by their attributes and are immutable. Aggregates are clusters of entities and value objects with a single root entity that enforces consistency boundaries.

Why it works: Without these distinctions, systems treat everything as a mutable, identity-bearing object with database-level relationships, leading to tangled state, inconsistent updates, and fragile concurrency. Aggregates draw a consistency boundary: everything inside is guaranteed consistent; everything outside is eventually consistent.

Key insights:

  • Entity: "Am I the same thing even if all my attributes change?" (a person changes name, address, job -- still the same person)
  • Value Object: "Am I defined only by my attributes?" (a $10 bill is interchangeable with any other $10 bill)
  • Most things in a domain model should be Value Objects, not Entities -- prefer immutability
  • Aggregate Root is the single entry point: external objects may only hold references to the root
  • Keep aggregates small -- one root entity plus a minimal cluster of closely related objects
  • Reference other aggregates by ID, not by direct object reference
  • Design for eventual consistency between aggregates; immediate consistency only within an aggregate

Code applications:

ContextPatternExample
Identity trackingEntity with IDOrder identified by orderId, survives state changes
Immutable attributesValue ObjectAddress(street, city, zip) -- replace, never mutate
Consistency boundaryAggregate RootOrder is root; OrderLine items exist only through it
Cross-aggregate referenceReference by IDOrder stores customerId, not a Customer object
Concurrency controlOptimistic locking on rootVersion field on Order; conflict if two edits race

See: references/building-blocks.md

4. Domain Events

Core concept: A domain event captures something that happened in the domain that domain experts care about. Events are named in past tense (OrderPlaced, PaymentReceived) and represent facts that have already occurred.

Why it works: Domain events decouple the cause from the effect. When OrderPlaced is published, the shipping context, billing context, and notification context can each react independently without the ordering context knowing about any of them. This reduces coupling, enables eventual consistency, and creates a natural audit trail.

Key insights:

  • Name events in past tense: something that happened, not something that should happen
  • Events are immutable facts -- once published, they cannot be changed or retracted
  • Domain events differ from integration events: domain events are internal to a bounded context; integration events cross boundaries
  • Events enable temporal decoupling: the producer does not wait for the consumer
  • Event sourcing stores the full history of events as the source of truth, deriving current state by replaying them
  • Not every state change needs an event -- only publish events that the domain cares about

Code applications:

ContextPatternExample
State transitionsRaise event on domain actionorder.place() raises OrderPlaced event
Cross-context integrationPublish integration eventOrderPlaced triggers ShippingLabelRequested in shipping context
Audit trailStore events as historyEvent log: OrderPlaced -> PaymentReceived -> OrderShipped
Eventual consistencyAsync event handlersInventoryReserved handler updates stock asynchronously after OrderPlaced
Event sourcingRebuild state from eventsReplay all Account* events to derive current account balance

See: references/domain-events.md

5. Repositories and Factories

Core concept: Repositories provide the illusion of an in-memory collection of domain objects, hiding persistence details. Factories encapsulate complex object creation logic, ensuring that aggregates are always created in a valid state.

Why it works: Domain logic should never depend on how objects are stored or constructed. Repositories abstract away SQL, ORMs, and data access so that domain code reads like business logic. Factories ensure that invariants are satisfied from the moment an aggregate is born, preventing invalid objects from ever existing.

Key insights:

  • A Repository interface belongs in the domain layer; its implementation belongs in infrastructure
  • Repository methods should speak the ubiquitous language: findPendingOrders(), not getByStatusCode(3)
  • Collection-oriented repositories mimic add/remove; persistence-oriented repositories use save
  • Factories are warranted when object creation involves complex rules, conditional logic, or assembling multiple parts
  • Simple creation (a Value Object with two fields) does not need a factory -- a constructor suffices
  • The Specification pattern encapsulates query criteria as domain objects: OverdueInvoiceSpecification

Code applications:

ContextPatternExample
Data access abstractionRepository interfaceOrderRepository.findByCustomer(customerId) in domain layer
Complex creationFactory methodOrder.createFromQuote(quote) validates and assembles from a Quote aggregate
Query encapsulationSpecificationspec = OverdueBy(days=30); repo.findMatching(spec)
ReconstitutionRepository loads aggregateRepository assembles Order + OrderLines from DB rows into a complete aggregate
Ports and adaptersInterface in domain, impl in infrainterface OrderRepository in domain; PostgresOrderRepository in infrastructure

See: references/repositories-factories.md

6. Strategic Design and Distillation

Core concept: Not all parts of a system are equally important. Strategic design identifies the Core Domain -- the part that provides competitive advantage -- and distinguishes it from Supporting Subdomains (necessary but not differentiating) and Generic Subdomains (commodity, buy or use off-the-shelf).

Why it works: Teams that apply the same rigor to every module spread their best talent thin and over-engineer commodity functionality. By identifying the Core Domain, organizations invest their best developers, deepest modeling, and most careful design where it matters most, while using simpler approaches or third-party solutions elsewhere.

Key insights:

  • Core Domain: where competitive advantage lives; invest your best people and deepest modeling here
  • Supporting Subdomain: necessary for the business but not a differentiator; build it, but don't over-engineer
  • Generic Subdomain: commodity functionality (authentication, email, payments); buy or use open-source
  • Domain distillation extracts and highlights the Core Domain from the surrounding complexity
  • A Domain Vision Statement is a short document (one page) describing the Core Domain's value proposition
  • The Highlighted Core marks the most critical parts of the model so they receive the most attention
  • Revisit what is "core" as the business evolves -- today's differentiator may become tomorrow's commodity

Code applications:

ContextPatternExample
Build vs. buy decisionClassify subdomain typeBuild custom pricing engine (core); use Stripe for payments (generic)
Team allocationBest developers on Core DomainSenior engineers model the underwriting rules; juniors integrate the email service
Code organizationSeparate core from genericdomain/pricing/ (deep model) vs. infrastructure/email/ (thin adapter)
SimplificationDistill core conceptsExtract a PolicyRatingEngine from a monolithic InsuranceService
DocumentationDomain Vision StatementOne-page doc: "Our competitive advantage is real-time risk scoring using..."

See: references/strategic-design.md

Common Mistakes

MistakeWhy It FailsFix
Using technical names instead of domain languageDomain logic is hidden behind DataManager and ProcessorService; experts cannot validate the modelRename to domain terms: ClaimAdjudicator, PolicyUnderwriter; if no domain term exists, the concept may be wrong
One model to rule them allA single Customer class serving billing, shipping, and marketing becomes bloated and contradictoryDefine bounded contexts; each context gets its own Customer model with only the attributes it needs
Giant aggregates with many nested entitiesConcurrency conflicts, slow loads, transactional bottlenecksKeep aggregates small; reference other aggregates by ID; use eventual consistency between aggregates
Anemic domain model (all logic in services)Domain objects are data bags; business rules scatter across service classes; duplication and inconsistencyMove behavior into entities and value objects; services only orchestrate, never contain domain logic
No Anti-Corruption Layer at integration pointsForeign models leak into your domain; your code becomes coupled to external schemas and namingWrap every external system behind a translation layer that converts to your ubiquitous language
Treating bounded contexts as microservicesPremature service extraction; distributed system complexity without the benefitA bounded context is a model boundary, not a deployment unit; start with modules in a monolith
Skipping domain expert collaborationDevelopers invent a model that does not match business reality; expensive reworkRegular modeling sessions with domain experts; refine the model until experts say "yes, that is how it works"

Quick Diagnostic

QuestionIf NoAction
Can a domain expert read your class names and understand them?Code uses technical jargon instead of domain languageRename classes, methods, and events to use ubiquitous language
Are bounded context boundaries explicitly defined?Models bleed across boundaries; the same term means different thingsDraw a context map; define explicit boundaries and translation strategies
Are aggregates small (one root + minimal cluster)?Aggregates are large, slow, and have concurrency issuesBreak into smaller aggregates; reference by ID; accept eventual consistency
Do domain objects contain behavior, not just data?Anemic model; logic scattered in service classesMove business rules into entities and value objects
Are domain events used for cross-aggregate communication?Tight coupling between aggregates; synchronous chainsIntroduce domain events; let aggregates react to events asynchronously
Is there an Anti-Corruption Layer at every external integration?Foreign models pollute your domainAdd a translation layer at each integration boundary
Have you identified which subdomain is core?Equal effort on everything; best talent spread thinClassify subdomains; focus deep modeling on the Core Domain

Reference Files

  • ubiquitous-language.md: Building a shared language, glossary maintenance, naming in code, language evolution
  • bounded-contexts.md: Context boundaries, nine mapping patterns, team relationships, integration strategies
  • building-blocks.md: Entities, Value Objects, Aggregates, aggregate design rules, consistency boundaries
  • domain-events.md: Event naming, event sourcing, event-driven architecture, integration events
  • repositories-factories.md: Repository pattern, Factory pattern, Specification pattern, ports and adapters
  • strategic-design.md: Core Domain, Generic and Supporting Subdomains, distillation, build vs. buy

Further Reading

This skill is based on the Domain-Driven Design methodology developed by Eric Evans. For the complete methodology, patterns, and deeper insights, read the original book:

About the Author

Eric Evans is a software design consultant and the originator of Domain-Driven Design. He has worked on large-scale systems in industries including finance, insurance, and logistics, where he developed the patterns and practices that became DDD. His 2003 book Domain-Driven Design: Tackling Complexity in the Heart of Software is widely regarded as one of the most influential software architecture books ever written. Evans founded Domain Language, a consulting firm that helps teams apply DDD to complex software projects. He is a frequent keynote speaker at software conferences worldwide and continues to refine and evolve DDD concepts through workshops, community engagement, and collaboration with practitioners. His work has shaped modern approaches to microservices, event sourcing, and strategic software design.

GitHub 저장소

wondelai/skills
경로: domain-driven-design
0
agent-skillsai-skillsclaude-codeclaude-code-marketplaceclaude-code-pluginclaude-code-skills

연관 스킬

executing-plans

디자인

executing-plans 스킬은 검토 체크포인트가 포함된 통제된 배치로 실행할 완전한 구현 계획이 있을 때 사용합니다. 이 스킬은 계획을 불러와 비판적으로 검토한 후, 소규모 배치(기본값 3개 작업)로 작업을 실행하면서 각 배치 사이에 진행 상황을 아키텍트 검토를 위해 보고합니다. 이를 통해 내재된 품질 관리 체크포인트를 갖춘 체계적인 구현이 보장됩니다.

스킬 보기

requesting-code-review

디자인

이 스킬은 코드 변경 사항을 요구 사항에 따라 분석하기 위해 코드 리뷰어 하위 에이전트를 호출합니다. 작업 완료 후, 주요 기능 구현 후, 또는 메인 브랜치에 병합하기 전에 사용해야 합니다. 이 리뷰는 현재 구현체와 원래 계획을 비교하여 문제를 조기에 발견하는 데 도움이 됩니다.

스킬 보기

connect-mcp-server

디자인

이 스킬은 개발자들이 HTTP, stdio 또는 SSE 전송 방식을 통해 MCP 서버를 Claude Code에 연결하는 포괄적인 가이드를 제공합니다. GitHub, Notion 및 사용자 정의 API와 같은 외부 서비스를 통합하기 위한 설치, 구성, 인증 및 보안을 다룹니다. MCP 통합 설정, 외부 도구 구성 또는 Claude의 모델 컨텍스트 프로토콜 작업 시 활용하세요.

스킬 보기

web-cli-teleport

디자인

이 스킬은 작업 분석을 기반으로 개발자가 Claude Code 웹 인터페이스와 CLI 인터페이스 중 선택할 수 있도록 돕고, 두 환경 간 원활한 세션 텔레포트를 가능하게 합니다. 웹, CLI 또는 모바일 환경 전환 시 세션 상태와 컨텍스트를 관리하여 워크플로를 최적화합니다. 다양한 단계에서 서로 다른 도구가 필요한 복잡한 프로젝트에 사용하세요.

스킬 보기