원클릭으로
원클릭으로
| name | rails-dev-guidelines |
| description | Rails MVC patterns, ActiveRecord, conventions, and best practices |
This skill ensures consistent, maintainable Rails code following conventions and best practices for the application.
app/
├── models/ # Business logic and data models
├── components/ # View components (ViewComponent gem)
├── controllers/ # Request handling and response rendering
├── views/ # Templates (ERB, Turbo)
├── helpers/ # View helpers
├── jobs/ # Background jobs
├── mailers/ # Email sending
├── services/ # Complex business logic
├── queries/ # Complex database queries
├── forms/ # Form objects for complex forms
└── concerns/ # Shared modules (models & controllers)
When creating or modifying a model:
User, not Users)app/models/concerns/When creating or modifying a controller:
params.require().permit())before_action for authentication/authorizationrescue_from or inline rescues❌ Don't put business logic in controllers:
# app/controllers/users_controller.rb
def create
@user = User.new(user_params)
@user.email = params[:email].downcase
@user.role = 'member'
@user.save
UserMailer.welcome_email(@user).deliver_later
Analytics.track_signup(@user)
redirect_to @user
end
✅ Do move logic to models or services:
# app/controllers/users_controller.rb
def create
@user = User.create_with_defaults(user_params)
if @user.persisted?
redirect_to @user, notice: 'Welcome!'
else
render :new, status: :unprocessable_entity
end
end
# app/models/user.rb
class User < ApplicationRecord
after_create :send_welcome_email
after_create :track_signup
def self.create_with_defaults(params)
user = new(params)
user.email = user.email.downcase
user.role ||= 'member'
user.save
user
end
private
def send_welcome_email
UserMailer.welcome_email(self).deliver_later
end
def track_signup
Analytics.track_signup(self)
end
end
❌ Don't cram complex workflows into models:
class Order < ApplicationRecord
def process_payment
# 50+ lines of payment processing logic
end
end
✅ Do extract to service objects:
# app/services/order_processor.rb
class OrderProcessor
def initialize(order)
@order = order
end
def process
return false unless valid?
ActiveRecord::Base.transaction do
charge_payment
update_inventory
send_confirmation
end
true
rescue PaymentError => e
@order.errors.add(:base, e.message)
false
end
private
def valid?
@order.valid? && @order.items.any?
end
# ... more private methods
end
❌ Don't use raw params:
def create
User.create(params[:user]) # Security vulnerability!
end
✅ Do use strong parameters:
def create
@user = User.new(user_params)
# ...
end
private
def user_params
params.require(:user).permit(:email, :name, :role)
end
❌ Don't rely only on ActiveRecord validations:
# app/models/user.rb
validates :email, uniqueness: true
✅ Do add database constraints too:
# db/migrate/xxx_create_users.rb
create_table :users do |t|
t.string :email, null: false
t.timestamps
end
add_index :users, :email, unique: true
# app/models/user.rb
validates :email, presence: true, uniqueness: true
❌ Don't lazy load associations in loops:
@users = User.all
@users.each do |user|
puts user.posts.count # N+1 query!
end
✅ Do eager load associations:
@users = User.includes(:posts).all
@users.each do |user|
puts user.posts.count # No extra queries
end
Callbacks make models hard to test and reason about. Use service objects instead.
Never use save(validate: false) or update_column unless absolutely necessary.
Extract complex queries to query objects or scopes.
find_by_sql Without ReasonUse ActiveRecord query interface when possible.
Use slugs or UUIDs for public-facing URLs.
belongs_to :user
has_many :posts
has_many :comments, through: :posts
has_one :profile, dependent: :destroy
validates :email, presence: true, uniqueness: true, format: { with: URI::MailTo::EMAIL_REGEXP }
validates :age, numericality: { greater_than: 18 }
validates :terms, acceptance: true
scope :active, -> { where(active: true) }
scope :recent, -> { order(created_at: :desc).limit(10) }
scope :by_role, ->(role) { where(role: role) }
before_validation :normalize_email
after_create :send_notification
after_destroy :cleanup_associated_data
Need help with:
resources/associations.mdresources/validations.mdresources/queries.mdresources/service-objects.mdresources/background-jobs.mdresources/security.mdStatus: Core skill (~450 lines) | 6 resource files