home / skills / everyinc / compound-engineering-plugin / dhh-rails-style

This skill applies DHH's Rails conventions to Ruby and Rails code, emphasizes fat models, RESTful controllers, and Turbo-driven views for clarity.

npx playbooks add skill everyinc/compound-engineering-plugin --skill dhh-rails-style

Review the files below or copy the command above to add this skill to your agents.

Files (7)
SKILL.md
6.7 KB
---
name: dhh-rails-style
description: This skill should be used when writing Ruby and Rails code in DHH's distinctive 37signals style. It applies when writing Ruby code, Rails applications, creating models, controllers, or any Ruby file. Triggers on Ruby/Rails code generation, refactoring requests, code review, or when the user mentions DHH, 37signals, Basecamp, HEY, or Campfire style. Embodies REST purity, fat models, thin controllers, Current attributes, Hotwire patterns, and the "clarity over cleverness" philosophy.
---

<objective>
Apply 37signals/DHH Rails conventions to Ruby and Rails code. This skill provides comprehensive domain expertise extracted from analyzing production 37signals codebases (Fizzy/Campfire) and DHH's code review patterns.
</objective>

<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>

<intake>
What are you working on?

1. **Controllers** - REST mapping, concerns, Turbo responses, API patterns
2. **Models** - Concerns, state records, callbacks, scopes, POROs
3. **Views & Frontend** - Turbo, Stimulus, CSS, partials
4. **Architecture** - Routing, multi-tenancy, authentication, jobs, caching
5. **Testing** - Minitest, fixtures, integration tests
6. **Gems & Dependencies** - What to use vs avoid
7. **Code Review** - Review code against DHH style
8. **General Guidance** - Philosophy and conventions

**Specify a number or describe your task.**
</intake>

<routing>

| Response | Reference to Read |
|----------|-------------------|
| 1, controller | [controllers.md](./references/controllers.md) |
| 2, model | [models.md](./references/models.md) |
| 3, view, frontend, turbo, stimulus, css | [frontend.md](./references/frontend.md) |
| 4, architecture, routing, auth, job, cache | [architecture.md](./references/architecture.md) |
| 5, test, testing, minitest, fixture | [testing.md](./references/testing.md) |
| 6, gem, dependency, library | [gems.md](./references/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`, `latest`
- `preloaded` (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

```ruby
# 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:**
```ruby
Card.joins(:closure)         # closed cards
Card.where.missing(:closure) # open cards
```

**Current Attributes:**
```ruby
belongs_to :creator, default: -> { Current.user }
```

**Authorization on Models:**
```ruby
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](./references/controllers.md) | REST mapping, concerns, Turbo responses, API patterns, HTTP caching |
| [models.md](./references/models.md) | Concerns, state records, callbacks, scopes, POROs, authorization, broadcasting |
| [frontend.md](./references/frontend.md) | Turbo Streams, Stimulus controllers, CSS layers, OKLCH colors, partials |
| [architecture.md](./references/architecture.md) | Routing, authentication, jobs, Current attributes, caching, database patterns |
| [testing.md](./references/testing.md) | Minitest, fixtures, unit/integration/system tests, testing patterns |
| [gems.md](./references/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>

<credits>
Based on [The Unofficial 37signals/DHH Rails Style Guide](https://github.com/marckohlbrugge/unofficial-37signals-coding-style-guide) by [Marc Köhlbrugge](https://x.com/marckohlbrugge), generated through deep analysis of 265 pull requests from the Fizzy codebase.

**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
</credits>

Overview

This skill applies DHH / 37signals Rails conventions to Ruby and Rails code. It guides code generation, refactors, and code review to favor REST purity, fat models, thin controllers, Current attributes, and Hotwire patterns. Use it to align apps with pragmatic decisions from production 37signals codebases: clarity, database-backed solutions, and minimalism.

How this skill works

When given Ruby or Rails code, a description of a task, or a code review request, the skill inspects controllers, models, views, routes, jobs, tests, and Gem choices. It flags departures from DHH style, suggests idiomatic rewrites (RESTful resources, state-as-records, concerns, Turbo/Stimulus patterns), and proposes concrete code edits or examples. It also recommends testing patterns, fixtures, and dependency alternatives consistent with the philosophy.

When to use it

  • Generating or refactoring controllers, models, routes, or views for a Rails app
  • Reviewing code for adherence to 37signals/DHH conventions
  • Choosing gems, architecture, or background job strategies
  • Designing authorization, state modeling, or Current attributes
  • Implementing Turbo/Stimulus interactivity or native CSS layers

Best practices

  • Prefer rich ActiveRecord models and concerns over service objects for core domain logic
  • Map actions to RESTful resources; create small resources for state transitions (e.g., closures, archivals)
  • Model state as records and use joins/where.missing instead of boolean columns
  • Default associations with Current attributes and keep controllers thin; use create! for fail-fast behavior
  • Favor database-backed queues, minimal dependencies, and ship-prototype-then-refine mentality
  • Use Minitest with fixtures, Turbo/Stimulus for interactivity, and native CSS layers instead of heavy frameworks

Example use cases

  • Refactor a controller with multiple custom actions into RESTful nested resources
  • Convert boolean state columns into associated state records and update scopes
  • Review a feature implementation and replace an unnecessary service object with model methods and a concern
  • Recommend gem alternatives and tradeoffs (avoid devise/pundit/sidekiq by default)
  • Suggest Turbo Streams and a Stimulus controller for a form that used to rely on custom JS

FAQ

Do you always reject gems like Devise or Sidekiq?

Not always; prefer simple, maintainable in-house solutions first. Recommend a gem only when it clearly simplifies maintenance and matches the app's scale.

When should I introduce a service object?

Introduce a PORO when logic spans multiple models or external systems and extracting it improves clarity; prefer rich models and concerns for domain behavior first.

How do I model state transitions?

Create associated state records (e.g., closures, archivals) and use joins or where.missing for queries. Expose expressive predicates and scopes on the model.