CtrlK
BlogDocsLog inGet started
Tessl Logo

igmarin/ruby-core-skills

Curated library of 16 public Ruby AI agent skills: 10 atomic skills (YARD docs, service objects, calculator pattern, API clients, DDD, bug triage, code review, skill routing), 5 process-discipline skills (TDD, refactoring, review, security, test planning), and 1 planning skill (TDD task generation). Zero agents — this is a foundational library consumed by framework-specific tiles like rails-agent-skills and hanakai-yaku.

95

1.05x
Quality

96%

Does it follow best practices?

Impact

95%

1.05x

Average score across 16 eval scenarios

SecuritybySnyk

Passed

No known issues

Overview
Quality
Evals
Security
Files

SKILL.mdskills/ddd/model-domain/

name:
model-domain
license:
MIT
description:
Use when modeling DDD concepts in Ruby: start from domain invariants and ownership before choosing patterns (document each concept with its invariant example, e.g., cancelled→active state transition guard, and patterns to avoid), prefer default Ruby classes over extra abstractions, entity when identity matters, value object when equality by value is correct, aggregate root guards state transitions as single entry point, domain service for behavior spanning multiple entities, application service orchestrates one use case — hand off to test-planning-process and tdd-process before implementation, and only add repository when real persistence boundary exists. Covers mapping of entities, aggregates, value objects, domain services, application services, repositories, and domain events without over-engineering.
metadata:
{"version":"1.0.0","user-invocable":"true","origin":"Extracted from igmarin/rails-agent-skills v5.1.17"}

Model Domain

Quick Reference

DDD conceptRuby defaultAvoid by defaultTypical home
EntityPlain class when persisted identity mattersExtra wrapper object with no added meaningmodels/ or project namespace
Value objectPORO — immutable, equality by valueShoving logic into helpers or primitivesmodels/ or near the domain
Aggregate rootThe class that guards invariants and is the single entry pointSplitting invariants across multiple models/classesmodels/ or project namespace
Domain servicePORO for behavior spanning multiple entitiesArbitrary model chosen just to hold codeservices/
Application serviceOrchestrator for one use caseFat controllers, callback chains, or leaking rulesservices/
RepositoryOnly when a real persistence boundary exists beyond simple data accessRepositories for every queryrepositories/
Domain eventExplicit object when multiple downstream consumers justify itCallback-driven hidden side effectsevents/ or project namespace

HARD-GATE

DO NOT introduce repositories, aggregates, or domain events just to sound "DDD".
DO NOT fight simple Ruby defaults when a normal class or service expresses the domain clearly.
ALWAYS start from domain invariants, ownership, and lifecycle before choosing a pattern.
MODELING OUTPUT IS NOT IMPLEMENTATION: do not include Ruby implementation code,
database migrations, or service bodies unless the user explicitly asks to move from
modeling into build work.

Core Process

Core principle: Model real domain pressure, not textbook DDD vocabulary.

Modeling Order

  1. List domain concepts: Entities, values, policies, workflows, and events from the ubiquitous language.
  2. Identify invariants: Ask: what must always be true after any state change? What breaks if two callers mutate the same state simultaneously? Decide which object or boundary must keep each rule true.
  3. Choose the aggregate entry point: Name the object that guards state transitions and consistency. Ask: is there exactly one place a caller must go to change this state?
  4. Place behavior: Keep behavior on the entity/aggregate when cohesive; extract a domain service only when behavior spans multiple concepts cleanly. Ask: does moving this out clarify ownership, or just spread it?
  5. Pick project homes: Choose the simplest directory location that matches the boundary and repo conventions.
  6. Verify with tests: Hand off to test-planning-process and tdd-process before implementation.

Common Mistakes

MistakeReality
Turning every concept into a serviceMany behaviors belong naturally on entities or value objects
Treating aggregates as folder names onlyAggregates exist to protect invariants, not to look architectural
Adding domain events for one local callbackEvents justify their cost only when multiple downstream consumers exist
Pattern choice justified only with "DDD says so"The reason must be an invariant, ownership boundary, or clear coordination need
Same invariant enforced from multiple unrelated entry pointsSingle aggregate root guards state transitions — one entry point per invariant
New abstractions that increase indirection without clarifying ownershipIf the boundary is unclear after modeling, the abstraction is premature

Extended Resources

  • assets/examples.md — Worked examples for multi-aggregate domains using pure Ruby.
  • assets/modeling_template.md — Blank output template for a single domain concept.

[!NOTE] For Rails-specific DDD mapping (e.g. mapping ActiveRecord associations and migrations), see model-domain in rails-agent-skills when using the Rails pack.

Output Style

For each domain concept, return a compact entry covering:

  1. Domain concept — name from the ubiquitous language
  2. Recommended modeling choice — entity, value object, service, etc.
  3. Suggested file path — file location (e.g., lib/ or services/)
  4. Invariant or ownership reason — the rule that must stay true and the exact object responsible for enforcing it
  5. Patterns to avoid — what not to reach for
  6. Test handoff — first behavior to verify, likely test/spec type, and that implementation is deferred until test-planning-process selects the test and tdd-process writes it
  7. Next skill to chain — e.g. test-planning-process

Inline Example — Order aggregate

Domain concept: Order | Modeling choice: Aggregate root | Suggested file path: lib/orders/order.rb

Invariant: An Order must never transition from cancelled back to active, and its total must always reflect current line items. Order is the single entry point; no external caller may mutate line items or status directly.

Avoid: Do not extract an OrderService just to hold place and cancel — that behavior belongs on the aggregate. Do not introduce OrderRepository unless a non-standard database persistence backend is required.

Test handoff: First behavior — Order#cancel raises when already cancelled. Spec type: unit model spec (spec/orders/order_spec.rb). Implementation deferred until test-planning-process selects this spec and tdd-process writes it.

Next: test-planning-process to select the first failing spec for Order#cancel.

Integration

SkillWhen to chain
define-domain-languageWhen the terms are not clear enough to model yet
review-domain-boundariesWhen the modeling problem is really a context boundary problem
test-planning-processWhen the next step is choosing the best first failing spec

skills

README.md

tile.json