-
-
Notifications
You must be signed in to change notification settings - Fork 9.6k
[Messenger][Amqp] delayed quorum queues #60298
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: 7.2
Are you sure you want to change the base?
[Messenger][Amqp] delayed quorum queues #60298
Conversation
Hey! I see that this is your first PR. That is great! Welcome! Symfony has a contribution guide which I suggest you to read. In short:
Review the GitHub status checks of your pull request and try to solve the reported issues. If some tests are failing, try to see if they are failing because of this change. When two Symfony core team members approve this change, it will be merged and you will become an official Symfony contributor! I am going to sit back now and wait for the reviews. Cheers! Carsonbot |
I've already done it in a nodeJS project and it works well in production. async function createDelayedQueue(channel, exchange, queue, delay) {
const date = new Date().toISOString().split('T')[0];
const delayedQueue = `delay_${exchange}_${queue}_${delay}_${date}`;
const delayedExchange = 'delays';
const baseExpire = 24 * 60 * 60 * 1000;
await channel.assertQueue(delayedQueue, {
durable: true,
arguments: {
'x-message-ttl': delay,
'x-expires': delay + 60 * 1000 + baseExpire,
'x-dead-letter-exchange': exchange,
'x-dead-letter-routing-key': queue,
'x-queue-type': 'quorum'
}
});
await channel.bindQueue(delayedQueue, delayedExchange, delayedQueue);
return { delayedQueue, delayedExchange };
} |
This probably also affect 6.4, right? |
Yes, that's right |
Sorry, I've tried to change the base branch to 6.4 and by mistake it has added automatically the reviewers, feel free to leave 🙏 |
No worries, we can target 6.4 while merging this PR 👍 |
Hi there! Just a friendly ping to see if there's anything else needed on this PR or if it can be reviewed when you get a chance. Thanks a lot for your time! |
Hi everyone! |
src/Symfony/Component/Messenger/Bridge/Amqp/Transport/Connection.php
Outdated
Show resolved
Hide resolved
src/Symfony/Component/Messenger/Bridge/Amqp/Tests/Transport/ConnectionTest.php
Outdated
Show resolved
Hide resolved
As this is a new feature, it must target |
Well, I consider it as a bug fix, because is causing problems when you use symfony with quorum rabbitmq queues, but, for me it's not a problem target it to |
296358d
to
0480703
Compare
I would say that adding support for quorum queues could be considered as a new feature, and it seems safer to deliver this in a minor version. Can you target the 7.4 branch please ? |
Well, in fact, you can actually work with quorum queues in symfony 5.4, it's not a new feature, and nothing prevents you to use in delay queues, so, for my point of view is a bug, because if you use them, you'll lost messages, as the delay queue is removed. I don't thing that is a new feature, it is an already feature that didn't work as expected. |
Long live message concerns I have a real-life use case involving messages that need to be processed after a delay of more than 72 hours, and I wanted to clarify how the delay queues behave for messages published later in the day, after the very first message. After reviewing the PR and studying the underlying mechanism, the solution seems solid and should work as expected for long delays. Here’s an example scenario:
My concern was whether the second message (published at 11:00 PM) would be processed too early because it enters a delay queue that already exists due to the earlier message. Target version I’ve reviewed the discussion and would like to support the points made by @stof.
From what I see, the issue with temporary queues (retry/delay) only affects quorum queues. Classic queues have never been impacted by this bug, since their behavior remains unchanged. In my opinion, both perspectives are legitimate:
Personally, as a community member, I tend to agree with the view that this is a bugfix that naturally fits in 7.2. However, I also understand the reasoning for not backporting it to 6.4, given it was never supported there and that changing this in an LTS might have unexpected side effects for existing users. Just wanted to share my perspective, without polemic, as someone who has been affected by this limitation and appreciates the improvement brought by this PR. Alternative I would like to challenge another approach with you all. Given Symfony Messenger already supports per-message TTL for delays using DelayStamp on AMQP transports (the AMQP expiration property). We could introduce a new configuration to allow Messenger to use a single durable quorum delay queue without a queue-level TTL and rely entirely on per-message TTLs. By doing this, all messages keep their individual TTL, and expire themselves. RabbitMQ will automatically dead-letter them to the target queue using DLX. This would avoid the issues with dynamic creation/deletion of quorum delay queues. I am not an AMQP expert at all, but this idea crossed my mind and I wanted to share it with you. It seems fully compatible with Messenger’s current features and RabbitMQ’s capabilities, and it would simplify the delay queue logic, especially for users with quorum queues and long-lived messages. |
No need, we will do this while merging
No need, we do while merging
No need |
Hi,
Actually, these are two separate concepts: the queue's duration and the message's TTL. The second message will be dead-lettered to the final queue only after its own TTL expires, regardless of when the queue was created or when the first message entered it. The mechanism is exactly the same as the one already used today. The difference is that with classic queues, if you redeclare the queue, its expiration is refreshed. With quorum queues, this doesn't happen—so we must ensure that the queue lasts long enough for all its messages. That’s why I create a delay queue per day with a TTL of delay + 1 day: this way, we ensure messages are processed properly, and we can still clean up old queues.
This is not entirely accurate. You can set the default queue type for a RabbitMQ Virtual Host. That’s actually how I discovered this issue: in our setup, I’m not arguing whether this is a bug or a missing feature, and I’m OK with this being addressed in Symfony 7.4. My goal is simply to move the PR forward, because RabbitMQ is deprecating classic mirrored queues. In RabbitMQ 4,
This was actually the first solution I tried when we started migrating from Beanstalkd (which has native support for delayed messages, and is very simple to use). Unfortunately, this approach has a critical issue: delay queues process messages in order. If the first message has a delay of 5 minutes, and the second one only 5 seconds, the second one will wait for the first to expire before being processed. This is why Symfony Messenger creates one queue per delay duration. It avoids the head-of-line blocking problem that comes with FIFO processing in delay queues. As I said previously, we even have some services in Node.js that use the same pattern. They’ve been running smoothly for over 6 months, with quorum queues, no message loss, and proper queue cleanup. I know this is a complex topic to fully explain in a PR comment, and English is not my first language—so if anything is unclear, please let me know and I’ll try to explain it differently. Thanks again for your feedback and time. |
@miquel-angel thanks for your time answering my comments. I learn a lot. Big thanks for your contribution. I'm looking forward to your PR to be merged :D @OskarStark I edited my comment to remove inexact information. Thanks for the feedback. |
d53a3ce
to
0480703
Compare
…nnectionTest.php Co-authored-by: Oskar Stark <[email protected]>
0480703
to
40d03f5
Compare
Hi, Thanks! |
As described in the issue: #57867 when we create delayed quorum queues this doesn't update the expire time.
I've first tried what is suggested by rabbit maintainers here: rabbitmq/rabbitmq-server#5894 (comment) but it doesn't supported either.
So the final solution is create a delayed quorum queue per day, with expire of one day + ttl + 10 seconds, with this approach, when you enqueue a message to a delayed queue, you'll be sure that all the delayed message of the same day will be in the same queue and this queue won't die before the last message is processed.
For example you delay a message with 10s delay at 10:00 of 2025-04-29, this will create a delayed queue named:
delay_test_test_delay_2025-04-29 with expire on 2025-04-30 at 10:10:10.
If another message comes at 23:59:59, it will go to the same queue, that is already created.
And in the next day will create a new queue for the 30th.