😮 You looked at the source!

Dimitris Zorbas

Code spells, smells and sourcery

The 10-minute Rails Pub/Sub

This time we’ll experiment with a quick way to architecture a Rails application to use Pub/Sub instead of model callbacks.

What’s wrong with callbacks

Rails active record models easily become bloated, that’s where most of the business logic tends to live after all. One of the most common sources of technical debt in Rails apps is callbacks. Models become god-objects with dependencies to other models, mailers and even 3rd party services.

When it comes to refactoring this coupling, I usually recommend extracting all callbacks to stateless functions which can be composed to form pipelines. One can use dry-transaction for that. My love for such composable architectures led me to create Opus for Elixir.

I’m also quite proud that callbacks got deprecated in Ecto 🎉.

About Pub/Sub

The solution which is the focus of this post is Pub/Sub. The models will publish events concerning database updates. A database record gets created / updated / destroyed and then a subscriber does something, or ignores the event.

Enter ActiveSupport::Notifications

We’ll lay the foundations for this ten minute implementation on top of ActiveSupport::Notifications. Originally introduced as an instrumentation API for Rails, but there’s nothing preventing us from using it for custom events.

Some facts about ActiveSupport::Notifications.

  • It’s basically a thread-safe queue
  • Events are synchronous
  • Events are process-local
  • It’s simple to use 😎

The Code

In this experiment, we’ll cover the following scenario:

When a User "zorbash" is created
And "zorbash" had been invited by "gandalf"
Then the field signups_count for "gandalf" should increase by 1

First we’ll create a model concern which we can include to our User model to publish events each time a record is created.

# frozen_string_literal: true

module Publishable
  extend ActiveSupport::Concern

  included do
    after_create_commit :publish_create
    after_update_commit :publish_update
    after_destroy_commit :publish_destroy
  end

  class_methods do
    def subscribe(event = :any)
      event_name = event == :any ? /#{table_name}/ : "#{table_name}.#{event}"

      ActiveSupport::Notifications.subscribe(event_name) do |_event_name, **payload|
        yield payload
      end

      self
    end
  end

  private

  def publish_create
    publish(:create)
  end

  def publish_update
    publish(:update)
  end

  def publish_destroy
    publish(:destroy)
  end

  def publish(event)
    event_name = "#{self.class.table_name}.#{event}"

    ActiveSupport::Notifications.publish(event_name, event: event, model: self)
  end
end

Then we must include it in our model.

# frozen_string_literal: true

class User < ApplicationRecord
  include Publishable # 👈 Added here

  devise :invitable

  # other omitted code
end

Let’s implement a subscriber.

module UserSubscriber
  extend self

  def subscribe
    User.subscribe(:create) do |event|
      event[:model].increment!(:signups_count)
    end
  end
end

Finally, we have to initialize the subscription.

# File: config/initializers/subscriptions.rb
Rails.application.config.after_initialize do
  UserSubscriber.subscribe
end

Caveats

The more listeners you add, the slower it becomes for an event to be handled in sequence across all listeners. This is similar to how an object would call all callback handler methods one after the other.

See: active_support/notifications/fanout.rb

def publish(name, *args)
  listeners_for(name).each { |s| s.publish(name, *args) }
end

They’re also not suitable for callbacks used to mutate a record like before_validation or after_initialize.

Furthermore there are no guarantees that an event will be processed successfully. Where things can go wrong, will go wrong. Prefer a solution with robust recovery semantics.

Next Steps

For enhanced flexibility, we can push events to Redis or RabbitMQ or Kafka. How to pick one according to your needs is beyond the scope of this post. However you can consider yourself lucky, since there are tons of resources out there and mature libraries to build your event-driven system on top of.

Alternatives

Notable Pub/Sub gems:

For other handy libraries and posts, subscribe to my Tefter Ruby & Rails list.