Im Symfony Messenger gibt die Möglichkeit, fehlgeschlagene Nachrichten an eine failed queue zu leiten. In diesem Sammelbecken können diese dann inspiziert und nach Korrektur des zu Grunde liegenden Problems erneut verarbeitet werden.
Schlagen sie dabei erneut fehl, landen sie allerdings per Standard nicht wieder in der failed queue, sondern werden verworfen. Dies hat damit zu tun, dass die Nachricht bereits einen SentToFailureTransportStamp besitzt.
Möchte man jedoch die Nachricht immer wieder auffangen, bietet es sich an, diesen Stamp einfach vorher wieder zu entfernen. Dies kann mit der folgenden Middleware getan werden.
<?php
namespace App\...\Middleware;
use Symfony\Component\Messenger\Envelope;
use Symfony\Component\Messenger\Middleware\MiddlewareInterface;
use Symfony\Component\Messenger\Middleware\StackInterface;
use Symfony\Component\Messenger\Stamp\SentToFailureTransportStamp;
/**
* This middleware ensures that a message that had been on the failed queue and retried again
* fails again is sent to the failed queue instead of being discarded.
*
* @see https://github.com/symfony/symfony/blob/5.4/src/Symfony/Component/Messenger/EventListener/SendFailedMessageToFailureTransportListener.php#L63-L66
* @see https://github.com/symfony/symfony/issues/33685
* @see https://github.com/symfony/symfony/issues/35449
* @see https://github.com/symfony/symfony/issues/35459
*/classEnforceFailedQueueMiddlewareimplements MiddlewareInterface
{
publicfunctionhandle(Envelope $envelope, StackInterface $stack): Envelope
{
if (null!==$envelope->last(SentToFailureTransportStamp::class)) {
$envelope=$envelope->withoutAll(SentToFailureTransportStamp::class);
}
return$stack->next()->handle($envelope, $stack);
}
}
Übrigens ist bei Symfony derzeit eine Erweiterung in der Mache, mit der sich mehrere failed queues konfigurieren lassen. Dies eröffnet wiederum neue Einsatzmöglichkeiten. Zum Beispiel lassen sich so unterschiedliche Arten von Fehlern kanalisieren.