Ruby on Rails is an awesome framework beloved by many. But as your application grows you get a questions very soon, where to put this piece of code or that piece over there. From some point it is on place to introduce new patterns for common scenarios. I will introduce few of them here.
We will start with concerns which I guess most Rails developers know well. And many don't consider it as a good practice. Helpers are the same case.
For complex forms composed of more models we will have a look on form objects.
We will talk about decorators as well which are presented by gem Draper
(https://github.com/drapergem/draper).
Very close to decorators are policies, for which we will use Pundit (https://github.com/elabs/pundit).
Many other languages have publisher-listener mechanisms built in. In ruby you can easily achieve the same functionality by Wisper (https://github.com/krisleech/wisper).
And we will close it with services, probably the most hackneyed pattern of those a bit advanced ones.
To keep track of all those new classes that will appear in the project it is good to come with new folders. The folders are not fixed, but the following structure makes sense:
app/
• controllers
• decorators
• helpers
• listeners
• models/
• concerns
• policies
• publishers
• services
• views
Concerns are pieces of code which you can include to as many classes as you need, so you can stay DRY. It is also an elegant way you can avoid inheritance.
# app/models/article.rb:
class Article < ActiveRecord::Base
has_many :comments, as: :commentable
def find_first_comment
comments.first(created_at DESC)
end
def self.least_commented
# too lazy to implement
end
end
# app/models/article.rb:
class Article < ActiveRecord::Base
include Commentable
end
# app/models/news.rb
class News < ActiveRecord::Base
include Commentable
end
# app/models/concerns/commentable.rb:
module Commentable
extend ActiveSupport::Concern
included do
has_many :comments, as: :commentable
end
def find_first_comment
comments.first(created_at DESC)
end
module ClassMethods
def least_commented
# too lazy to implement
end
end
end
Sometimes you can have a form which is not possible to map to one entity. You can have form with two or three or more entities in one form and those entities doesn't even need to have a relation between each other. In such case you can use a form object and still use standard Rails validations.
# app/models/register_form.rb:
class RegisterForm
include ActiveModel::Model
validates_presence_of :name
validates_presence_of :description
def user
@user ||= User.new
end
def note
@note ||= Note.new
end
def submit(params)
user.attributes = params.slice(:name)
note.attributes = params.slice(:description)
if valid?
user.save!
note.save!
true
else
false
end
end
end
# app/views/users/new.html.erb:
<%= form_for @register_form do |f| %>
<%= f.text :name %>
<%= f.text :description %>
<%= f.submit "Register" %>
<% end %>
# app/controllers/users_controller.rb
def new
@register_form = RegisterForm.new
end
def create
@register_form = RegisterForm.new
if @register_form.submit(params[:user])
session[:user_id] = @register_form.user.id
redirect_to @register_form.user
else
render "new"
end
end
Decorators add another thin presentation layer in which you can adjust view output based on some decission-making, that exactly shouldn't be done in helpers. A typical example is showing different html based on same state.
# app/controllers/articles_controller.rb:
def show
@article = Article.find(params[:id])
end
# app/controllers/articles_helper.rb:
def publication_status(article)
if article.published?
"Published at #{article.published_at.strftime('%A, %B %e')}"
else
"Unpublished"
end
end
# app/views/articles/show.html.erb:
<div class="time">
<%= publication_status @article.published_at %>
</div>
# app/decorators/article_decorator.rb:
class ArticleDecorator < Draper::Decorator
delegate_all
def publication_status
if published?
"Published at #{published_at}"
else
"Unpublished"
end
end
def published_at
object.published_at.strftime("%A, %B %e")
end
end
# app/controllers/articles_controller.rb:
def show
@article = Article.find(params[:id]).decorate
end
# app/views/articles/show.html.erb:
<div class="time">
<%= @article.publication_status %>
</div>
Helpers are methods accessible in views usually accepting some value and returning it back in some modified form.
It should be used for adjusting the output, no business logic should be involved here. Very often it is used for formatting.
# app/helpers/application_helper.rb:
class ApplicationHelper
def humanize_date
date.strftime("%d. %m. %Y")
end
end
# app/views/articles/show.html.erb:
<div class="time">
<%= humanize_date @article.published_at %>
</div>
Policies solve authorizations. It is very close to decorators, but decorators are solving the view logic (what to show based on some conditions), policies are solving the authorization logic instead (if the user has right to see some piece of page or execute an action).
The boundary between decorators and policies is very thin.
# app/views/articles/show.html.erb:
<div class="buttons">
<% if not @article.published? && @user.admin? %>
<%= button_to article_edit(@article), method: :get %>
<% end %>
</div>
# app/policies/article_policy.rb:
class ArticlePolicy
attr_reader :user, :article
def initialize(user, article)
@user = user
@article = article
end
def update?
user.admin? or not article.published?
end
end
# app/controllers/articles_controller.rb:
def show
@article = Article.find params[:id]
@article_policy = ArticlePolicy.new current_user, @article
end
# app/views/articles/show.html.erb:
<div class="buttons">
<% if not @article_policy.update? %>
<%= button_to article_edit(@article), method: :get %>
<% end %>
</div>
# app/policies/article.rb:
class ArticlePolicy < ApplicationPolicy
# automatically gives argument user
# and article, based on file name
def update?
user.admin? or not article.published?
end
end
# app/controllers/articles_controller.rb:
def show
@article = Article.find params[:id]
authorize @article
end
# app/views/articles/show.html.erb:
<div class="buttons">
<% if policy(@article).update? %>
<%= button_to article_edit(@article), method: :get %>
<% end %>
</div>
Publishers and Listeners are two components of eventing functionality. Imagine an example when you delete an article. When the deletion is executed you want other system parts can react somehow (for example send an email).
The important thing is, you don't want to hardcode those reactions to the deletion code. You just send a message (publisher) to anyone who is listening (listeners).
# app/models/article.rb:
class Article < ActiveRecord::Base
after_commit :notify_editors, on: :create
after_commit :generate_feed_item, on: :create
private
def notify_editors
EditorMailer.send_notification(self).deliver_later
end
def generate_feed_item
FeedItem.create(self)
end
end
# config/initializers/subscribers.rb:
Article.subscribe(ArticleListener) / published synchronously
# Article.subscribe(ArticleListener, async: true) / for asynchronous publishing
# app/models/article.rb:
class Article < ActiveRecord::Base
include Wisper.model
end
# app/listeners/article_listener.rb:
class ArticleListener
def self.after_create(article)
EditorMailer.send_notification(article).deliver_later
FeedItem.create(article)
end
end
# app/publishers/article.rb:
class DeleteArticle
include Wisper::Publisher
def call(article_id)
article = Article.find_by_id(article_id)
# do some evil things with the article
if article.deleted?
broadcast(:delete_successful, article.id)
else
broadcast(:delete_failed, article.id)
end
end
end
# app/controllers/articles_controller.rb:
def destroy
delete_article = DeleteArticle.new
delete_article.subscribe(ArticleMailer, async: true)
delete_article.subscribe(StatisticsRecorder, async: true)
delete_article.on(:delete_successful) { |article_id| redirect_to order_path(order_id) }
delete_article.on(:delete_failed) { |article_id| render action: :new }
delete_article.call(params[:id])
end
If the logic in your controller is starting to blow up, it is time to extract it so a separate service object.
# app/controllers/user_controller.rb:
def show
@user = GetUserAndNotify.find(params[:id])
end
# app/services/get_user_and_notify.rb:
class GetUserAndNotify
def self.find(id)
user = User.find id
if user.notify_on_get?
Logger.log "User #{user.name} with id #{user.id} was requested"
EventNotifier.log_event user.class.name, user.id
end
user.get_count += 1
user.save
user
end
end
This sheet was created by Jiří Procházka https://www.linkedin.com/in/jiriprochazka/ and is based on the lecture of Honza Minárik https://www.linkedin.com/in/janminarik which he has made for CSRUG. It was created with his permission.