lib/cucumber/core/event_bus.rb
require 'cucumber/core/events' module Cucumber module Core # Event Bus # # Implements an in-process pub-sub event broadcaster allowing multiple observers # to subscribe to events that fire as your tests are executed. # class EventBus attr_reader :event_types # @param [Hash{Symbol => Class}] a hash of event types to use on the bus def initialize(registry = Events.registry) @event_types = registry.freeze @handlers = {} @event_queue = [] end # Register for an event. The handler proc will be called back with each of the attributes # of the event. def on(event_id, handler_object = nil, &handler_proc) handler = handler_proc || handler_object validate_handler_and_event_id!(handler, event_id) event_class = event_types[event_id] handlers_for(event_class) << handler broadcast_queued_events_to handler, event_class end # Broadcast an event def broadcast(event) raise ArgumentError, "Event type #{event.class} is not registered. Try one of these:\n#{event_types.values.join("\n")}" unless is_registered_type?(event.class) handlers_for(event.class).each { |handler| handler.call(event) } @event_queue << event end def method_missing(event_id, *args) event_class = event_types.fetch(event_id) { super } broadcast event_class.new(*args) end private def broadcast_queued_events_to(handler, event_type) @event_queue.select { |event| event.class == event_type }.each { |event| handler.call(event) } end def handlers_for(event_class) @handlers[event_class.to_s] ||= [] end def is_registered_id?(event_id) event_types.keys.include?(event_id) end def is_registered_type?(event_type) event_types.values.include?(event_type) end def validate_handler_and_event_id!(handler, event_id) raise ArgumentError, "Please pass either an object or a handler block" unless handler raise ArgumentError, "Please use a symbol for the event_id" unless event_id.is_a?(Symbol) raise ArgumentError, "Event ID #{event_id} is not recognised. Try one of these:\n#{event_types.keys.join("\n")}" unless is_registered_id?(event_id) end end end end