Delegation patterns in Ruby

Reducing object bloat begins with doing less

[inlinetweet]In almost every project there are those objects which seemingly get involved in every aspect of the application.[/inlinetweet] These are the so-called “god objects”: they can do everything (omnipotent), they know everything (omniscient), and they are everywhere in the application (omnipresent). Most often these are objects which are at the intersections of business logic: User or Account, Project, and Order are all usual suspects.

One of the core tenants of object-oriented programming is that large problems are made up of many smaller problems, and as such, can be solved by providing solutions to those smaller problems in the form of objects. [inlinetweet]God objects violate this core tenet by trying to be one solution for too many problems.[/inlinetweet]

[inlinetweet]A typical example of a god object is the `User` model of many Rails applications.[/inlinetweet] Here, `User` might be responsible for user specific information as well as knowing about phone numbers, emails, profile information, preferences, and handling authentication. It’s too much and it results in `User` being coupled to every aspect of the application.

Although there are many tactics a developer can employ to limit the influence of these “god objects”, one of the simplest is just reassigning some of their responsibility, and using a delegation pattern may be the simplest way to do just that.

[inlinetweet]Let’s look at four different ways we can use delegation in Ruby.[/inlinetweet]

Delegation by Default

The act of delegation in Object Oriented Programming is so common we oftentimes don’t even realize when it’s happening. Take “inheritance” for example: when you create a subclass, you inherit all the parent class’s public methods, and unless those methods are overwritten, any calls to the inherited methods are passed to the parent class.

Even in methods which are overwritten, specifically calling the superclass’s method through the use of super, is delegation, because the child class is relying on the parent to accomplish what it doesn’t need to.

Here’s an example of what I mean:


class Report
  def records
    # return a list of report records
  end

  def print
    # output the report
  end
end

class InTriplicateReport < Report
  def print
    super
    super
    super
  end
end

report = InTriplicateReport.new
report.records # => returns the records in the report
report.print # => Prints the report in triplicate

Here we see that InTriplicateReport inherits both records and print from the Report class. InTriplicateReport doesn’t overwrite records, but instead relies on the parent to handle the responsibility. On the other hand, print is overwritten, but we still rely on the parent through our multiple calls to super. In each case, InTriplicateReport delegates responsibility to the parent Report class rather than duplicating effort and code. Later, as requirements change, records or print can be further modified, but until then, it’s best to allow Report to handle the logic.

Explicit Delegation

So inheritance is delegation, but really any time you use another object to perform logic instead of keeping it within the class itself, you are delegating.

In the Report class below, the #print method accepts an object which will handle transforming the report data into a specified format (XML, CSV, and JSON). The Report class doesn’t “care” about the final output, it just wants to output it, and so it delegates the actual formatting to one of the formatter objects.

class Report
  def records
    # lists all the reports records
  end

  def print(delegate)
    # logic to handle output
      delegate.generate(self.records)
    # closing logic
  end
end

class CSVFormatter
  def generate(records)
    # formats to CSV
  end
end

class XMLFormatter
  def generate(records)
    # formats to XML
  end
end

class JSONFormatter
  def generate(records)
    # formats to JSON
  end
end

report = Report.new
report.print(CSVFormatter.new) # outputs to CSV
report.print(XMLFormatter.new) # outputs to XML
report.print(JSONFormatter.new) # outputs to JSON

The formatter objects being passed in are the delegates; they handle converting the data to the desired format, while Report maintains focus on outputting the formatted data. By delegating responsibility in this instance, we keep our objects focused on performing a single function (see: Single Responsibility Principle), decouple our logic, and generally make life easier for ourselves.

This style of delegation is commonly used in statically typed languages such as .Net, Java, and Objective-C. In those languages, the delegate must conform to a sort of contract called an “Interface” or “Protocol”. Thankfully, Ruby doesn’t have such hang-ups and assumes that if the object walks like a duck and quacks like a duck, it’s a duck.

Delegation with method_missing

[inlinetweet]Delegation in Ruby begins to get really interesting when you hide the fact that another object is even involved.[/inlinetweet]

For this example, let’s assume we’re working on an e-commerce system. This system will have “orders” which, in turn, can have zero or more products and will relate to each of those products through a line item. Let’s further assume that we would like to access product information, such as “sku”, “name”, “description”, and “price”, directly from the line_item object rather than chaining out to the product instance (e.g. line_item.product_sku instead of line_item.product.sku).

We can do this using Ruby’s method_missing, a method inherited from BasicObject, and which catches messages sent to an object which are not explicitly defined.

class Order
  def line_items
    # collection of LineItems
  end
end

class LineItem
  attr_reader :product

  def initialize(product)
    @product = product
  end

  def method_missing(method, *args)
    if method.to_s.match(/product_(.+)/)
      self.product.send($1, *args)
    else
      super
    end
  end
end

class Product
  attr_accessor :sku, :name, :description, :price
  # product related methods
end

In our example, `method_missing` looks to see if the message passed begins with “product_”, if it does, it passes that message (sans “product_”) on to the `product` instance. If the message doesn’t match the pattern, it passes it on up the call chain with `super`.

Now, if we were to list the line items from an order we could do something like this:

puts "SKU,Name,Price"

order.line_items.each do |line_item|
  puts "%s,%s,%s" % [line_item.product_sku, line_item.product_name, line_item.product_price]
end

Although it looks as if we are calling specific methods on the line_item object, in reality we’re delegating the messages to the product object.

Delegation with Ruby’s Forwardable

In the above example, we used method_missing to delegate all methods beginning with “product_” not defined in LineItem to the Product class. The advantage here is that it handles all current and future Product methods. If a new method, such as serial_number, is added, LineItem can immediately begin using it without any alterations to the code.

On the other hand, if you need to be more specific about which methods are allowed to be delegated, Ruby’s Forwardable module is a better solution.

Here’s the same LineItem class, rewritten using the Forwardable module.

include Forwardable

class LineItem
  extend Forwardable

  def_delegators :@product, :sku, :name, :description, :price

  attr_reader :product

  def initialize(product)
    @product = product
  end
end

In the above code, we extend Forwardable into LineItem, add the def_delegators line, and remove method_missing altogether.

The def_delegators line is pretty straightforward: it states that any call to sku, name, description, and price, should be instead handled by the @product instance variable (note: we could have written :product since we’ve defined it with attr_reader)

Where previously we prepended “product_” to delegated methods, now we can call the method directly, because there’s no longer a need to distinguish product specific methods for method_missing.

Here’s our output rewritten to take advantage of the refactoring:

puts "SKU,Name,Price"

order.line_items.each do |line_item|
  puts "%s,%s,%s" % [line_item.sku, line_item.name, line_item.price]
end

If you wanted to keep the “product_” prepended to the method calls – maybe to keep from overwriting another method of the same name – you would need to use def_delegator instead. However, doing so will limit you to defining one delegator at a time.

class LineItem
  extend Forwardable

  attr_reader :id

  # avoid overwriting LineItem#id
  def_delegator :@product, :id, :product_id
  def_delegators :@product, :sku, :name, :description, :price

  ...

end

In Object Oriented Programming, delegation is an incredibly useful pattern, and – as is so often the case in Ruby – there are numerous ways to implement it. It is not some elusive concept which is difficult to nail down, but one which we encounter regularly through typical OO development.

[inlinetweet]Not only is delegation simple, it is powerful in that it allows us to limit the power of our objects.[/inlinetweet] Through effective use of inheritance, message passing, and forwarding of responsibility, we can create opportunities to segment our classes and delineate responsibility. [inlinetweet]Delegation helps us bring those “god objects” back down to earth.[/inlinetweet]

tags: ,