require 'stomp' require 'forwardable' module MessageDriver class Broker def stomp_adapter MessageDriver::Adapters::StompAdapter end end module Adapters class StompAdapter < Base class Message < MessageDriver::Message::Base attr_reader :stomp_message def initialize(ctx, stomp_message) @stomp_message = stomp_message super(ctx, stomp_message.body, stomp_message.headers, {}) end end class Destination < MessageDriver::Destination::Base def queue_path @queue_path ||= begin name.start_with?('/') ? name : "/queue/#{name}" end end end attr_reader :config, :poll_timeout def initialize(broker, config) validate_stomp_version @broker = broker @config = config.symbolize_keys connect_headers = @config[:connect_headers] ||= {} connect_headers.symbolize_keys connect_headers[:"accept-version"] = '1.1,1.2' vhost = @config.delete(:vhost) connect_headers[:host] = vhost if vhost @poll_timeout = 1 end class StompContext < ContextBase extend Forwardable def_delegators :adapter, :with_connection, :poll_timeout # def subscribe(destination, consumer) # destination.subscribe(&consumer) # end def create_destination(name, dest_options = {}, message_props = {}) Destination.new(adapter, name, dest_options, message_props) end def publish(destination, body, headers = {}, _properties = {}) with_connection do |connection| connection.publish(destination.queue_path, body, headers) end end def pop_message(destination, options = {}) with_connection do |connection| sub_id = connection.uuid msg = nil count = 0 connection.subscribe(destination.queue_path, options, sub_id) while msg.nil? && count < max_poll_count msg = connection.poll if msg.nil? count += 1 sleep 0.1 end end connection.unsubscribe(destination.queue_path, options, sub_id) Message.new(self, msg) if msg end end private def max_poll_count (poll_timeout / 0.1).to_i end end def build_context StompContext.new(self) end def with_connection @connection ||= open_connection yield @connection rescue SystemCallError, IOError => e raise MessageDriver::ConnectionError.new(e) end def stop super @connection.disconnect if @connection end private def open_connection conn = Stomp::Connection.new(@config) fail MessageDriver::ConnectionError, conn.connection_frame.to_s unless conn.open? conn end def validate_stomp_version required = Gem::Requirement.create('~> 1.3.1') current = Gem::Version.create(Stomp::Version::STRING) unless required.satisfied_by? current fail MessageDriver::Error, 'stomp 1.3.1 or a later version of the 1.3.x series is required for the stomp adapter' end end end end end