CtrlK
BlogDocsLog inGet started
Tessl Logo

igmarin/rails-agent-skills

Curated library of 42 public AI agent skills for Ruby on Rails development, plus 5 callable workflow skills. Organized by category: planning, testing, code-quality, ddd, engines, infrastructure, api, patterns, context, orchestration, and workflows. Covers code review, architecture, security, testing (RSpec), engines, service objects, DDD patterns, and TDD automation.

96

Quality

96%

Does it follow best practices?

Impact

No eval scenarios have been run

SecuritybySnyk

Risky

Do not use without reviewing

Overview
Quality
Evals
Security
Files

SKILL.mdskills/ddd/model-domain/

name:
model-domain
license:
MIT
description:
Use when modeling Domain-Driven Design concepts in a Ruby on Rails codebase. Covers Rails-first mapping of entities, aggregates, value objects, domain services, application services, repositories, and domain events without over-engineering or fighting Rails conventions.
metadata:
{"version":"1.0.0","user-invocable":"true"}

Model Domain

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

HARD-GATE

DO NOT introduce repositories, aggregates, or domain events just to sound "DDD".
DO NOT fight Rails defaults when a normal model or service expresses the domain clearly.
ALWAYS start from domain invariants, ownership, and lifecycle before choosing a pattern.

Modeling Order

  1. List domain concepts: Entities, values, policies, workflows, and events from the ubiquitous language.
  2. Identify invariants: 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.
  4. Place behavior: Keep behavior on the entity/aggregate when cohesive; extract a domain service only when behavior spans multiple concepts cleanly.
  5. Pick Rails homes: Choose the simplest location that matches the boundary and repo conventions.
  6. Verify with tests: Hand off to plan-tests and write-tests before implementation.

Rails-First Mapping

DDD conceptRails-first defaultAvoid by defaultTypical home
EntityActiveRecord model when persisted identity mattersExtra wrapper object with no added meaningapp/models/
Value objectPORO — immutable, equality by valueShoving logic into helpers or primitivesapp/models/ or near the domain
Aggregate rootThe model that guards invariants and is the single entry pointSplitting invariants across multiple modelsapp/models/
Domain servicePORO for behavior spanning multiple entitiesArbitrary model chosen just to hold codeapp/services/
Application serviceOrchestrator for one use caseFat controller or callback chainsapp/services/
RepositoryOnly when a real persistence boundary exists beyond ActiveRecordRepositories for every queryapp/repositories/ (rare)
Domain eventExplicit object when multiple downstream consumers justify itCallback-driven hidden side effectsapp/events/ or project namespace

Output Style

When using this skill, return for each domain concept:

  1. Domain concept — name from the ubiquitous language
  2. Recommended modeling choice — entity, value object, service, etc.
  3. Suggested Rails home — file path
  4. Invariant or ownership reason — why this boundary
  5. Patterns to avoid — what not to reach for
  6. Next skill to chaingenerate-tasks, plan-tests, etc.

Examples

Value Object — Money (Quick Reference)

# app/models/money.rb
class Money
  attr_reader :amount_cents, :currency
  def initialize(amount_cents, currency = "USD")
    @amount_cents, @currency = Integer(amount_cents), currency.upcase.freeze
    freeze
  end
  def ==(other) = other.is_a?(Money) && amount_cents == other.amount_cents && currency == other.currency
end

See assets/examples.md for detailed Application Service and Domain Event examples.

Common Mistakes

MistakeReality
Turning every concept into a serviceMany behaviors belong naturally on entities or value objects
Creating repositories for all reads and writesActiveRecord already provides a strong default persistence boundary
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

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
generate-tasksAfter the tactical design is clear and ready for implementation planning
plan-testsWhen the next step is choosing the best first failing spec
apply-code-conventionsWhen validating the modeling choice against Rails simplicity and repo conventions

Assets

skills

README.md

tile.json