Show Us Your Production Code #1: Concerns
Michiel Sikkes of Firmhouse responded to my Show Us Your Production Code appeal with a blog post called Production code: I am concerned.
Michiel asked for suggestions and what follows are my ideas on how I would refractor the code (you need to read Michiel's post first for his solution).
Offender
Here is the offending code:
def index
# Code omitted
if params[:sort] == "product_name"
if params[:direction] == "desc"
@products = @products.order("products.name desc")
else
@products = @products.order("products.name asc")
end
elsif params[:sort] == "shop_code"
if params[:direction] == "desc"
@products = @products.order("products.final_article_number_cache desc")
else
@products = @products.order("products.final_article_number_cache asc")
end
else
if ENV['SITE_ALIAS'].nil?
# Sort based on product description for Stofzuigermarkt
@products = @products.order("products.name")
else
@products = @products.order('products.final_article_number_cache')
end
end
# Code omitted
end
Ain't that some production code?
Michiel decided to use controller concerns to refactor this code but I don't agree with this decision. What this code does is orders products based on the request's params.
For me that looks like a model scope with some logic because incoming params differ from the model's column names.
My decision is to create a simple scope on the Product
model. The method will be somewhat big and to keep our models from getting fat we can futher extract the code into a model concern.
Refactoring
# app/models/concerns/orderable_products.rb
module OrderableProducts
extend ActiveSupport::Concern
module ClassMethods
def ordered(column, direction = nil)
column = map_column(column)
order = [column, direction].reject(&:blank?).join(' ')
order(order)
end
private
def map_column(column)
mappings = {
product_name: 'name',
shop_code: 'final_article_number_cache'
}
column = mappings[column]
column ||= ENV['SITE_ALIAS'].nil? ? 'name' : 'final_article_number_cache'
end
end
end
In your model:
class Product < ActiveRecord::Base
include OrderableProducts
# ...
end
And in your controller you can now use ordered
as a scope:
@products = Product.upcoming.active.ordered(params[:sort], params[:direction])
As you see, Michiel and I approached this problem differently. And that's awesome because we both can learn something from each other.
If you think you've got a better solution for this problem, let me know. Getting other people's perspective is a great way to become better.
You're the 9800th person to read this article.