2023-05-12 15:06:01 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is part of the Symfony package.
|
|
|
|
*
|
|
|
|
* (c) Fabien Potencier <fabien@symfony.com>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Symfony\Component\Messenger\Bridge\Amqp\Transport;
|
|
|
|
|
|
|
|
use Symfony\Component\Messenger\Envelope;
|
|
|
|
use Symfony\Component\Messenger\Exception\LogicException;
|
|
|
|
use Symfony\Component\Messenger\Exception\MessageDecodingFailedException;
|
|
|
|
use Symfony\Component\Messenger\Exception\TransportException;
|
|
|
|
use Symfony\Component\Messenger\Transport\Receiver\MessageCountAwareInterface;
|
|
|
|
use Symfony\Component\Messenger\Transport\Receiver\QueueReceiverInterface;
|
|
|
|
use Symfony\Component\Messenger\Transport\Serialization\PhpSerializer;
|
|
|
|
use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Symfony Messenger receiver to get messages from AMQP brokers using PHP's AMQP extension.
|
|
|
|
*
|
|
|
|
* @author Samuel Roze <samuel.roze@gmail.com>
|
|
|
|
*/
|
|
|
|
class AmqpReceiver implements QueueReceiverInterface, MessageCountAwareInterface
|
|
|
|
{
|
|
|
|
private $serializer;
|
|
|
|
private $connection;
|
|
|
|
|
2024-05-17 20:26:01 +02:00
|
|
|
public function __construct(Connection $connection, ?SerializerInterface $serializer = null)
|
2023-05-12 15:06:01 +02:00
|
|
|
{
|
|
|
|
$this->connection = $connection;
|
|
|
|
$this->serializer = $serializer ?? new PhpSerializer();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function get(): iterable
|
|
|
|
{
|
|
|
|
yield from $this->getFromQueues($this->connection->getQueueNames());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function getFromQueues(array $queueNames): iterable
|
|
|
|
{
|
|
|
|
foreach ($queueNames as $queueName) {
|
|
|
|
yield from $this->getEnvelope($queueName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getEnvelope(string $queueName): iterable
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
$amqpEnvelope = $this->connection->get($queueName);
|
|
|
|
} catch (\AMQPException $exception) {
|
|
|
|
throw new TransportException($exception->getMessage(), 0, $exception);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (null === $amqpEnvelope) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$body = $amqpEnvelope->getBody();
|
|
|
|
|
|
|
|
try {
|
|
|
|
$envelope = $this->serializer->decode([
|
|
|
|
'body' => false === $body ? '' : $body, // workaround https://github.com/pdezwart/php-amqp/issues/351
|
|
|
|
'headers' => $amqpEnvelope->getHeaders(),
|
|
|
|
]);
|
|
|
|
} catch (MessageDecodingFailedException $exception) {
|
|
|
|
// invalid message of some type
|
|
|
|
$this->rejectAmqpEnvelope($amqpEnvelope, $queueName);
|
|
|
|
|
|
|
|
throw $exception;
|
|
|
|
}
|
|
|
|
|
|
|
|
yield $envelope->with(new AmqpReceivedStamp($amqpEnvelope, $queueName));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function ack(Envelope $envelope): void
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
$stamp = $this->findAmqpStamp($envelope);
|
|
|
|
|
|
|
|
$this->connection->ack(
|
|
|
|
$stamp->getAmqpEnvelope(),
|
|
|
|
$stamp->getQueueName()
|
|
|
|
);
|
|
|
|
} catch (\AMQPException $exception) {
|
|
|
|
throw new TransportException($exception->getMessage(), 0, $exception);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function reject(Envelope $envelope): void
|
|
|
|
{
|
|
|
|
$stamp = $this->findAmqpStamp($envelope);
|
|
|
|
|
|
|
|
$this->rejectAmqpEnvelope(
|
|
|
|
$stamp->getAmqpEnvelope(),
|
|
|
|
$stamp->getQueueName()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function getMessageCount(): int
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
return $this->connection->countMessagesInQueues();
|
|
|
|
} catch (\AMQPException $exception) {
|
|
|
|
throw new TransportException($exception->getMessage(), 0, $exception);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function rejectAmqpEnvelope(\AMQPEnvelope $amqpEnvelope, string $queueName): void
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
$this->connection->nack($amqpEnvelope, $queueName, \AMQP_NOPARAM);
|
|
|
|
} catch (\AMQPException $exception) {
|
|
|
|
throw new TransportException($exception->getMessage(), 0, $exception);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function findAmqpStamp(Envelope $envelope): AmqpReceivedStamp
|
|
|
|
{
|
|
|
|
$amqpReceivedStamp = $envelope->last(AmqpReceivedStamp::class);
|
|
|
|
if (null === $amqpReceivedStamp) {
|
|
|
|
throw new LogicException('No "AmqpReceivedStamp" stamp found on the Envelope.');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $amqpReceivedStamp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!class_exists(\Symfony\Component\Messenger\Transport\AmqpExt\AmqpReceiver::class, false)) {
|
|
|
|
class_alias(AmqpReceiver::class, \Symfony\Component\Messenger\Transport\AmqpExt\AmqpReceiver::class);
|
|
|
|
}
|