Sto utilizzando Pika per elaborare i dati da RabbitMQ. Come mi è sembrato di imbattersi in diversi tipi di problemi, ho deciso di scrivere una piccola applicazione di test per vedere come posso gestire i disconnessi.RabbitMQ, Pika e strategia di riconnessione
Ho scritto questo app di prova che non seguente:
- Connetti a Broker, riprovare fino a quando il successo
- Quando è collegato creare una coda.
- Consumare questa coda e mettere risultare in un pitone Queue.Queue (0) voce
- ottenere da Queue.Queue (0) e produrre di nuovo nella coda di broker.
Quello che ho notato sono stati 2 problemi:
- Quando eseguo il mio script da un host connessione a RabbitMQ su un altro host (all'interno di una VM), allora questo script esce in momenti casuali senza produrre un errore.
- Quando eseguo il mio script sullo stesso host su cui è installato RabbitMQ, gira bene e continua a funzionare.
Questo potrebbe essere spiegato a causa di problemi di rete, i pacchetti sono scesi anche se trovo la connessione non molto robusta.
Quando lo script viene eseguito localmente sul server RabbitMQ e mi uccidere il RabbitMQ allora lo script esce con l'errore: "ERRORE pika SelectConnection: Errore socket il 3: 104"
così sembra che non posso ottenere la strategia di riconnessione funzionante come dovrebbe essere. Qualcuno potrebbe dare un'occhiata al codice, quindi guarda cosa sto facendo male?
Grazie,
Jay
#!/bin/python
import logging
import threading
import Queue
import pika
from pika.reconnection_strategies import SimpleReconnectionStrategy
from pika.adapters import SelectConnection
import time
from threading import Lock
class Broker(threading.Thread):
def __init__(self):
threading.Thread.__init__(self)
self.logging = logging.getLogger(__name__)
self.to_broker = Queue.Queue(0)
self.from_broker = Queue.Queue(0)
self.parameters = pika.ConnectionParameters(host='sandbox',heartbeat=True)
self.srs = SimpleReconnectionStrategy()
self.properties = pika.BasicProperties(delivery_mode=2)
self.connection = None
while True:
try:
self.connection = SelectConnection(self.parameters, self.on_connected, reconnection_strategy=self.srs)
break
except Exception as err:
self.logging.warning('Cant connect. Reason: %s' % err)
time.sleep(1)
self.daemon=True
def run(self):
while True:
self.submitData(self.from_broker.get(block=True))
pass
def on_connected(self,connection):
connection.channel(self.on_channel_open)
def on_channel_open(self,new_channel):
self.channel = new_channel
self.channel.queue_declare(queue='sandbox', durable=True)
self.channel.basic_consume(self.processData, queue='sandbox')
def processData(self, ch, method, properties, body):
self.logging.info('Received data from broker')
self.channel.basic_ack(delivery_tag=method.delivery_tag)
self.from_broker.put(body)
def submitData(self,data):
self.logging.info('Submitting data to broker.')
self.channel.basic_publish(exchange='',
routing_key='sandbox',
body=data,
properties=self.properties)
if __name__ == '__main__':
format=('%(asctime)s %(levelname)s %(name)s %(message)s')
logging.basicConfig(level=logging.DEBUG, format=format)
broker=Broker()
broker.start()
try:
broker.connection.ioloop.start()
except Exception as err:
print err
Grazie per aver dedicato del tempo passando attraverso il codice e trovare tutte le problematiche ad essa collegate. Attualmente sto usando http://barryp.org/software/py-amqplib/ che è una libreria più semplice/più semplice ma che soddisfa completamente le mie esigenze. In combinazione con gevent, ho dei risultati davvero interessanti. Di questi tempi non mi annoio più con Pika. –
puoi usare Channel.confirm_delivery() per attendere ack dopo la pubblicazione, una volta chiusa la connessione, verrà timeout, quindi saprai che il messaggio non viene consegnato al broker –