#-- # Licensed to the Apache Software Foundation (ASF) under one # or more contributor license agreements. See the NOTICE file # distributed with this work for additional information # regarding copyright ownership. The ASF licenses this file # to you under the Apache License, Version 2.0 (the # "License"); you may not use this file except in compliance # with the License. You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, # software distributed under the License is distributed on an # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY # KIND, either express or implied. See the License for the # specific language governing permissions and limitations # under the License. #++ require 'qpid_proton' require 'optparse' require 'pathname' # Thread safe message queue that notifies waiting senders when messages arrive. class MessageQueue def initialize @lock = Mutex.new # Make ations on the queue atomic @messages = [] # Messages on the queue @waiting = [] # Senders that are waiting for messages end # Push a message onto the queue and notify any waiting senders def push(message) @lock.synchronize do @messages << message unless @waiting.empty? # Notify waiting senders # NOTE: the call to self.send_to is added to the sender's work_queue, # and will be executed in the sender's thread @waiting.each { |s| s.work_queue.add { self.send_to(s); } } @waiting.clear end end end # Pop a message off the queue. # If no messages available, record sender as waiting and return nil. def pop(sender) @lock.synchronize do if @messages.empty? @waiting << sender nil else @messages.shift end end end # NOTE: Called in sender's thread. # Pull messages from the queue as long as sender has credit. # If queue runs out of messages, record sender as waiting. def send_to(sender) while sender.credit > 0 && (message = pop(sender)) sender.send(message) end end def forget(sender) @lock.synchronize { @waiting.delete(sender) } end end # Handler for broker connections. In a multi-threaded application you should # normally create a separate handler instance for each connection. class BrokerHandler < Qpid::Proton::MessagingHandler def initialize(broker) @broker = broker end def on_sender_open(sender) if sender.remote_source.dynamic? sender.source.address = SecureRandom.uuid elsif sender.remote_source.address sender.source.address = sender.remote_source.address else sender.connection.close("no source address") return end q = @broker.queue(sender.source.address) q.send_to(sender) end def on_receiver_open(receiver) if receiver.remote_target.address receiver.target.address = receiver.remote_target.address else receiver.connection.close("no target address") end end def on_sender_close(sender) q = @broker.queue(sender.source.address) q.forget(sender) if q end def on_connection_close(connection) connection.each_sender { |s| on_sender_close(s) } end def on_transport_close(transport) transport.connection.each_sender { |s| on_sender_close(s) } end def on_sendable(sender) @broker.queue(sender.source.address).send_to(sender) end def on_message(delivery, message) @broker.queue(delivery.receiver.target.address).push(message) end end # Broker manages the queues and accepts incoming connections. class Broker < Qpid::Proton::Listener::Handler def initialize @queues = {} @connection_options = {} ssl_setup end def ssl_setup # Optional SSL setup ssl = Qpid::Proton::SSLDomain.new(Qpid::Proton::SSLDomain::MODE_SERVER) cert_passsword = "tserverpw" if Gem.win_platform? # Use P12 certs for windows schannel ssl.credentials("ssl-certs/tserver-certificate.p12", "", cert_passsword) else ssl.credentials("ssl-certs/tserver-certificate.pem", "ssl-certs/tserver-private-key.pem", cert_passsword) end ssl.allow_unsecured_client # SSL is optional, this is not secure. @connection_options[:ssl_domain] = ssl if ssl rescue # Don't worry if we can't set up SSL. end def on_open(l) STDOUT.puts "Listening on #{l.port}\n"; STDOUT.flush end # Create a new BrokerHandler instance for each connection we accept def on_accept(l) { :handler => BrokerHandler.new(self) }.update(@connection_options) end def queue(address) @queues[address] ||= MessageQueue.new end end if ARGV.size != 1 STDERR.puts "Usage: #{__FILE__} URL Start an example broker listening on URL" return 1 end url, = ARGV container = Qpid::Proton::Container.new container.listen(url, Broker.new) # Run the container in multiple threads. threads = 4.times.map { Thread.new { container.run }} threads.each { |t| t.join }