| name | rspec-testing |
| description | This skill should be used when writing, reviewing, or improving RSpec tests for Ruby on Rails applications. Use this skill for all testing tasks including model specs, controller specs, system specs, component specs, service specs, and integration tests. The skill provides comprehensive RSpec best practices from Better Specs and thoughtbot guides. Use when this capability is needed. |
| metadata | {"author":"dgalarza"} |
RSpec Testing for Rails
Overview
Write comprehensive, maintainable RSpec tests following industry best practices. This skill combines guidance from Better Specs and thoughtbot's testing guides to produce high-quality test coverage for Rails applications.
Core Testing Principles
1. Test-Driven Development (TDD)
Follow the Red-Green-Refactor cycle:
- Red: Write failing tests that define expected behavior
- Green: Implement minimal code to make tests pass
- Refactor: Improve code while tests continue to pass
2. Test Structure (Arrange-Act-Assert)
Organize tests with clear phases separated by newlines:
it 'creates a new article' do
user = create(:user)
attributes = {title: 'Test Article', body: 'Content here'}
article = Article.create(attributes)
expect(article).to be_persisted
expect(article.title).to eq('Test Article')
end
3. Single Responsibility
Each test should verify one behavior. For unit tests, use one expectation per test. For integration tests, multiple expectations are acceptable when testing a complete flow.
4. Test Real Behavior
Avoid over-mocking. Test actual application behavior when possible. Only stub external services, slow operations, and dependencies outside your control.
Test Type Decision Tree
When to Write Model Specs
Use model specs (spec/models/) for:
- Validations
- Associations
- Scopes
- Instance methods
- Class methods
- Enums and constants
- Database constraints
Example:
RSpec.describe Article do
describe 'validations' do
it 'validates presence of title' do
article = build(:article, title: nil)
expect(article).not_to be_valid
expect(article.errors[:title]).to include("can't be blank")
end
end
describe 'associations' do
it { is_expected.to belong_to(:user) }
it { is_expected.to have_many(:comments) }
end
describe '#published?' do
it 'returns true when status is published' do
article = build(:article, status: :published)
expect(article.published?).to be true
end
end
end
When to Write Controller Specs
Use controller specs (spec/controllers/) for:
- Authorization checks (Pundit/CanCanCan)
- Request routing and parameter handling
- Response status codes
- Instance variable assignments
- Flash messages
- Redirects
Example:
RSpec.describe ArticlesController do
describe 'POST #create' do
context 'with valid parameters' do
it 'creates a new article and redirects' do
user = create(:user)
session[:user_id] = user.id
valid_attributes = {
title: 'Test Article',
body: 'Article content'
}
expect do
post :create, params: {article: valid_attributes}
end.to change(Article, :count).by(1)
expect(response).to redirect_to(Article.last)
end
end
context 'with invalid parameters' do
it 'does not create article and renders new template' do
user = create(:user)
session[:user_id] = user.id
invalid_attributes = {title: '', body: ''}
expect do
post :create, params: {article: invalid_attributes}
end.not_to change(Article, :count)
expect(response).to render_template(:new)
end
end
end
end
When to Write System Specs
Use system specs (spec/system/) for:
- End-to-end user workflows
- Multi-step interactions
- JavaScript functionality
- Form submissions
- Navigation flows
- Real user scenarios
Naming convention: user_action_spec.rb or feature_description_spec.rb
Example:
RSpec.describe 'Article Creation' do
it 'allows a user to create a new article' do
user = create(:user)
visit '/login'
fill_in 'Email', with: user.email
fill_in 'Password', with: 'password'
click_button 'Sign In'
click_link 'New Article'
expect(page).to have_current_path(new_article_path)
fill_in 'Title', with: 'My Test Article'
fill_in 'Body', with: 'This is the article content'
select 'Published', from: 'Status'
click_button 'Create Article'
expect(page).to have_content('Article created successfully!')
expect(page).to have_content('My Test Article')
end
end
When to Write Component Specs
Use component specs (spec/components/) for:
- ViewComponent rendering
- Variant behavior
- Slot functionality
- Conditional rendering
- Component attributes
Example:
RSpec.describe ButtonComponent, type: :component do
describe 'variants' do
it 'renders primary variant' do
render_inline(described_class.new(variant: :primary)) { 'Click me' }
button = page.find('button')
expect(button[:class]).to include('btn-primary')
expect(page).to have_button('Click me')
end
it 'renders secondary variant' do
render_inline(described_class.new(variant: :secondary)) { 'Cancel' }
button = page.find('button')
expect(button[:class]).to include('btn-secondary')
end
end
end
When to Write Service/Integration Specs
Use service/integration specs (spec/services/, spec/integration/) for:
- Complex business logic
- Multi-step workflows
- External API integrations
- Background job processing
- Data transformations
RSpec Syntax & Style Guide
Describe Blocks
Use Ruby documentation conventions:
.method_name for class methods
#method_name for instance methods
describe '.find_by_title' do
describe '#publish' do
describe 'validations' do
Context Blocks
Start with "when," "with," or "without":
context 'when user is admin' do
context 'with valid parameters' do
context 'without authentication' do
It Blocks
- Keep descriptions under 40 characters
- Use third-person present tense
- Never use "should" in descriptions
it 'creates a new article' do
it 'validates presence of title' do
it 'redirects to dashboard' do
it 'should create a new article' do
it 'should validate presence of title' do
Expectations
Always use expect syntax (never should):
expect(article).to be_valid
expect(response).to have_http_status(:success)
expect { action }.to change(Article, :count).by(1)
article.should be_valid
response.should have_http_status(:success)
One-Liners
Use is_expected for concise one-line specs:
subject { article }
it { is_expected.to be_valid }
it { is_expected.to be_persisted }
System Test Best Practices
Authentication in System Tests
Test authentication flows directly without stubbing:
visit '/login'
fill_in 'Email', with: user.email
fill_in 'Password', with: 'password'
click_button 'Sign In'
expect(page).to have_content('Dashboard')
Controller Test Authentication
For controller tests, use direct session assignment rather than stubbing:
session[:user_id] = user.id
allow_any_instance_of(Controller).to receive(:logged_in?).and_return(true)
Avoid CSS Class Testing
Don't test implementation details like CSS utility classes. Test semantic selectors and content:
expect(page).to have_selector(:test_id, 'user-modal')
expect(page).to have_css("[aria-hidden='false']")
expect(page).to have_content('Success message')
expect(page).to have_button('Submit')
expect(page).to have_css('.opacity-100')
expect(page).to have_css('.bg-red-500')
expect(page).to have_css('.rounded-lg')
Factory Patterns
Organization
- Associations (implicit) first
- Attributes (alphabetical)
- Traits (alphabetical)
FactoryBot.define do
factory :article do
user
category
body { 'Article content goes here...' }
published_at { Time.current }
status { :draft }
title { 'Sample Article Title' }
trait :published do
status { :published }
published_at { 1.day.ago }
end
trait :with_tags do
after(:create) do |article|
create_list(:tag, 3, article: article)
end
end
end
end
Prefer Build Over Create
Use build and build_stubbed when database persistence isn't needed:
it 'validates title format' do
article = build(:article, title: '')
expect(article).not_to be_valid
end
it 'validates title format' do
article = create(:article, title: '')
expect(article).not_to be_valid
end
Common Testing Patterns
Testing Validations
describe 'validations' do
it 'validates presence of title' do
article = build(:article, title: nil)
expect(article).not_to be_valid
expect(article.errors[:title]).to include("can't be blank")
end
it 'validates length of title' do
article = build(:article, title: 'a' * 256)
expect(article).not_to be_valid
end
it 'allows valid titles' do
article = build(:article, title: 'Valid Title')
expect(article).to be_valid
end
end
Testing Enums
describe 'enums' do
it 'defines status enum' do
expect(described_class.statuses).to eq({
'draft' => 'draft',
'published' => 'published',
'archived' => 'archived'
})
end
it 'has correct default' do
article = described_class.new
expect(article.status).to eq('draft')
end
end
Testing Authorization
context 'when user is not admin' do
it 'raises authorization error' do
user = create(:user, role: :member)
session[:user_id] = user.id
expect do
get :admin_dashboard
end.to raise_error(Pundit::NotAuthorizedError)
end
end
Using Shoulda Matchers
describe 'associations' do
it { is_expected.to belong_to(:user) }
it { is_expected.to have_many(:comments) }
end
describe 'validations' do
it { is_expected.to validate_presence_of(:title) }
it { is_expected.to validate_length_of(:title).is_at_most(255) }
end
What to Avoid
❌ Don't Stub the System Under Test
Never mock or stub methods on the class being tested:
it 'processes payment' do
order = Order.new
allow(order).to receive(:calculate_total).and_return(100)
expect(order.process_payment).to be true
end
it 'processes payment' do
order = Order.new(line_items: [line_item])
expect(order.process_payment).to be true
end
❌ Don't Test Private Methods
Test the public interface. Private methods are tested indirectly:
describe '#calculate_total (private)' do
it 'sums line items' do
order.send(:calculate_total)
end
end
describe '#total' do
it 'returns sum of line items' do
expect(order.total).to eq(100)
end
end
❌ Avoid any_instance_of
Use dependency injection instead:
allow_any_instance_of(PaymentService).to receive(:charge)
payment_service = instance_double(PaymentService)
allow(payment_service).to receive(:charge).and_return(success)
order = Order.new(payment_service: payment_service)
Quick Reference
Test Organization
RSpec.describe ClassName do
let(:resource) { create(:resource) }
before do
end
describe 'validations' do
end
describe 'associations' do
end
describe '.class_method' do
end
describe '#instance_method' do
context 'when condition' do
it 'does something' do
end
end
end
end
Expectation Matchers
expect(value).to eq(expected)
expect(value).to be(expected)
expect(value).to match(/regex/)
expect(object).to be_valid
expect(object).to be_persisted
expect(collection).to be_empty
expect(array).to include(item)
expect(array).to contain_exactly(1, 2, 3)
expect(hash).to have_key(:name)
expect { action }.to change(Model, :count).by(1)
expect { action }.to change { object.attribute }.from(old).to(new)
expect { action }.to raise_error(ErrorClass)
expect { action }.not_to raise_error
Resources
This skill includes detailed reference documentation in the references/ directory:
references/better_specs_guide.md
Comprehensive patterns from Better Specs including:
- Describe/context/it block conventions
- Subject and let usage
- Mocking strategies
- Shared examples
- Factory patterns
references/thoughtbot_patterns.md
thoughtbot's RSpec best practices covering:
- Modern RSpec syntax
- Test structure and organization
- What to avoid in tests
- Capybara patterns for system tests
- Factory organization
Load these references when you need detailed examples or are unsure about a specific pattern.
Converted and distributed by TomeVault — claim your Tome and manage your conversions.