The feature has been around since Contao 5.1, but we’re describing how it works as of version 5.3.10 where it was revamped in order to address various issues with the previous implementation.
Contao provides an integration of the Symfony Messenger in the Contao Managed Edition which is documented here. This chapter assumes thorough understanding of the Symfony Messenger Component and its concepts. So in case you do not understand the concepts of
messenger:consume
please stop reading here and head to the Symfony documentation. We also recommend, you have worked with the Symfony Messenger in the context of a regular Symfony application before. It will help you differentiate between what is specific to Contao and what’s not.
The default Contao Managed Edition Symfony Messenger configuration looks like this:
framework:
messenger:
buses:
messenger.bus.default:
middleware:
- doctrine_ping_connection
- doctrine_close_connection
failure_transport: contao_failure
transports:
sync: sync://
contao_failure: doctrine://default?table_name=tl_message_queue&queue_name=failure&auto_setup=false
contao_prio_high: doctrine://default?table_name=tl_message_queue&queue_name=prio_high&auto_setup=false
contao_prio_normal: doctrine://default?table_name=tl_message_queue&queue_name=prio_normal&auto_setup=false
contao_prio_low: doctrine://default?table_name=tl_message_queue&queue_name=prio_low&auto_setup=false
The sync
transport as well as the contao_failure
transport are not special in any way. The only thing you’ll
notice is that we use the Doctrine Transport and store messages in the tl_message_queue
table. This table does not
have any DCA assigned as we’d need to stay up to date with the changes in Symfony. If they added another column for
example, it would fail. That’s why the table is dynamically added and configured in our
Contao\CoreBundle\EventListener\DoctrineSchemaListener
meaning that anytime you run contao:migrate
, any schema
changes will be detected and your database will get updated. Hence, we use auto_setup=false
.
Then, we define 3 default transports that represent priorities:
WebWorker
For the Contao Managed Edition, we cannot assume that every user is able to have a messenger:consume
worker
running all the time. It’s fair to assume that probably most of the Contao setups run on some shared hosting
provider without any access to any process manager like Supervisor
, systemd
, launchd
, runit
and Co.
So when you as an extension developer want to use the Symfony Messenger integration, we somehow have to make sure,
your messages aren’t lost and being worked on , even if the Contao user installs Contao somewhere where no
messenger:consume
worker is running.
This is exactly what the WebWorker
is all about. The WebWorker
uses the Symfony kernel.terminate
event to work
on the messages that are on the queue. This is how it works:
It will not do this for all transports but only for the ones configured. In the Contao Managed Edition, this is enabled by default for all our 3 default transports, like so:
contao:
messenger:
web_worker:
transports:
- contao_prio_high
- contao_prio_normal
- contao_prio_low
If you define an additional transport and want to make sure, the WebWorker
takes care of your transport as
well, make sure to add it to contao.messenger.web_worker.transports
.
It will try to be smart about working on those defined transports. If it detects that there is actually a real worker
working on one of those queues, it will not work on any message from that transport during the web request. For
this, the WebWorker
listens to the WorkerStartedEvent
as well as the WorkerRunningEvent
and remembers that the
given transport is running using a cache entry that is valid for a grace period of 10
minutes (configurable). On
kernel.terminate
, if that cache entry is still valid, it will conclude that a real worker is working on this
transport and thus do nothing. If, however, that cache entry does not exist (never did or the grace period has
expired) it will conclude that no worker is running, and thus it will start consuming messages from that
transport within the web process. This is basically as if you had the sync
transport configured with one additional advantage: As this happens in kernel.terminate
, depending on your PHP setup, it is deferred to after the response has been sent to the client
(fastcgi_finish_request()
). So it is a win for you in any case!
The WebWorker
always limits its inline messenger:consume
logic to a maximum of 30
seconds (not configurable)
in order to make sure the web process does not run forever.
The grace period to determine whether a worker is running or not is needed because if there’s only one worker
working on e.g. contao_prio_high
and the message the worker is currently working on takes - say - 15 minutes to
process, no WorkerRunningEvent
is going to be dispatched for contao_prio_high
which would in turn force the
WebWorker
to fall back to the kernel.terminate
logic. This might be the desired behavior after the default grace
period of 10
minutes but probably also not - this very much depends on the type of messages and the number of real
workers you have configured. You may adjust the grace period like so (use the \DateInterval
duration specification):
contao:
messenger:
web_worker:
grace_period: 'PT5M' # 5 minutes
Contao wouldn’t be Contao if it didn’t try to find an ingenious solution for the missing process manager on shared
hosting providers problem. Sure, most of them do not - and probably never will - provide an option for you to
register e.g. php bin/console messenger:consume contao_prio_high
.
But what most of them have is - you guessed it - cron jobs!
In the Contao Managed Edition - in case you configured the Contao Cron job Framework with a real, minutely
cronjob - Contao will automatically start asynchronous messenger:consume
commands which are configured to
stop after 60
seconds effectively resulting in having continuously running workers that are running for a minute.
Then the minutely cron job comes back around and our workes are started again - as if we had a real process manager
running! The workers even support simple autoscaling! Here’s the default configuration of the Contao Managed Edition:
contao:
messenger:
workers:
-
# Read: Start "messenger:consume contao_prio_high --time-limit=60 --sleep=5",
# try to achieve a low number of messages pending on the queue (5) and make
# sure, you never start more than 10 of these processes.
transports:
- contao_prio_high
options:
- --time-limit=60
- --sleep=5
autoscale:
desired_size: 5
max: 10
-
# Read: Start "messenger:consume contao_prio_normal --time-limit=60 --sleep=10",
# try to achieve a low number of messages pending on the queue (10) and make
# sure, you never start more than 10 of these processes.
transports:
- contao_prio_normal
options:
- --time-limit=60
- --sleep=10
autoscale:
desired_size: 10
max: 10
-
# Read: Start "messenger:consume contao_prio_low --time-limit=60 --sleep=20",
# try to achieve a normal number of messages pending on the queue (20) and make
# sure, you never start more than 10 of these processes.
transports:
- contao_prio_low
options:
- --time-limit=60
- --sleep=20
autoscale:
desired_size: 20
max: 10
In reality, things are a bit more complex than just starting the messenger:consume
commands every minute because as
messages could take longer than one minute to finish being processed, we also have to supervise how many processes are running
and not blindly start them as we might accumulate too many processes like that. But this would go beyond the scope
of the documentation. Just know that Contao has your back and takes care of that problem for you! 😎
You don’t need Supervisor
, systemd
or the likes when using the Contao Managed Edition! Just configure a real
minutely cron job triggering contao:cron
and you’re good to go!
So we know how the WebWorker
fallback works, and we have a solution for running the messenger:consume
commands. One
piece is missing, though: How does Contao know that your message (let’s assume a CreateAsyncZipFileMessage
in this example)
should be routed to the contao_prio_low
, contao_prio_normal
or contao_prio_high
transports? The routing part is missing!
So as an extension developer, you would need to specify the target like so:
framework:
messenger:
routing:
'App\Messenger\CreateAsyncZipFileMessage': contao_prio_high
This would be totally doable using a Contao Manager Plugin
and adjusting the Symfony Framework configuration,
appending your entry. However, because Contao ships with the 3 default priorities, there are also built-in
interfaces for those 3 which are then routed automatically:
framework:
messenger:
routing:
'Contao\CoreBundle\Messenger\Message\HighPriorityMessageInterface': contao_prio_high
'Contao\CoreBundle\Messenger\Message\NormalPriorityMessageInterface': contao_prio_normal
'Contao\CoreBundle\Messenger\Message\LowPriorityMessageInterface': contao_prio_low
Instead of fiddling with the container and the configuration, all you need to do is implement one of the priority interfaces and the routing is configured.
The entire setup presented above ensures that - as a developer - you can enjoy a zero-configuration asynchronous message processing setup, provided you have the Contao cronjob framework running. You only need your message and the respective message handler:
contao:cron
- aka configure the Contao cron job framework.namespace App\Messenger;
use Contao\CoreBundle\Messenger\Message\HighPriorityMessageInterface;
class CreateAsyncZipFileMessage implements HighPriorityMessageInterface
{
public function __construct(public array $fileIds)
{
}
}
namespace App\Messenger;
use Symfony\Component\Messenger\Attribute\AsMessageHandler;
#[AsMessageHandler]
class CreateAsyncZipFileMessageHandler
{
public function __invoke(CreateAsyncZipFileMessage $message): void
{
foreach ($message->fileIds as $fileId) {
// Create your zip file asynchronously which can take a long while now 🔥
}
}
}
For a working example, take a look at the SearchIndexMessage
, SearchIndexMessageHandler
and SearchIndexListener
classes to see how Contao uses the Messenger to create and update the search index outside the actual HTTP
request to serve responses to the users faster.
In case you want to work with a real process manager, there is no point in using built-in cron job workers. You can disable them by adjusting the configuration:
contao:
messenger:
workers: [] # No workers will disable the cron job worker feature
The WebWorker
will automatically detect that you have real workers running. This means you don’t have to adjust the
configuration. It will only kick in if your grace period is over because e.g. a worker takes too long to process a
message or something on your infrastructure failed and thus no workers are running. Having this fallback
solution to ensure your messages are always being worked on might be even a desirable state! If you want to absolutely
avoid having any messages being worked on in the web process, however, you can easily disable this behavior as well by
having the WebWorker
listen to no transport at all:
contao:
messenger:
web_worker:
transports: [] # No transports will disable the web worker feature
Because PHP (or your code) might leak memory, it’s usually a good idea to use any of the limit options (see Symfony
docs) and have the messenger:consume
process stop after some time or RAM usage to free those resources. Just have
your process manager respawn the process again.