domain-decomposition-api-design-advisor
Domain Decomposition API Design Advisor
Source mapping: Tier 3 specialized skill derived from Kotlin_Spring_Developer_Pipeline.md (SK-25).
Mission
Convert vague product intent into explicit technical boundaries and contracts that can survive implementation and change. Optimize for clear ownership, explicit invariants, and low coupling rather than premature microservice enthusiasm.
Inputs To Gather
- Business goals, user journeys, and success criteria.
- Known constraints: SLA, throughput, latency, audit, privacy, compliance, data retention.
- Existing system boundaries, shared data, and integration points.
- Failure and retry expectations, especially around money, inventory, identity, and messaging.
- Team topology and deployment constraints when they influence boundary choices.
Decomposition Workflow
- Break the feature into use cases and state transitions.
- Identify the core nouns, aggregates, and decision points.
- Identify where consistency must be strong and where eventual consistency is acceptable.
- Separate commands from queries when that improves clarity, not by default.
- Identify external actors and dependencies.
- Decide whether the right boundary is:
- package
- module
- bounded context inside a modular monolith
- separate service
- Design API and event contracts only after the domain boundary is clear.
Boundary Rules
- Prefer a module boundary over a service boundary when independent deployment, ownership, or scaling pressure is weak.
- Shared database tables across service boundaries are usually a warning sign, not a convenience.
- Draw boundaries around invariants and ownership, not around CRUD screens.
- Distinguish reference data sharing from operational write ownership.
- A boundary is only real if dependency direction, data ownership, and failure handling agree.
API Design Rules
- Start from consumer use cases, not only internal data shape.
- Make idempotency, concurrency, versioning, and error semantics explicit.
- Choose synchronous request-response only when latency, consistency, and dependency reliability make it appropriate.
- Prefer additive evolution and stable machine-readable codes.
- Decide whether the API is public, partner-facing, internal synchronous, or asynchronous event-driven. Each has different compatibility expectations.
Kotlin And Modeling Nuances
- Use sealed hierarchies for genuinely closed state machines or domain outcomes.
- Use value classes for domain primitives when they improve clarity and the stack can support them.
- Keep DTOs as transport shapes; do not let them become the entire domain model.
- Use nullability to express meaning, not missing analysis.
Advanced Architecture Traps
- "Microservice by default" often creates distributed transactions, duplicated auth, and fractured observability before it creates value.
- A modular monolith with strong boundaries may be a better target state than several chatty services.
- Event-driven decomposition without clear ownership and replay semantics creates ambiguity, not decoupling.
- If a feature needs read-your-write guarantees, an eventually consistent split may impose hidden UX or support costs.
- Public APIs and internal orchestration APIs should not necessarily look the same.
- ADRs that record only the chosen option are weak. Capture rejected alternatives and why they lost.
Advanced Boundary Nuances
- Reporting boundaries and transactional boundaries are often different. Do not let analytics-driven query convenience dictate write ownership.
- Anti-corruption layers are often cheaper than pretending two bounded contexts share the same ubiquitous language.
- Team ownership, deployment cadence, and support rotation are architecture inputs when boundaries are long-lived. Ignore them only if the code will remain single-team.
- Multi-tenant behavior, data residency, and audit obligations can force a boundary that pure domain language does not reveal immediately.
- Event choreography without a clear owner for recovery and replay becomes shared confusion. If no one owns correction, the boundary is weak.
- Some features deserve process-manager or saga modeling, but only when the business truly spans separate consistency boundaries.
Expert Heuristics
- If two modules change together for every feature, they probably are not separate bounded contexts yet.
- Prefer boundaries that reduce the number of concepts a team must hold in working memory during a single change.
- If an API contract must survive multiple client generations, design for behavioral compatibility, not only field-level compatibility.
- Write the failure story for each boundary. If the team cannot explain how retries, compensation, and partial success work, the design is not finished.
Output Contract
Return these sections:
Problem framing: the use cases, constraints, and unknowns.Proposed boundaries: module or service decomposition and ownership.Consistency map: where transactions, idempotency, and eventual consistency apply.API or event contracts: the principal commands, queries, and error semantics.Tradeoffs: why this boundary shape is better than the main alternatives.ADR outline: decision, options, rationale, and follow-up risks.
Guardrails
- Do not propose microservices where a disciplined module boundary is sufficient.
- Do not ignore non-functional requirements just because the feature narrative sounds simple.
- Do not mistake CRUD decomposition for domain decomposition.
- Do not design APIs before clarifying ownership and invariants.
Quality Bar
A good run of this skill gives the team a boundary model that survives implementation pressure and future change. A bad run produces a plausible architecture diagram with no ownership, no consistency story, and no contract discipline.
More from jetbrains/skills
spring-kotlin-code-review
Review Kotlin + Spring changes for behavioral regressions, transaction and proxy bugs, API and serialization mistakes, persistence risks, security issues, configuration drift, and missing tests. Use when reviewing a PR, diff, patch, or design change where generic style-focused review would miss Spring-specific correctness and operational risks.
4dependency-conflict-resolver
Diagnose and resolve Gradle and Spring classpath conflicts, version drift, and binary incompatibilities in Kotlin applications. Use when `NoSuchMethodError`, `ClassNotFoundException`, linkage errors, duplicate logging bindings, Jackson or Hibernate mismatches, or BOM-versus-explicit-version conflicts appear, and the fix must respect the repository's real version authorities.
3doc
Use when the task involves reading, creating, or editing `.docx` documents, especially when formatting or layout fidelity matters; prefer `python-docx` plus the bundled `scripts/render_docx.py` for visual checks.
3kotlin-spring-proxy-compatibility
Diagnose and prevent Kotlin plus Spring proxy failures around `@Transactional`, `@Cacheable`, `@Async`, method security, retry, configuration proxies, and JPA entity requirements. Use when AOP annotations appear to do nothing, transactional or cache behavior is inconsistent, compiler plugins may be missing, self-invocation is suspected, or Kotlin final-by-default semantics may break Spring behavior.
3ci-cd-containerization-advisor
Design reproducible build, image, and deployment pipelines for Kotlin plus Spring applications, including CI verification, layered containers, rollout safety, and deployment-time migration coordination. Use when creating or improving Dockerfiles, CI workflows, image hardening, Kubernetes manifests, release gates, or deployment strategies for Spring Boot services, especially where build reproducibility and operational safety matter.
3kotlin-idiomatic-refactorer-spring-aware
Refactor Kotlin code toward clearer, more idiomatic design without breaking Spring behavior, serialization, persistence, or public contracts. Use when Java-flavored Kotlin needs cleanup, domain modeling should become more expressive, or boilerplate should be reduced, but the refactoring must remain safe for proxies, Jackson, JPA, configuration binding, and existing tests.
3