Skip to content

Latest commit

 

History

History
executable file
·
786 lines (472 loc) · 23 KB

README.rst

File metadata and controls

executable file
·
786 lines (472 loc) · 23 KB

Example Messaging Patterns

Here are some straightforward, example-based tutorials for building basic messaging applications using [RabbitMQ](https://www.rabbitmq.com/download.html). To run the code examples, make sure both RabbitMQ and the [Pika](https://github.com/pika/pika/) Python library are installed.

Installation

Basic Patterns

This is a basic pattern for sending messages using RabbitMQ as the broker. The example consists of three components: the Producer, Queue, and Consumer. The producer creates and sends messages to the queue, which resides on the RabbitMQ server. The consumer then retrieves and processes these messages from the queue.

./images/basic.png

Producer

A producer is the application that sends the messages. Imagine sending text messages to your friend.

#!/usr/bin/env python
from queue import Queue

class Producer(Queue):
    def __init__(self):
        Queue.__init__(self, queue='basic')

if __name__ == '__main__':
    p = Producer()
    p.send('Basic - Hello World')
rangertaha@Coder:~/messaging-patterns/basic$ python producer.py
0 2014-04-22 00:19:59.497745 - Basic - Hello World sent
1 2014-04-22 00:19:59.497745 - Basic - Hello World sent
2 2014-04-22 00:19:59.497745 - Basic - Hello World sent
3 2014-04-22 00:19:59.497745 - Basic - Hello World sent
4 2014-04-22 00:19:59.497745 - Basic - Hello World sent

Use the rabbitmqctl command line admin tool to list the queues.

rangertaha@Coder:~/messaging-patterns/basic$ sudo rabbitmqctl list_queues
Listing queues ...
basic       5
...done.

Consumer

A consumer is the application that receives the messages. Imagine your friend who is receiving your text messages.

#!/usr/bin/env python
from queue import Queue

class Consumer(Queue):
    def __init__(self):
        Queue.__init__(self, queue='basic')

    def callback(self, ch, method, properties, body):
        print '{0} received '.format(body)

if __name__ == '__main__':
    p = Consumer()
    p.receive()

The following is the execution and output of the consumer.py script.

rangertaha@Coder:~/messaging-patterns/basic$ python consumer.py
0 2014-04-22 00:19:59.497745 - Basic - Hello World received
1 2014-04-22 00:19:59.497745 - Basic - Hello World received
2 2014-04-22 00:19:59.497745 - Basic - Hello World received
3 2014-04-22 00:19:59.497745 - Basic - Hello World received
4 2014-04-22 00:19:59.497745 - Basic - Hello World received

Queue

The Queue in this setup is the RabbitMQ server, which communicates using the AMQP protocol. It receives and stores messages, allowing the consumer to pick them up whenever it's ready. Think of it like texting a friend whose phone is off—your messages are held in the Queue until your friend turns their phone back on and receives them.

#!/usr/bin/env python
import pika

class Queue:
    def __init__(self, queue='queue', host='localhost', routing='route'):
        self.connection = pika.BlockingConnection(pika.ConnectionParameters(
        host=host))
        self.channel = self.connection.channel()
        self.channel.queue_declare(queue=queue)
        self.routing = routing
        self.queue = queue

    def send(self, msg):
        self.channel.basic_publish(exchange='', routing_key=self.queue,
                                   body=msg)

    def receive(self):
        self.channel.basic_consume(self.callback, queue=self.queue,
                                   no_ack=True)
        self.channel.start_consuming()

Worker

This example demonstrates a work queue, designed to distribute messages across multiple workers. It is the next simplest pattern after a basic messaging setup. Like before, this pattern has three components: the Producer, Queue, and Consumers. The producer creates and sends messages to the queue, which resides on the RabbitMQ server. Multiple consumers can connect to the queue, with messages distributed evenly among them. You can add as many consumers as needed to share the workload.

./images/worker.png

Producer

This producer functions just like in the previous basic example. It’s the application responsible for sending messages.

#!/usr/bin/env python
from datetime import datetime
from queue import Queue

class Producer(Queue):
    def __init__(self):
        Queue.__init__(self, queue='basic')

if __name__ == '__main__':
    NOW = datetime.now()
    p = Producer()
    for i in range(15):
        p.send('{0} {1} - Basic - Hello World'.format(i, NOW))
        print '{0} {1} - Basic - Hello World sent'.format(i, NOW)
    p.close()
rangertaha@Coder:~/messaging-patterns/workers$ python producer.py
0 2014-04-22 00:10:16.946810 - Basic - Hello World sent
1 2014-04-22 00:10:16.946810 - Basic - Hello World sent
2 2014-04-22 00:10:16.946810 - Basic - Hello World sent
3 2014-04-22 00:10:16.946810 - Basic - Hello World sent
4 2014-04-22 00:10:16.946810 - Basic - Hello World sent
5 2014-04-22 00:10:16.946810 - Basic - Hello World sent
6 2014-04-22 00:10:16.946810 - Basic - Hello World sent
7 2014-04-22 00:10:16.946810 - Basic - Hello World sent
8 2014-04-22 00:10:16.946810 - Basic - Hello World sent
9 2014-04-22 00:10:16.946810 - Basic - Hello World sent
10 2014-04-22 00:10:16.946810 - Basic - Hello World sent
11 2014-04-22 00:10:16.946810 - Basic - Hello World sent
12 2014-04-22 00:10:16.946810 - Basic - Hello World sent
13 2014-04-22 00:10:16.946810 - Basic - Hello World sent
14 2014-04-22 00:10:16.946810 - Basic - Hello World sent
rangertaha@Coder:~/messaging-patterns/workers$

Use the rabbitmqctl command line admin tool to list the queues.

rangertaha@Coder:~/messaging-patterns/workers$ sudo rabbitmqctl list_queues
Listing queues ...
basic       15
...done.

Consumers

A consumer is the application that receives messages. This consumer retrieves a message, prints it to the terminal, waits for 1 second, and then repeats the process.

#!/usr/bin/env python
import time
from queue import Queue

class Consumer(Queue):
    def __init__(self):
        Queue.__init__(self, queue='basic')

    def callback(self, ch, method, properties, body):
        print 'Received: {0}'.format(body)
        time.sleep(1)

if __name__ == '__main__':
    p = Consumer()
    p.receive(p.callback)

Here, I'm running three separate instances of consumer.py in different terminal windows. Notice that the numbers at the beginning of each line are unique—each consumer receives a different message from the set sent by the producer. Each consumer processes one message, waits for one second, and then repeats the process.

rangertaha@Coder:~/messaging-patterns/workers$ python consumer.py
0 2014-04-22 00:10:16.946810 - Basic - Hello World received
3 2014-04-22 00:10:16.946810 - Basic - Hello World received
6 2014-04-22 00:10:16.946810 - Basic - Hello World received
9 2014-04-22 00:10:16.946810 - Basic - Hello World received
12 2014-04-22 00:10:16.946810 - Basic - Hello World received
rangertaha@Coder:~/messaging-patterns/workers$ python consumer.py
1 2014-04-22 00:10:16.946810 - Basic - Hello World received
4 2014-04-22 00:10:16.946810 - Basic - Hello World received
7 2014-04-22 00:10:16.946810 - Basic - Hello World received
10 2014-04-22 00:10:16.946810 - Basic - Hello World received
13 2014-04-22 00:10:16.946810 - Basic - Hello World received
rangertaha@Coder:~/messaging-patterns/workers$ python consumer.py
2 2014-04-22 00:10:16.946810 - Basic - Hello World received
5 2014-04-22 00:10:16.946810 - Basic - Hello World received
8 2014-04-22 00:10:16.946810 - Basic - Hello World received
11 2014-04-22 00:10:16.946810 - Basic - Hello World received
14 2014-04-22 00:10:16.946810 - Basic - Hello World received

Queue

The Queue in this setup is the RabbitMQ server, which communicates using the AMQP protocol. It receives and stores messages, allowing the consumer to retrieve them when ready. Imagine it like sending a text to a friend whose phone is off—the messages you send are held in the Queue until your friend turns their phone back on to receive them.

#!/usr/bin/env python
import pika

class Queue:
    def __init__(self, queue='queue', host='localhost', routing='route'):
        self.connection = pika.BlockingConnection(pika.ConnectionParameters(
        host=host))
        self.channel = self.connection.channel()
        self.channel.queue_declare(queue=queue)
        self.routing = routing
        self.queue = queue

    def send(self, msg):
        self.channel.basic_publish(exchange='',
                      routing_key=self.queue,
                      body=msg)

    def receive(self, callback):
        self.channel.basic_consume(callback,
                      queue=self.queue,
                      no_ack=True)
        self.channel.start_consuming()

Publish/Subscribe

The publish/subscribe pattern enables a message to be delivered to multiple consumers, unlike the worker pattern. Here, the producer sends messages directly to an exchange, which then applies its rules to distribute the messages to multiple consumers.

./images/pubsub.png

Producer

The producer sends messages to the exchange. Same as in the basic example

#!/usr/bin/env python
from datetime import datetime
from exchange import Exchange

class Producer(Exchange):
    def __init__(self):
        Exchange.__init__(self, exchange='exchange-001', type='fanout')

    def send(self, msg):
        self.channel.basic_publish(exchange=self.exchange, routing_key='', body=msg)

if __name__ == '__main__':
    NOW = datetime.now()
    p = Producer()
    for i in range(5):
        p.send('{0} {1} - Pub/Sub - Hello World'.format(i, NOW))
        print '{0} {1} - Pub/Sub - Hello World sent'.format(i, NOW)
    p.close()
rangertaha@Coder:~/messaging-patterns/pubsub$ python producer.py
0 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World sent
1 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World sent
2 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World sent
3 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World sent
4 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World sent

Use the rabbitmqctl command line admin tool to list the queues.

rangertaha@Coder:~/messaging-patterns/pubsub$ sudo rabbitmqctl list_queues
Listing queues ...
basic       15
...done.

Exchange

The producer doesn’t send messages directly to a queue; instead, it sends them to an exchange. The exchange receives messages from producers and decides how to route them—either by delivering them to one or more queues or by discarding them. This routing behavior depends on the type of exchange. Here are the available exchange types:

The rules, known as the exchange types are: direct, topic, headers and fanout.

rangertaha@Coder:~/messaging-patterns/pubsub$ sudo rabbitmqctl list_exchanges
Listing exchanges ...
    direct
amq.direct  direct
amq.fanout  fanout
amq.headers headers
amq.match   headers
amq.rabbitmq.log    topic
amq.rabbitmq.trace  topic
amq.topic   topic
...done.

In terms of learning and clarification, I am representing the exchange as a class.

#!/usr/bin/env python
import pika
from queue import Queue

class Exchange(Queue):
    def __init__(self, exchange='exchange-001', type='fanout'):
        Queue.__init__(self)
        self.channel.exchange_declare(exchange=exchange, type=type)
        self.exchange = exchange
        self.type = type

Consumers

#!/usr/bin/env python
import time
from exchange import Exchange


class Consumer(Exchange):
    def __init__(self):
        Exchange.__init__(self, exchange='exchange-001', type='fanout')
        self.bind()

    def bind(self):
        result = self.channel.queue_declare(exclusive=True)
        self.channel.queue_bind(exchange=self.exchange, queue=result.method.queue)
        self.queue = result.method.queue

    def callback(self, ch, method, properties, body):
        print '{0} received '.format(body)
        #time.sleep(5)

if __name__ == '__main__':
    p = Consumer()
    p.receive(p.callback)

Here, I'm running three separate instances of consumer.py in different terminals. You’ll notice that the numbers at the beginning of each line are unique. Each consumer receives a different message from the set of messages sent by the producer.

rangertaha@Coder:~/messaging-patterns/pubsub$ tty
/dev/pts/7

rangertaha@Coder:~/messaging-patterns/pubsub$ python consumer.py
0 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
1 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
2 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
3 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
4 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
rangertaha@Coder:~/messaging-patterns/pubsub$ tty
/dev/pts/4

rangertaha@Coder:~/messaging-patterns/pubsub$ python consumer.py
0 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
1 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
2 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
3 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
4 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
rangertaha@Coder:~/messaging-patterns/pubsub$ tty
/dev/pts/9

rangertaha@Coder:~/messaging-patterns/pubsub$ python consumer.py
0 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
1 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
2 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
3 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received
4 2014-04-22 09:39:16.483488 - Pub/Sub - Hello World received

Queue

The Queue is the RabbitMQ server, which uses AMQP for communication. It receives and stores messages, allowing the consumer to retrieve them when ready.

#!/usr/bin/env python
import pika

class Queue:
    def __init__(self, queue='queue', host='localhost', routing='route'):
        self.connection = pika.BlockingConnection(pika.ConnectionParameters(
        host=host))
        self.channel = self.connection.channel()
        self.channel.queue_declare(queue=queue)
        self.routing = routing
        self.queue = queue

    def send(self, msg):
        self.channel.basic_publish(exchange='',
                      routing_key=self.queue,
                      body=msg)

    def receive(self, callback):
        self.channel.basic_consume(callback,
                      queue=self.queue,
                      no_ack=True)
        self.channel.start_consuming()

Routing

This routing pattern uses the direct exchange type along with a routing_key. Consumers use this key to access the messages from the queue.

./images/routing.png

Producer

The producer sends messages to the exchange, which in this case is of the direct type. The producer also accepts an argument that is used as the routing_key.

#!/usr/bin/env python
import sys
from datetime import datetime
from exchange import Exchange

class Producer(Exchange):
    def __init__(self):
        Exchange.__init__(self, exchange='exchange_001', type='direct')
        self.routing=sys.argv[1]

    def send(self, msg):
        self.channel.basic_publish(exchange=self.exchange,
                                   routing_key=self.routing,
                                   body=msg)

if __name__ == '__main__':
    NOW = datetime.now()
    p = Producer()
    for i in range(5):
        p.send('{0} {1} - Routing - {2}'.format(i, NOW, p.routing))
        print '{0} {1} - Routing - {2} sent'.format(i, NOW, p.routing)
    p.close()

Below, you can see that I ran the producer with blue, red, and then green as a single argument. This argument is used as the routing_key, which consumers will need to retrieve the corresponding message. .. code-block:: bash

rangertaha@Coder:~/messaging-patterns/routing$ python producer.py blue 0 2014-04-22 12:08:08.657679 - Routing - blue sent 1 2014-04-22 12:08:08.657679 - Routing - blue sent 2 2014-04-22 12:08:08.657679 - Routing - blue sent 3 2014-04-22 12:08:08.657679 - Routing - blue sent 4 2014-04-22 12:08:08.657679 - Routing - blue sent rangertaha@Coder:~/messaging-patterns/routing$ python producer.py red 0 2014-04-22 12:08:12.715046 - Routing - red sent 1 2014-04-22 12:08:12.715046 - Routing - red sent 2 2014-04-22 12:08:12.715046 - Routing - red sent 3 2014-04-22 12:08:12.715046 - Routing - red sent 4 2014-04-22 12:08:12.715046 - Routing - red sent rangertaha@Coder:~/messaging-patterns/routing$ python producer.py green 0 2014-04-22 12:08:19.934197 - Routing - green sent 1 2014-04-22 12:08:19.934197 - Routing - green sent 2 2014-04-22 12:08:19.934197 - Routing - green sent 3 2014-04-22 12:08:19.934197 - Routing - green sent 4 2014-04-22 12:08:19.934197 - Routing - green sent
rangertaha@Coder:~/Projects/messaging-patterns/pubsub$ sudo rabbitmqctl list_bindings
Listing bindings ...
    exchange        amq.gen-BXvvwbg12wVC3XJsPQPz9A  queue   amq.gen-BXvvwbg12wVC3XJsPQPz9A  []
    exchange        basic   queue   basic   []
    exchange        queue   queue   queue   []
exchange-001        exchange        amq.gen-BXvvwbg12wVC3XJsPQPz9A  queue   amq.gen-BXvvwbg12wVC3XJsPQPz9A  []
...done.

Exchange

The exchange receives messages from the producer and routes them to queues. It determines how to handle each message, with options to send it to a single queue, multiple queues, or discard it entirely. The routing decision depends on the type of exchange.

This example uses the direct exchange type. For clarity, I am representing the exchange as a class.

#!/usr/bin/env python
import pika
from queue import Queue

class Exchange(Queue):
    def __init__(self, exchange='exchange_001', type='direct'):
        Queue.__init__(self)
        self.channel.exchange_declare(exchange=exchange, type=type)
        self.exchange = exchange
        self.type = type

Consumers

A consumer is the application that receives messages. This consumer accepts one argument, which is used as the routing_key. It then prints all messages with that routing_key to the terminal.

#!/usr/bin/env python
import sys
import time
from exchange import Exchange


class Consumer(Exchange):
    def __init__(self):
        Exchange.__init__(self, exchange='exchange_001', type='direct')
        self.routing = sys.argv[1]
        self.bind()

    def bind(self):
        result = self.channel.queue_declare(exclusive=True)
        self.channel.queue_bind(exchange=self.exchange, queue=result.method
        .queue, routing_key=self.routing)
        self.queue = result.method.queue


    def callback(self, ch, method, properties, body):
        print '{0} received '.format(body)


if __name__ == '__main__':
    p = Consumer()
    p.receive()

In these examples, the consumer is provided with an argument that serves as the routing_key. It then retrieves the messages associated with that routing_key.

rangertaha@Coder:~/messaging-patterns/pubsub$ tty
/dev/pts/3

rangertaha@Coder:~/messaging-patterns/routing$ python consumer.py blue
0 2014-04-22 12:08:08.657679 - Routing - blue received
1 2014-04-22 12:08:08.657679 - Routing - blue received
2 2014-04-22 12:08:08.657679 - Routing - blue received
3 2014-04-22 12:08:08.657679 - Routing - blue received
4 2014-04-22 12:08:08.657679 - Routing - blue received
rangertaha@Coder:~/messaging-patterns/pubsub$ tty
/dev/pts/4

rangertaha@Coder:~/messaging-patterns/routing$ python consumer.py red
0 2014-04-22 12:08:12.715046 - Routing - red received
1 2014-04-22 12:08:12.715046 - Routing - red received
2 2014-04-22 12:08:12.715046 - Routing - red received
3 2014-04-22 12:08:12.715046 - Routing - red received
4 2014-04-22 12:08:12.715046 - Routing - red received
rangertaha@Coder:~/messaging-patterns/pubsub$ tty
/dev/pts/5

rangertaha@Coder:~/messaging-patterns/routing$ python consumer.py green
0 2014-04-22 12:08:19.934197 - Routing - green received
1 2014-04-22 12:08:19.934197 - Routing - green received
2 2014-04-22 12:08:19.934197 - Routing - green received
3 2014-04-22 12:08:19.934197 - Routing - green received
4 2014-04-22 12:08:19.934197 - Routing - green received

Queue

The Queue is the RabbitMQ server, which communicates using AMQP. It receives and stores messages, allowing the consumer to retrieve them when ready. .. code-block:: python

#!/usr/bin/env python import pika

class Queue:
def __init__(self, queue='queue', host='localhost', routing='route'):
self.connection = pika.BlockingConnection(pika.ConnectionParameters( host=host)) self.channel = self.connection.channel() self.channel.queue_declare(queue=queue) self.routing = routing self.queue = queue
def send(self, msg):
self.channel.basic_publish(exchange='',
routing_key=self.queue, body=msg)
def receive(self, callback):
self.channel.basic_consume(callback,
queue=self.queue, no_ack=True)

self.channel.start_consuming()

ToDo

  • Topics
  • Remote Procedure Call (RPC)

Topics

./images/topic.png