我正在寻找一种方法来使用kombu作为tornado-sockjs和Django应用服务器之间的MQ适配器。我做了类似的事情:
class BrokerClient(ConsumerMixin):
clients = []
def __init__(self):
self.connection = BrokerConnection(settings.BROKER_URL)
self.io_loop = ioloop.IOLoop.instance()
self.queue = sockjs_queue
self._handle_loop()
@staticmethod
def instance():
if not hasattr(BrokerClient, '_instance'):
BrokerClient._instance = BrokerClient()
return BrokerClient._instance
def add_client(self, client):
self.clients.append(client)
def remove_client(self, client):
self.clients.remove(client)
def _handle_loop(self):
try:
if self.restart_limit.can_consume(1):
for _ in self.consume(limit=5):
pass
except self.connection.connection_errors:
print ('Connection to broker lost. '
'Trying to re-establish the connection...')
self.io_loop.add_timeout(datetime.timedelta(0.0001), self._handle_loop)
def get_consumers(self, Consumer, channel):
return [Consumer([self.queue, ], callbacks=[self.process_task])]
def process_task(self, body, message):
for client in self.clients:
if hasattr(body, 'users') and client.user.pk in body.users:
client.send(body)
message.ack()
但龙卷风阻止了_handle_loop的执行(如预期的那样)。
有什么方法可以阻止这种情况吗?
我知道Tikaado的Pika库适配器,但我想使用kombu因为它已经在项目中使用并且具有灵活的传输。
更新:
将_handle_loop更改为生成器功能
def drain_events(self, callback):
with self.Consumer() as (connection, channel, consumers):
with self.extra_context(connection, channel):
try:
connection.drain_events(timeout=1)
except:
pass
callback(None)
@tornado.gen.engine
def _handle_loop(self):
response = yield tornado.gen.Task(self.drain_events)
self.io_loop.add_timeout(datetime.timedelta(0.0001), self._handle_loop)
答案 0 :(得分:4)
最后,我找到了RabbitMQ后端的正确解决方案:
class BrokerClient(object):
clients = []
@staticmethod
def instance():
if not hasattr(BrokerClient, '_instance'):
BrokerClient._instance = BrokerClient()
return BrokerClient._instance
def __init__(self):
self.connection = BrokerConnection(settings.BROKER_URL)
self.consumer = Consumer(self.connection.channel(), [queue, ], callbacks=[self.process_task])
self.consumer.consume()
io_loop = tornado.ioloop.IOLoop.instance()
for sock, handler in self.connection.eventmap.items():
def drain_nowait(fd, events):
handler()
io_loop.add_handler(sock.fileno(), drain_nowait, l.READ | l.ERROR)
def process_task(self, body, message):
#something
message.ack()
def add_client(self, client):
self.clients.append(client)
def remove_client(self, client):
self.clients.remove(client)
对于其他后端,您可以使用问题中公布的解决方案
注意:不适用于librabbitmq
答案 1 :(得分:2)
我有类似的需求,在Kombu / RabbitMQ和ZeroMQ之间切换非阻塞方式。解决方案是使用Gevent来修补套接字库,这样Kombu也会变得无阻塞。我的#34;主要"线程有Kombu drain_events回调运行,而在另一个gevent线程中我有一个从ZeroMQ套接字接收消息的循环。工作得很好。
这对librabbitmq也不起作用,因为它在C中没有受Gevent影响的自己的套接字内容。