PHP

RabbitMQ + PHP quick start

1 Установка RabbitMQ

Сам RMQ на Убунту ставится так:

echo 'deb http://www.rabbitmq.com/debian/ testing main' | sudo tee /etc/apt/sources.list.d/rabbitmq.list
wget -O- https://www.rabbitmq.com/rabbitmq-release-signing-key.asc | sudo apt-key add -
sudo apt-get update
sudo apt-get install rabbitmq-server
sudo rabbitmq-plugins enable rabbitmq_management


Теперь если открыть ваш localhost:15672 под учеткой: guest guest и увидеть много прикольных вещей.


По умолчанию гостевой пользователь существует и может подключаться только с localhost.
Создание пользователя с правами админа с доступом отовсюду выглядит так:

sudo rabbitmqctl add_user admin StrongPassword
sudo rabbitmqctl set_user_tags admin administrator

По умолчанию новый пользователь не имеет доступа к Virtual Hosts.
Вам следует войти в админку, там на вкладке "Admin" нажать кнопку "Ser Permission" для нового пользователя.

Важная инфа:

  • Порт 15672 - для Веб интерфейса rabbitmq
  • Порт 5672 - для подключения клиента rabbitmq

 

 

2 Пример RabbitMQ на PHP

Пример будет показан на консольном приложении где один процесс будет рассылать сообщения, а другой принимать.

Установим RMQ-PHP библиотеку в корне проекта примера, например /var/www/rmq:

composer require php-amqplib/php-amqplib
  • Код фаила конфигурации config.php
<?php

require_once __DIR__ . '/vendor/autoload.php';

putenv('TEST_AMQP_DEBUG=1');
require_once __DIR__ . '/config.php';

const HOST = 'localhost';
const PORT = '5672';
const USER = 'admin';
const PASS = 'StrongPassword';
const VHOST = '/';


Нужно будет открыть 2 вкладки в терминале, условимся называть одну - Издатель, другую - Подписчик

  • Код Издателя amqp_publisher.php
<?php

include(__DIR__ . '/config.php');

use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Exchange\AMQPExchangeType;
use PhpAmqpLib\Message\AMQPMessage;

$exchange = 'router';
$queue = 'msgs';

$connection = new AMQPStreamConnection(HOST, PORT, USER, PASS, VHOST);
$channel = $connection->channel();

/*
    The following code is the same both in the consumer and the producer.
    In this way we are sure we always have a queue to consume from and an
        exchange where to publish messages.
*/

/*
    name: $queue
    passive: false
    durable: true // the queue will survive server restarts
    exclusive: false // the queue can be accessed in other channels
    auto_delete: false //the queue won't be deleted once the channel is closed.
*/
$channel->queue_declare($queue, false, true, false, false);

/*
    name: $exchange
    type: direct
    passive: false
    durable: true // the exchange will survive server restarts
    auto_delete: false //the exchange won't be deleted once the channel is closed.
*/

$channel->exchange_declare($exchange, AMQPExchangeType::DIRECT, false, true, false);

$channel->queue_bind($queue, $exchange);

$messageBody = implode(' ', array_slice($argv, 1));
$message = new AMQPMessage($messageBody, array('content_type' => 'text/plain', 'delivery_mode' => AMQPMessage::DELIVERY_MODE_PERSISTENT));
$channel->basic_publish($message, $exchange);

$channel->close();
$connection->close();

 

  • Код Подписчика amqp_consumer.php
<?php

include(__DIR__ . '/config.php');
use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Exchange\AMQPExchangeType;

$exchange = 'router';
$queue = 'msgs';
$consumerTag = 'consumer';

$connection = new AMQPStreamConnection(HOST, PORT, USER, PASS, VHOST);
$channel = $connection->channel();

/*
    The following code is the same both in the consumer and the producer.
    In this way we are sure we always have a queue to consume from and an
        exchange where to publish messages.
*/

/*
    name: $queue
    passive: false
    durable: true // the queue will survive server restarts
    exclusive: false // the queue can be accessed in other channels
    auto_delete: false //the queue won't be deleted once the channel is closed.
*/
$channel->queue_declare($queue, false, true, false, false);

/*
    name: $exchange
    type: direct
    passive: false
    durable: true // the exchange will survive server restarts
    auto_delete: false //the exchange won't be deleted once the channel is closed.
*/

$channel->exchange_declare($exchange, AMQPExchangeType::DIRECT, false, true, false);

$channel->queue_bind($queue, $exchange);

/**
 * @param \PhpAmqpLib\Message\AMQPMessage $message
 */
function process_message($message)
{
    echo "\n--------\n";
    echo $message->body;
    echo "\n--------\n";

    $message->ack();

    // Send a message with the string "quit" to cancel the consumer.
    if ($message->body === 'quit') {
        $message->getChannel()->basic_cancel($message->getConsumerTag());
    }
}

/*
    queue: Queue from where to get the messages
    consumer_tag: Consumer identifier
    no_local: Don't receive messages published by this consumer.
    no_ack: If set to true, automatic acknowledgement mode will be used by this consumer. See https://www.rabbitmq.com/confirms.html for details.
    exclusive: Request exclusive consumer access, meaning only this consumer can access the queue
    nowait:
    callback: A PHP Callback
*/

$channel->basic_consume($queue, $consumerTag, false, false, false, false, 'process_message');

/**
 * @param \PhpAmqpLib\Channel\AMQPChannel $channel
 * @param \PhpAmqpLib\Connection\AbstractConnection $connection
 */
function shutdown($channel, $connection)
{
    $channel->close();
    $connection->close();
}

register_shutdown_function('shutdown', $channel, $connection);

// Loop as long as the channel has callbacks registered
while ($channel->is_consuming()) {
    $channel->wait();
}

 

  • Теперь во вкладке терминала "Подписчик" выполним команду
php amqp_consumer.php


Если ошибок не появится значит клиент успешно подключился к RMQ и процесс перешел в режим ожидания сообщений

 

  • Теперь во вкладке терминала "Издатель" выполним команду
php amqp_publisher.php some text to publish

Издатель отправит сообщение на сервер RMQ который в свою очередь перешлет его всем подписчикам
Вернемся на вкладку "Подписчик" и увидим что он получил сообщение от издателя.

Источники:

Как установить RabbitMQ на Ubuntu 20.04 LTS https://infoit.com.ua/linux/kak-ustanovit-rabbitmq-na-ubuntu-20-04-lts/
Symfony + RabbitMQ Быстрый старт для молодых https://habr.com/ru/post/338950/
php-amqplib/php-amqplib https://github.com/php-amqplib/php-amqplib

 

 

Афоризм дня:
Многочисленность фактов и сочинений растет так быстро, что в недалеком будущем придется сводить все к извлечениям и словарям. (547)

Leave a reply

Яндекс.Метрика