Using Modules in Ruby

The idea for this post came when I was reading the source code of Gems that I have used in the past, something I have been making a point to do while I am in between roles. I make a point of noting any methods, patterns, or paradigms I am unfamiliar with for later research. I was looking through the code for acts_as_tenant when I came across the prepend method, which I had not seen before. prepend is related to Modules, but I didn’t find the docs for that method very intuitive, so I felt like I needed some background information first. What follows is an exploration of what Modules are, what they do, how they are used, and where prepend fits in.

Definition

The official documentation for Ruby defines modules as collections of methods and constants, where methods can be either:

  • instance: methods that will become available in a class that includes or extends this module or
  • module: methods that will not be available to the including class but can be called directly through the module.
module SomeModule
  def an_instance_method
    'can be called on instances of classes including this module'
  end
  def a_class_method
    'can be called on classes that extend this module'
  end
  def self.a_module_method
    'can only be called on this module'
  end
end

class SomeClass
  include SomeModule
end

sc = SomeClass.new
puts sc.an_instance_method 
# => can be called on instances of classes including this module

class SomeOtherClass
  extend SomeModule
end

puts SomeOtherClass.a_class_method
# => can be called on classes that extend this module

begin
  puts sc.a_module_method
rescue => e
  puts e 
  # => undefined method `a_module_method' for #<SomeClass:123>
end

begin
  puts SomeOtherClass.a_module_method
rescue => e
  puts e 
  # => undefined method `a_module_method' for SomeOtherClass:Class
end

puts SomeModule.a_module_method 
# => can only be used in this module

How modules really work

Understanding how modules work requires understanding how method lookup is implemented as well. Ruby uses something called the ancestor chain to determine if an object responds to a given message and including/extending a module in a class adds the module to the ancestor chain. When including a module in a class that module is after the class in the ancestor chain, but using prepend makes the module come before the class, so the module could override an identically named method in the class.

class SomeClass; end
module SomeModule; end
class OtherClass; end

SomeClass.include SomeModule
puts SomeClass.ancestors 
# => [SomeClass, SomeModule, Object, Kernel, BasicObject]
OtherClass.prepend SomeModule
puts OtherClass.ancestors 
# => [SomeModule, OtherClass, Object, Kernel, BasicObject]

Applications of modules

The 2 main uses of modules I would like to focus on are namespacing and code reuse.

Namespacing

Empty modules are used for organization and managing identically named classes. Often seen in Rails code:

# app/controllers/admin/users_controller.rb
module Admin
  class UsersController < ApplicationController
    # ...
  end
end

# app/controllers/users_controller.rb
class UsersController < ApplicationController
  # ...
end

Code reuse / Composition

Real production Ruby code often groups similar methods into modules so that classes can be composed using them. An example of this is in Ruby on Rails’ ActionController::Base class, which Rails applications inherit their controllers from. This class includes modules like Caching, Cookies, Logging, etc that encapsulate those concepts. In fact, the base class contains little code and all the functionality comes from the modules it includes. For example, the Logging module adds a class method called log_at to classes that include it. The ActionController::API class uses a subset of the modules that the Base class includes, displaying a use of composition.

Sources