Optimizing test suites when using Rails Event Store

Using domain events in DDD make it easier to tackle complex workflows. If we are working in a monolith infrastructure, it may cause our event store to have thousands handlers and running all of them in test environment is short way for long test suites. However, there’s a trick which may allow you to increase the speed of your test suite by disabling unnecessary handlers.

The idea is to reinstantiate an event store configuration before each test. Developer will have the control over what will and what will not be run from the test file, by using rspec test metadata feature. You can probably find respective features in other testing frameworks as well.

But let’s start with the basics. If you are having only one instance of event store, there’s a chance that you have global configuration file. In one of our projects, it looks like this:

class Subscriptions
  def setup
    {
      Ordering::OrderCompleted => [
        SomeHandler,
        OtherHandler,
        GenerateOrderReceiptPdf,
      ],
      # ...
    }
  end

  # Switch from hash (event => [handlers]) to (handler => [events])
  def handlers
    setup.reduce({}) do |memo, (event, handlers)|
      handlers.each do |handler|
        memo[handler] ||= []
        memo[handler] << event
      end
      memo
    end
  end
end

There is a map, from each event to list of handlers it should trigger, and the method handlers, which converts to the opposite mapping – from each handler, to list of events on which it should react.

The first optimization may be, that we want to disable some handler, which is computing very long – for example, PDF generation.

RSpec.configure do |config|
  config.before(:each) do |_example|
    event_store = RailsEventStore::Client.new

    disabled_handlers = [
      GenerateOrderReceiptPdf,
    ]

    Subscriptions.new.handlers.except(*disabled_handlers).each do |handler, events|
      event_store.subscribe(handler, to: events)
    end

    Rails.configuration.event_store = event_store
  end
end

That should give a major speed up, but because handler for generating order receipt pdf is not running at all, some tests will probably fail. That’s why we want to tag the specific tests in which we are interested in that handler.

RSpec.configure do |config|
  config.before(:each) do |_example|
    event_store = RailsEventStore::Client.new

    disabled_handlers = [
      GenerateOrderReceiptPdf,
    ]
    if example.metadata[:enable_handlers]
      disabled_handlers -= Array(example.metadata[:enable_handlers])
    end

    Subscriptions.new.handlers.except(*disabled_handlers).each do |handler, events|
      event_store.subscribe(handler, to: events)
    end

    Rails.configuration.event_store = event_store
  end
end

# In tests:
describe "some pdf spec", enable_handlers: [GenerateOrderReceiptPdf] do
  # ...
end

There may also be situations, where you want to disable some handlers, but only in specific tests:

# ...
if example.metadata[:disable_handlers]
  disabled_handlers -= Array(example.metadata[:disable_handlers]) 
end
# ...

Last but not least, if you are working on fresh bounded context, in which you don’t have your usual legaccy baggage, you may want to disable whole default handlers configuration and enable the handlers selectively. The final code:

RSpec.configure do |config|
  config.before(:each) do |_example|
    event_store = RailsEventStore::Client.new

    if example.metadata[:only_handlers]
      only_handlers = example.metadata[:only_handlers]

      Subscriptions.new.handlers.slice(*only_handlers).each do |handler, events|
        event_store.subscribe(handler, to: events)
      end
    else
      disabled_handlers = [
        GenerateOrderReceiptPdf,
      ]
      if example.metadata[:enable_handlers]
        disabled_handlers -= Array(example.metadata[:enable_handlers])
      end
      if example.metadata[:disable_handlers]
        disabled_handlers -= Array(example.metadata[:disable_handlers]) 
      end

      Subscriptions.new.handlers.except(*disabled_handlers).each do |handler, events|
        event_store.subscribe(handler, to: events)
      end
    end

    Rails.configuration.event_store = event_store
  end
end

I hope that those of you who work with bigger applications and Rails Event Store, will find that snippet useful. If you would like to become better in working with complex rails applications and have more holistic knowledge about software architecture, consider joining Rails Architect MasterClass.