Appearance
This guide outlines Mercure Hub setup, and installation and configuration of the Mercure service and channel.
Installation
Composer dependencies
From a project with a pre-installed version of Notifier and Notifier Chat Channel, bring in the Mercure notifier and Mercure Drupal project dependencies:
sh
composer require drupal/mercure symfony/mercure-notifier
drush module:enable mercure
Mercure Hub
Set up Mercure Hub by following the guide at https://mercure.rocks/docs/hub/install. It's likely you'll have radically different setups for local vs production. The following describes a Docker Compose setup which contains both Drupal and web server, alongside Mercure Hub.
The template here is very similar to the template provided in Mercure documentation, based on the Symfony template with minor deviations:
yaml
services:
mercure:
image: dunglas/mercure
restart: unless-stopped
ports:
- "8099:80"
environment:
# Uncomment the following line to disable HTTPS,
SERVER_NAME: ':80'
MERCURE_PUBLISHER_JWT_KEY: '!ChangeThisMercureHubJWTSecretKey!'
MERCURE_SUBSCRIBER_JWT_KEY: '!ChangeThisMercureHubJWTSecretKey!'
# Set the URL of your Drupal project (without trailing slash!) as value of the cors_origins directive
MERCURE_EXTRA_DIRECTIVES: |
cors_origins http://localhost:8080
# Comment the following line to disable the development mode
command: /usr/bin/caddy run --config /etc/caddy/dev.Caddyfile
healthcheck:
test: [ "CMD", "curl", "-f", "https://localhost/healthz" ]
timeout: 5s
retries: 5
start_period: 60s
volumes:
- mercure_data:/data
- mercure_config:/config
volumes:
mercure_data:
mercure_config:
Replace port 8080
with the port you access Drupal/HTTP server from.
Configuration
In the site directory, usually sites/default/
, create a services.yml
file.
Add to settings.php
:
php
$settings['container_yamls'][] = __DIR__ . '/services.yml';
Add to services.yml
:
yaml
parameters:
chatter.transports:
my_mercure_transport: 'mercure://HUB_ID?topic=TOPIC'
mercure:
hubs:
default:
url: 'http://mercure/.well-known/mercure'
public_url: 'https://localhost:8099/.well-known/mercure'
jwt:
secret: '!ChangeThisMercureHubJWTSecretKey!'
publish: '*'
Chatter transports
Modify the DSN by replacing the HUB_ID
and TOPIC
parts.
HUB_ID
is the Mercure hub ID as configured in the Mercure module. If Mercure has not been customized heavily, this ID is likely default
. TOPIC
is the default topic, and may be overridden on the Notification
level by passing a MercureOptions
option object with topics. The default TOPIC
also accepts multiple topics using square-bracket ([]
) notation.
Mercure
Replace port 8099
with the port you can access the Mercure Hub from a host machine.
Replace mercure
in url
with the hostname your Drupal install can directly access the Mercure Hub in Docker networking.
WARNING
The provided configuration is for local testing only. Secrets must be updated appropriately when publicly accessible. See Mercure documentation.
Then clear Drupal caches:
sh
drush cr
Sending
To send a message to a Mercure Hub, create a notification then send to the notifier service.
Pass the name of the Mercure transport as configured in chatter.transports
to the Notification::channels()
method, prefixed with chat/
.
Message body
Chat notifications use subject()
for message contents, not contents()
.
php
namespace Drupal\my_module;
final class MyService {
public function __construct(
private readonly \Symfony\Component\Notifier\NotifierInterface $notifier,
) {}
public function myMethod(){
$notification = (new \Symfony\Component\Notifier\Notification\Notification())
->subject('Foo bar!')
# Specify the name of the Mercure transport as configured in `chatter.transports`.
->channels(['chat/my_mercure_transport']),
$this->notifier->send($notification);
}
}
yaml
services:
_defaults:
autowire: true
my_service:
class: Drupal\my_module\MyService
The above message dispatches this object to a browser listening for events:
json
{
"@context": "https://www.w3.org/ns/activitystreams",
"type": "Announce",
"summary": "Foo bar!"
}
INFO
As of writing the Mercure integration is only capable of sending ActivityStream Announce activities, not arbitrary payloads. To send any payload, either customize MercureTransport
, create a fully custom Mercure transport, or send data directly to Mercure Hub services.
Listening
To listen to events from a Mercure Hub, create HTML and Javascript including the minimum code:
html
<html>
<body>
<script>
const eventSource = new EventSource('http://localhost:8099/.well-known/mercure?topic=TOPIC');
eventSource.onmessage = event => {
console.log(JSON.parse(event.data));
}
window.addEventListener('beforeunload', function () {
eventSource.close();
})
</script>
</body>
</html>
No external libraries are necessary. EventSource is supported by all major web browsers.
Replace port 8099
with the port you can access the Mercure Hub from a host machine.
Replace TOPIC
with the topic the notification was dispatched to, or the default TOPIC in the mercure DSN.
Debug UI
Mercure Hub provides an interface to test and debug messages at http://localhost:8099/.well-known/mercure/ui/
.
Replace port 8099
with the port you can access the Mercure Hub from a host machine.
Resources
For more information about the Mercure integration, go to Mercure transport in the Symfony monorepo.
This documentation is not provided or endorsed by the maintainers of Mercure.