dhh-rails-style
npx machina-cli add skill everyinc/compound-engineering-plugin/dhh-rails-style --openclaw<essential_principles>
Core Philosophy
"The best code is the code you don't write. The second best is the code that's obviously correct."
Vanilla Rails is plenty:
- Rich domain models over service objects
- CRUD controllers over custom actions
- Concerns for horizontal code sharing
- Records as state instead of boolean columns
- Database-backed everything (no Redis)
- Build solutions before reaching for gems
What they deliberately avoid:
- devise (custom ~150-line auth instead)
- pundit/cancancan (simple role checks in models)
- sidekiq (Solid Queue uses database)
- redis (database for everything)
- view_component (partials work fine)
- GraphQL (REST with Turbo sufficient)
- factory_bot (fixtures are simpler)
- rspec (Minitest ships with Rails)
- Tailwind (native CSS with layers)
Development Philosophy:
- Ship, Validate, Refine - prototype-quality code to production to learn
- Fix root causes, not symptoms
- Write-time operations over read-time computations
- Database constraints over ActiveRecord validations </essential_principles>
- Controllers - REST mapping, concerns, Turbo responses, API patterns
- Models - Concerns, state records, callbacks, scopes, POROs
- Views & Frontend - Turbo, Stimulus, CSS, partials
- Architecture - Routing, multi-tenancy, authentication, jobs, caching
- Testing - Minitest, fixtures, integration tests
- Gems & Dependencies - What to use vs avoid
- Code Review - Review code against DHH style
- General Guidance - Philosophy and conventions
Specify a number or describe your task. </intake>
<routing>| Response | Reference to Read |
|---|---|
| 1, controller | controllers.md |
| 2, model | models.md |
| 3, view, frontend, turbo, stimulus, css | frontend.md |
| 4, architecture, routing, auth, job, cache | architecture.md |
| 5, test, testing, minitest, fixture | testing.md |
| 6, gem, dependency, library | gems.md |
| 7, review | Read all references, then review code |
| 8, general task | Read relevant references based on context |
After reading relevant references, apply patterns to the user's code. </routing>
<quick_reference>
Naming Conventions
Verbs: card.close, card.gild, board.publish (not set_style methods)
Predicates: card.closed?, card.golden? (derived from presence of related record)
Concerns: Adjectives describing capability (Closeable, Publishable, Watchable)
Controllers: Nouns matching resources (Cards::ClosuresController)
Scopes:
chronologically,reverse_chronologically,alphabetically,latestpreloaded(standard eager loading name)indexed_by,sorted_by(parameterized)active,unassigned(business terms, not SQL-ish)
REST Mapping
Instead of custom actions, create new resources:
POST /cards/:id/close → POST /cards/:id/closure
DELETE /cards/:id/close → DELETE /cards/:id/closure
POST /cards/:id/archive → POST /cards/:id/archival
Ruby Syntax Preferences
# Symbol arrays with spaces inside brackets
before_action :set_message, only: %i[ show edit update destroy ]
# Private method indentation
private
def set_message
@message = Message.find(params[:id])
end
# Expression-less case for conditionals
case
when params[:before].present?
messages.page_before(params[:before])
else
messages.last_page
end
# Bang methods for fail-fast
@message = Message.create!(params)
# Ternaries for simple conditionals
@room.direct? ? @room.users : @message.mentionees
Key Patterns
State as Records:
Card.joins(:closure) # closed cards
Card.where.missing(:closure) # open cards
Current Attributes:
belongs_to :creator, default: -> { Current.user }
Authorization on Models:
class User < ApplicationRecord
def can_administer?(message)
message.creator == self || admin?
end
end
</quick_reference>
<reference_index>
Domain Knowledge
All detailed patterns in references/:
| File | Topics |
|---|---|
| controllers.md | REST mapping, concerns, Turbo responses, API patterns, HTTP caching |
| models.md | Concerns, state records, callbacks, scopes, POROs, authorization, broadcasting |
| frontend.md | Turbo Streams, Stimulus controllers, CSS layers, OKLCH colors, partials |
| architecture.md | Routing, authentication, jobs, Current attributes, caching, database patterns |
| testing.md | Minitest, fixtures, unit/integration/system tests, testing patterns |
| gems.md | What they use vs avoid, decision framework, Gemfile examples |
| </reference_index> |
<success_criteria> Code follows DHH style when:
- Controllers map to CRUD verbs on resources
- Models use concerns for horizontal behavior
- State is tracked via records, not booleans
- No unnecessary service objects or abstractions
- Database-backed solutions preferred over external services
- Tests use Minitest with fixtures
- Turbo/Stimulus for interactivity (no heavy JS frameworks)
- Native CSS with modern features (layers, OKLCH, nesting)
- Authorization logic lives on User model
- Jobs are shallow wrappers calling model methods </success_criteria>
Important Disclaimers:
- LLM-generated guide - may contain inaccuracies
- Code examples from Fizzy are licensed under the O'Saasy License
- Not affiliated with or endorsed by 37signals
Source
git clone https://github.com/everyinc/compound-engineering-plugin/blob/main/plugins/compound-engineering/skills/dhh-rails-style/SKILL.mdView on GitHub Overview
Applies 37signals/DHH conventions to Ruby and Rails code, guiding models, controllers, and Ruby files. It champions fat models, thin controllers, RESTful design, Current attributes, and Hotwire patterns, drawing on production 37signals code and the 'clarity over cleverness' philosophy.
How This Skill Works
It inspects the task (generation, refactor, code review) and enforces vanilla Rails patterns: push business logic into domain models, prefer RESTful resources and CRUD controllers, and share behavior via concerns. It also promotes Current attributes, Turbo/Hotwire usage, and a 'clarity over cleverness' approach learned from DHH's code style.
When to Use It
- Starting a new Rails feature or model aligned with 37signals patterns
- Refactoring toward rich domain models and thin controllers
- Reviewing code for REST purity, Turbo/Hotwire, and simple authorization
- Writing code when asked to adopt DHH/37signals/Basecamp style (DHH, Campfire, HEY)
- Architecting Rails apps with vanilla Rails: routing, concerns, and DB-backed state
Quick Start
- Step 1: Identify the resource and REST endpoints based on DHH patterns
- Step 2: Move business logic into models/concerns and keep controllers thin
- Step 3: Add Turbo/Hotwire patterns and review against REST-first philosophy
Best Practices
- Favor rich domain models over service objects; put business logic in models and POROs
- Use CRUD/restful controllers and resource routes instead of bespoke actions
- Use concerns for horizontal code sharing and reusable behavior
- Keep state in the database; prefer Current attributes and stateful models
- Lean tooling: rely on vanilla Rails, Minitest with fixtures; avoid Redis, Sidekiq, GraphQL, Tailwind
Example Use Cases
- Refactor a controller with custom actions into RESTful resources, e.g., Cards with closures and archival endpoints
- Move business logic from controllers into models and POROs to create fat models
- Introduce a Publishable/Trackable concern shared across multiple models
- Enhance views with Turbo Streams to reflect model state changes
- Replace Devise-based authentication with a lightweight, model-centered auth approach