Authentication, Authorisation, Access Control
Overview
This document describes authentication and authorisation features in RabbitMQ. Together they allow the operator to control access to the system.
Different users can be granted access only to specific virtual hosts. Their permissions in each virtual host also can be limited.
RabbitMQ supports two major authentication mechanisms as well as several authentication and authorisation backends.
This guide covers a variety of authentication, authorisation and user management topics such as
- Access control essentials
- Default virtual host and user
- Connectivity limitations imposed on the default user
- Authorisation and resource permissions
- How to manage users and permissions using CLI tools
- How to change an authentication or authorization backend used, or use a combination of backends
- How to authenticate clients using their TLS certificate information
- How to limit access to topics on a topic exchange
- User tags and how they are used
- How to rotate credentials and revoke access for a user
- Shell escaping of characters in generated passwords
- How to pre-create users and their permissions
- Troubleshooting of authentication and authorisation failures
Password-based authentication has a companion guide. Two closely related topics of OAuth 2 support and TLS support, including x.509-certificate based authentication, are covered in dedicated guides.
Terminology and Definitions
Authentication and authorisation are often confused or used interchangeably. That's wrong and in RabbitMQ, the two are separated. For the sake of simplicity, we'll define authentication as "identifying who the user is" and authorisation as "determining what the user is and isn't allowed to do."
The Basics
When clients connect to RabbitMQ, they specify a set of credentials: a username-password pair, a JWT token, or a x.509 certificate. Every connection has an associated user which is authenticated. It also targets a virtual host for which the user must have a certain set of permissions.
User credentials, target virtual host and (optionally) client certificate are specified at connection initiation time.
There is a default pair of credentials called the default user. This user can only be used for host-local connections by default. Remote connections that use the default user will be refused.
Production environments should not use the default user. Create new user accounts with generated credentials instead.
Default Virtual Host and User
When the server first starts running, and detects that its database is uninitialised or has been reset or deleted (the node is a "blank node"), it initialises a fresh database with the following resources:
- a virtual host named /(a slash)
- a user named guestwith a default password ofguest, granted full access to the/virtual host
If a blank node imports definitions on boot, this default user will not be created.
It is highly recommended to pre-configure a new user with a generated username and password or delete
the guest user or at least change its password
to reasonably secure generated value that won't be known to the public.
Authentication: Who Do You Say You Are?
After an application connects to RabbitMQ and before it can perform operations, it must authenticate, that is, present and prove its identity. With that identity, RabbitMQ nodes can look up its permissions and authorize access to resources such as virtual hosts, queues, exchanges, and so on.
Two primary ways of authenticating a client are username/password pairs and X.509 certificates. Username/password pairs can be used with a variety of authentication backends that verify the credentials.
Connections that fail to authenticate will be closed with an error message in the server log.
Authentication using Client TLS (x.509) Certificate Data
To authenticate client connections using X.509 certificate a built-in plugin, rabbitmq-auth-mechanism-ssl, must be enabled and clients must be configured to use the EXTERNAL mechanism. With this mechanism, any client-provided password will be ignored.
"guest" user can only connect from localhost
By default, the guest user is prohibited from
connecting from remote hosts; it can only connect over
a loopback interface (localhost). This
applies to connections regardless of the protocol.
Any other users will not (by default) be restricted in this way.
Remote connections that use the default user will be refused with a log message similar to this:
[error] <0.918.0> PLAIN login refused: user 'guest' can only connect via localhost
The recommended way to address this in production systems is to create a new user with generated credentials, or set of users, with the permissions to access the necessary virtual hosts. This can be done using CLI tools, HTTP API or definitions import.
Fedora Linux's RabbitMQ package is patched to allow for remote connections with well known credentials.
This practice is very strongly recommended against. Fedora Linux users should consider installing RabbitMQ from Team RabbitMQ's package repositories and avoid using the distribution-packaged version unless this serious distribution-specific security flaw is addressed.
This is configured via the loopback_users item
in the configuration file.
It is possible to allow the guest user to connect
from a remote host by setting the
loopback_users configuration to
none.
A minimalistic RabbitMQ config file
which allows remote connections for guest looks like so:
# DANGER ZONE!
#
# allowing remote connections for default user is highly discouraged
# as it dramatically decreases the security of the system. Delete the default user
# instead and create a new one with generated secure credentials, or use JWT tokens,
# or x.509 certificates for clients to authenticate themselves
loopback_users = none
Managing Users and Permissions
Users and permissions can be managed using CLI tools and definition import (covered below).
Before We Start: Shell Escaping and Generated Passwords
It is a common security practice to generate complex passwords, often involving non-alphanumeric characters. This practice is perfectly applicable to RabbitMQ users.
Shells (bash, zsh, and so on) interpret certain characters
(!, ?, &, ^, ", ', *, ~, and others) as control characters.
When a password is specified on the command line for rabbitmqctl add_user, rabbitmqctl change_password,
and other commands that accept a password, such control characters must be escaped appropriately
for the shell used.
With inappropriate escaping the command will fail or RabbitMQ CLI tools will receive a different value
from the shell.
When generating passwords that will be passed on the command line,
long (say, 40 to 100 characters) alphanumeric value with a very limited set of
symbols (e.g. :, =) is the safest option.
When users are created via HTTP API without using a shell (e.g. curl),
the control character limitation does not apply. However, different escaping rules may be necessary
depending on the programming language used.
Adding a User
To add a user, use rabbitmqctl add_user or rabbitmqadmin users declare. Both have multiple ways of specifying a password:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
- cmd
# will prompt for password, only use this option interactively
rabbitmqctl add_user "username"
# Password is provided via standard input.
# Note that certain characters such as !, &, $, #, and so on must be escaped to avoid
# special interpretation by the shell.
echo '2a55f70a841f18b97c3a7db939b7adc9e34a0f1b' | rabbitmqctl add_user 'username'
# Password is provided as a command line argument.
# Note that certain characters such as !, &, $, #, and so on must be escaped to avoid
# special interpretation by the shell.
rabbitmqctl add_user 'username' '2a55f70a841f18b97c3a7db939b7adc9e34a0f1b'
# This example uses a salted password hash instead of a plaintext value.
# To produce a salted hash of a password, use
# 'rabbitmqctl hash_password "my-secret-password"'
rabbitmqctl add_user --pre-hashed-password 'username' '{value produced by "rabbitmqctl hash_password"}'
# Password is provided as a command line argument.
# Note that certain characters such as !, &, $, #, and so on must be escaped to avoid
# special interpretation by the shell.
rabbitmqadmin users declare \
    --name 'username' \
    --password '2a55f70a841f18b97c3a7db939b7adc9e34a0f1b'
# This exampel uses a salted password hash instead of a plaintext value.
# To produce a salted hash of a password, use
# 'rabbitmqadmin passwords salt_and_hash "my-secret-password"'
rabbitmqadmin users declare \
    --name 'username' \
    --password-hash "{value producted by 'rabbitmqadmin passwords salt_and_hash'}"
# password is provided as a command line argument
rabbitmqctl.bat add_user 'username' '9a55f70a841f18b97c3a7db939b7adc9e34a0f1d'
# passwords with special characters must be quoted correctly
rabbitmqctl.bat add_user 'username' '"w63pnZ&LnYMO(t"'
# This example uses a salted password hash instead of a plaintext value.
# To produce a salted hash of a password, use
# 'rabbitmqctl.bat hash_password "my-secret-password"'
rabbitmqctl.bat add_user --pre-hashed-password 'username' '{value produced by "rabbitmqctl.bat hash_password"}'
# Password is provided as a command line argument.
# Passwords with special characters must be quoted correctly.
rabbitmqadmin.exe users declare ^
    --name "username" ^
    --password "9a55f70a841f18b97c3a7db939b7adc9e34a0f1d"
# This exampel uses a salted password hash instead of a plaintext value.
# To produce a salted hash of a password, use
# 'rabbitmqadmin.exe passwords salt_and_hash "my-secret-password"'
rabbitmqadmin.exe users declare ^
    --name "username" ^
    --password-hash "{value producted by 'rabbitmqadmin.exe passwords salt_and_hash'}"
rem password is provided as a command line argument
rabbitmqctl.bat add_user "username" "9a55f70a841f18b97c3a7db939b7adc9e34a0f1d"
rem passwords with special characters must be quoted correctly
rabbitmqctl.bat add_user "username" "w63pnZ&LnYMO(t"
A newly added user must be granted permissions for one or more virtual hosts, otherwise its connections will be refused.
Listing Users
To list users in a cluster, use rabbitmqctl list_users or rabbitmqadmin users list:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
rabbitmqctl list_users
rabbitmqadmin users list
rabbitmqctl.bat list_users
rabbitmqadmin.exe users list
Deleting a User
To delete a user, use rabbitmqctl delete_user or rabbitmqadmin users delete:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
rabbitmqctl delete_user 'username'
rabbitmqadmin users delete --name 'username'
rabbitmqctl.bat delete_user 'username'
rabbitmqadmin.exe users delete --name "username"
Granting Permissions to a User
To grant permissions to a user in a virtual host, use rabbitmqctl set_permissions or rabbitmqadmin declare permissions:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
# First ".*" for configure permission on every entity
# Second ".*" for write permission on every entity
# Third ".*" for read permission on every entity
rabbitmqctl set_permissions -p "custom-vhost" "username" ".*" ".*" ".*"
# Grant full permissions to a user in a virtual host
rabbitmqadmin declare permissions \
    --vhost "custom-vhost" \
    --user "username" \
    --configure ".*" \
    --write ".*" \
    --read ".*"
# First ".*" for configure permission on every entity
# Second ".*" for write permission on every entity
# Third ".*" for read permission on every entity
rabbitmqctl.bat set_permissions -p 'custom-vhost' 'username' '.*' '.*' '.*'
# Grant full permissions to a user in a virtual host
rabbitmqadmin.exe declare permissions ^
    --vhost "custom-vhost" ^
    --user "username" ^
    --configure ".*" ^
    --write ".*" ^
    --read ".*"
Clearing Permissions of a User in a Virtual Host
To revoke permissions from a user in a virtual host, use rabbitmqctl clear_permissions or rabbitmqadmin delete permissions:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
# Revokes permissions in a virtual host
rabbitmqctl clear_permissions -p "custom-vhost" "username"
# Revokes permissions from a user in a virtual host
rabbitmqadmin delete permissions \
    --vhost "custom-vhost" \
    --user "username"
# Revokes permissions in a virtual host
rabbitmqctl.bat clear_permissions -p 'custom-vhost' 'username'
# Revokes permissions from a user in a virtual host
rabbitmqadmin.exe delete permissions ^
    --vhost "custom-vhost" ^
    --user "username"
Pre-configuring Default User Permissions for New Virtual Hosts
Default user permissions can be pre-configured in rabbitmq.conf to automatically grant
permissions when new virtual hosts whose names match a pattern are created.
## Grants user 'monitoring' access to all virtual hosts
## with certain permissions
default_users.monitoring.vhost_pattern = .*
default_users.monitoring.tags = monitoring
default_users.monitoring.configure = ^$
default_users.monitoring.read = .*
default_users.monitoring.write = ^$
This feature should only be used to grant virtual host access to service accounts (such as monitoring and automation tools).
Operations on Multiple Virtual Hosts
Every rabbitmqctl and rabbitmqadmin permission management operation is scoped to a single virtual host.
Bulk operations have to be scripted, with the list of virtual hosts coming from rabbitmqctl list_vhosts --silent or rabbitmqadmin vhosts list:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
# Assumes a Linux shell.
# Grants a user permissions to all virtual hosts.
for v in $(rabbitmqctl list_vhosts --silent); do rabbitmqctl set_permissions -p $v "a-user" ".*" ".*" ".*"; done
# Assumes a Linux shell.
# Grants a user permissions to all virtual hosts using rabbitmqadmin.
# Note: this example assumes rabbitmqadmin vhosts list outputs one vhost name per line
for v in $(rabbitmqadmin vhosts list --quiet | awk '{print $1}' | tail -n +2); do
  rabbitmqadmin declare permissions --vhost "$v" --user "a-user" --configure ".*" --write ".*" --read ".*"
done
rabbitmqctl.bat list_vhosts --silent | %{ rabbitmqctl.bat set_permissions -p $_ 'a-user' '.*' '.*' '.*' }
# Grants a user permissions to all virtual hosts using rabbitmqadmin
# This approach uses rabbitmqctl for vhost listing as it's more straightforward for scripting
rabbitmqctl.bat list_vhosts --silent | %{ rabbitmqadmin.exe declare permissions --vhost $_ --user 'a-user' --configure '.*' --write '.*' --read '.*' }
Seeding (Pre-creating) Users and Permissions
Production environments typically need to pre-configure (seed) a number of virtual hosts, users and user permissions.
This can be done in a few ways:
- Using CLI tools
- Definition export and import on node boot (recommended)
- Override default credentials in configuration file(s)
CLI Tools
See the section on User Management.
Definition Import at Node Boot Time
This process involves the following steps:
- Set up a temporary node and create the necessary virtual host, users, permissions, and so on using CLI tools
- Export definitions to a definition file
- Remove parts of the file that are not relevant
- Configure the node to import the file on node boot or after
See importing definitions on node boot in the definitions guide to learn more.
Definition Import After Node Boot
See importing definitions after node boot in the definitions guide.
Override Default User Credentials
Two configuration options can be used to override default user credentials. This user will only be created on first node boot so they must exist in the configuration file before first boot.
The settings are:
# default is "guest", and its access is limited to localhost only.
# See ./access-control#default-state
default_user = a-user
# default is "guest"
default_pass = 768a852ed69ce916fa7faa278c962de3e4275e5f
As with all values in rabbitmq.conf, the # character
starts a comment so this character must be avoided in generated credentials.
Default user credentials can also be encrypted. This topic is covered in more detail in Configuration Value Encryption.
Authorisation: How Permissions Work
When a RabbitMQ client establishes a connection to a server and authenticates, it specifies a virtual host within which it intends to operate. A first level of access control is enforced at this point, with the server checking whether the user has any permissions to access the virtual hosts, and rejecting the connection attempt otherwise.
Resources, i.e. exchanges and queues, are named entities inside a particular virtual host; the same name denotes a different resource in each virtual host. A second level of access control is enforced when certain operations are performed on resources.
RabbitMQ distinguishes between configure,
write and read operations on a
resource. The configure operations create or
destroy resources, or alter their behaviour. The
write operations inject messages into a
resource. And the read operations retrieve messages
from a resource.
In order to perform an operation on a resource the user must have been granted the appropriate permissions for it.
- AMQP 0-9-1
- AMQP 1.0
- MQTT
The following table shows what permissions on what type of resource are required for all the AMQP 0-9-1 commands which perform permission checks.
| AMQP 0-9-1 Operation | configure | write | read | |
|---|---|---|---|---|
| exchange.declare | (passive=false) | exchange | ||
| exchange.declare | (passive=true) | |||
| exchange.declare | (with AE) | exchange | exchange (AE) | exchange | 
| exchange.delete | exchange | |||
| queue.declare | (passive=false) | queue | ||
| queue.declare | (passive=true) | |||
| queue.declare | (with DLX) | queue | exchange (DLX) | queue | 
| queue.delete | queue | |||
| exchange.bind | exchange (destination) | exchange (source) | ||
| exchange.unbind | exchange (destination) | exchange (source) | ||
| queue.bind | queue | exchange | ||
| queue.unbind | queue | exchange | ||
| basic.publish | exchange | |||
| basic.get | queue | |||
| basic.consume | queue | |||
| queue.purge | queue | 
The following table shows what permissions on what type of resource are required for all the AMQP 1.0 commands which perform permission checks.
| AMQP Operation | configure | write | read | |
|---|---|---|---|---|
| declare exchange | exchange | |||
| declare exchange | with AE | exchange | exchange (AE) | exchange | 
| delete exchange | exchange | |||
| bind exchange to exchange | exchange (destination) | exchange (source) | ||
| unbind exchange from exchange | exchange (destination) | exchange (source) | ||
| declare queue | queue | |||
| declare queue | with DLX | queue | exchange (DLX) | queue | 
| delete queue | queue | |||
| purge queue | queue | |||
| bind queue to exchange | queue | exchange | ||
| unbind queue from exchange | queue | exchange | ||
| get queue | ||||
| attach (sender) | exchange addresss | exchange | ||
| attach (sender) | queue address | exchange (amq.default) | ||
| attach (receiver) | queue | 
MQTT protocol does not have the concept of exchanges, queues and bindings. Our MQTT plugin builds on top of AMQP protocol and uses the same permission system. You can read more about it in the MQTT plugin guide.
The following table shows what permissions on what type of resource are required for all the MQTT commands which perform permission checks.
| MQTT Operation | configure | write | read | |
|---|---|---|---|---|
| SUBSCRIBE | queue | queue | exchange | |
| PUBLISH | amq.topic | |||
| PUBLISH | non-default exchange | exchange | 
PUBLISH operation also requires passing topic authorisation.
Permissions are expressed as a triple of regular expressions — one each for configure, write and read — on a per-vhost basis. The user is granted the respective permission for operations on all resources with names matching the regular expressions.
For example, the table above demonstrates that the queue.bind
protocol operation requires write permission on the target queue
and the read permission is required on the target exchange.
In other words, in order to allow a user to bind a queue named queueA to
an exchange named exchangeB the user will need the write
permission regex (for the correct virtual host) to match queueA,
and the read permission regex to match exchangeB.
For convenience RabbitMQ maps AMQP 0-9-1's default exchange's blank name to 'amq.default' when performing permission checks.
The regular expression '^$', i.e. matching
nothing but the empty string, covers all resources and
effectively stops the user from performing any operation.
Built-in AMQP 0-9-1 resource names are prefixed with
amq. and server generated names are prefixed
with amq.gen.
For example, '^(amq\.gen.*|amq\.default)$' gives a user access to
server-generated names and the default exchange.  The empty
string, '' is a synonym for '^$'
and restricts permissions in the exact same way.
RabbitMQ may cache the results of access control checks on a per-connection or per-channel basis. Hence changes to user permissions may only take effect when the user reconnects.
For details of how to set up access control, please see the User management section as well as the rabbitmqctl man page.
User Tags and Management UI Access
In addition to the permissions covered above, users can have tags associated with them. Currently only management UI access is controlled by user tags.
The tags are managed using rabbitmqctl. Newly created users do not have any tags set on them by default.
Please refer to the management plugin guide to learn more about what tags are supported and how they limit management UI access.
Topic Authorisation
RabbitMQ supports topic authorisation for topic exchanges. The routing key of a message published to a topic exchange is taken into account when publishing authorisation is enforced (e.g. in RabbitMQ default authorisation backend, the routing key is matched against a regular expression to decide whether the message can be routed downstream or not). Topic authorisation targets protocols like STOMP and MQTT, which are structured around topics and use topic exchanges under the hood.
Topic authorisation is an additional layer on top of
existing checks for publishers. Publishing a
message to a topic-typed exchange will go through both the
basic.publish and the routing key checks.
The latter is never called if the former refuses access.
Topic authorisation can also be enforced for topic consumers. Note that it works differently for different protocols. The concept of topic authorisation only really makes sense for the topic-oriented protocols such as MQTT and STOMP. In AMQP 0-9-1, for example, consumers consume from queues and thus the standard resource permissions apply. In addition for AMQP 0-9-1, binding routing keys between an AMQP 0-9-1 topic exchange and a queue/exchange are checked against the topic permissions configured, if any. For more information about how RabbitMQ handles authorisation for topics, please see the STOMP and MQTT documentation guides.
When default authorisation backend is used, publishing to a topic exchange or consuming from a topic is always authorised if no topic permissions are defined (which is the case on a fresh RabbitMQ installation). With this authorisation backend, topic authorisation is optional: you don't need to approve any exchanges. To use topic authorisation therefore you need to opt in and define topic permissions for one or more exchanges. For details please see the rabbitmqctl man page.
Internal (default) authorisation backend supports variable expansion
in permission patterns.
Three variables are supported: username, vhost,
and client_id. Note that client_id only
applies to MQTT. For example, if tonyg is the
connected user, the permission ^{username}-.* is expanded to
^tonyg-.*
If a different authorisation backend (e.g. LDAP, HTTP, OAuth 2) is used, please refer to the documentation of those backends.
If a custom authorisation backend is used, topic
authorisation is enforced by implementing the
check_topic_access callback of the
rabbit_authz_backend behavior.
Revoking User Access and Credential Rotation
Revoking User Access
To revoke user access, the recommended procedure is deleting the user. All open connections that belong to a deleted user will be closed.
It is also possible to clear user permissions but that will not affect any currently open connections. Connections use an authorization operation cache, so client operations will be refused eventually. The period of time depends on the authorization backend used.
Credential Rotation
A credential rotation for credentials of users stored in the internal data store usually involves the following steps:
- changing user password using CLI tools
or update it using the PUT /api/users/{user}HTTP API endpoint
- Using rabbitmqctl close_all_user_connectionsorrabbitmqctl close_all_connectionsto close all existing connections
- Making sure that applications can discover the new credentials and reconnect, for example, by restarting them
In case of a suspected credential leak, a more thorough credential rotation procedure can be employed:
- Assessing the permissions of the user and obtaining a complete cluster definitions file
- Delete the user, thus revoking its access and closing all existing connections that used those credentials
- Re-adding the user with a new generated password
- Re-granting the user access to all the virtual hosts it had access to earlier
- Making sure that applications can discover the new credentials and reconnect, for example, by restarting them
With external authN backends such as LDAP, user accounts are managed externally to RabbitMQ, therefore the credential rotation routine will also be external to RabbitMQ.
Authentication and Authorisation Backends
Authentication and authorisation are pluggable. Plugins can provide implementations of:
- authentication ("authn") backends: they determine client identity and decide whether the client should be allowed to connect
- authorisation ("authz") backends: they determine whether an identified (authenticated) client is authorized to perform a certain operation
It is possible and common for a plugin to provide both backends. RabbitMQ ships with the following built-in plugins which provide both authentication and authorisation backends:
The following built-in plugins provide authorisation backend implementations:
Some plugins such as Source IP range one also only provide an authorisation backend.
Authentication is supposed to be handled by the internal database, LDAP, etc.
A special cache backend can be used in combination with other backends to significantly reduce the load they generate on external services, such as LDAP or HTTP servers.
Combining Backends
It is possible to use multiple backends for
authn or authz using the
auth_backends configuration key. When
several authentication backends are used then the first
positive result returned by a backend in the chain is
considered to be final. This should not be confused with
mixed backends (for example, using LDAP for authentication and internal
backend for authorisation).
The following example configures RabbitMQ to use the internal backend only (and is the default):
# rabbitmq.conf
#
# 1 here is a backend name. It can be anything.
# Since we only really care about backend
# ordering, we use numbers throughout this guide.
#
# "internal" is an alias for rabbit_auth_backend_internal
auth_backends.1 = internal
The example above uses an alias, internal for rabbit_auth_backend_internal.
The following aliases are available:
- internalfor- rabbit_auth_backend_internal
- ldapfor- rabbit_auth_backend_ldap(from the LDAP plugin)
- oauthor- oauth2for- rabbit_auth_backend_oauth2(from the OAuth 2.0 plugin)
- httpfor- rabbit_auth_backend_http(from the HTTP auth backend plugin)
- dummyfor- rabbit_auth_backend_dummy
For plugins that do not have a shortcut, a full module (not the name of the plugin!) must be used:
# note that the module name begins with a "rabbit_", not "rabbitmq_", like plugin
# names usually do
auth_backends.1 = rabbit_auth_backend_ip_range
When using third party plugins, providing a full module name is necessary.
The following example configures RabbitMQ to use the LDAP backend for both authentication and authorisation. Internal database will not be consulted:
auth_backends.1 = ldap
This will check LDAP first, and then fall back to the internal database if the user cannot be authenticated through LDAP:
auth_backends.1 = ldap
auth_backends.2 = internal
Same as above but will fall back to the HTTP backend instead:
# rabbitmq.conf
#
auth_backends.1 = ldap
# uses module name instead of a short alias, "http"
auth_backends.2 = rabbit_auth_backend_http
# See HTTP backend docs for details
auth_http.user_path = http://my-authenticator-app/auth/user
auth_http.vhost_path = http://my-authenticator-app/auth/vhost
auth_http.resource_path = http://my-authenticator-app/auth/resource
auth_http.topic_path = http://my-authenticator-app/auth/topic
The following example configures RabbitMQ to use the internal database for authentication and the source IP range backend for authorisation:
# rabbitmq.conf
#
auth_backends.1.authn = internal
# uses module name because this backend is from a 3rd party
auth_backends.1.authz = rabbit_auth_backend_ip_range
The following example configures RabbitMQ to use the LDAP backend for authentication and the internal backend for authorisation:
# rabbitmq.conf
#
auth_backends.1.authn = ldap
auth_backends.1.authz = internal
The example below is fairly advanced. It will check LDAP first. If the user is found in LDAP then the password will be checked against LDAP and subsequent authorisation checks will be performed against the internal database (therefore users in LDAP must exist in the internal database as well, but do not need a password there). If the user is not found in LDAP then a second attempt is made using only the internal database:
# rabbitmq.conf
#
auth_backends.1.authn = ldap
auth_backends.1.authz = internal
auth_backends.2       = internal
Authentication Mechanisms
RabbitMQ supports multiple SASL authentication
mechanisms. There are four such mechanisms built into the
server: PLAIN, AMQPLAIN, ANONYMOUS,
and RABBIT-CR-DEMO, and one — EXTERNAL —
available as a plugin.
More authentication mechanisms can be provided by plugins. See the plugin development guide for more information on general plugin development.
Built-in Authentication Mechanisms
The built-in mechanisms are:
| Mechanism | Description | 
| PLAIN | SASL PLAIN authentication. This is enabled by default in the RabbitMQ server and clients, and is the default for most other clients. | 
| AMQPLAIN | Non-standard version of PLAIN retained for backwards compatibility. This is enabled by default in the RabbitMQ server. | 
| ANONYMOUS | This mechanism is enabled by default allowing anonymous clients to connect without providing
any credentials. RabbitMQ will internally authenticate and authorize the client using the credentials
configured in  | 
| EXTERNAL | Authentication happens using an out-of-band mechanism such as x509 certificate peer verification, client IP address range, or similar. Such mechanisms are usually provided by RabbitMQ plugins. | 
| RABBIT-CR-DEMO | Non-standard mechanism which demonstrates
challenge-response authentication. This mechanism has
security equivalent to PLAIN, and
is not enabled by default in the RabbitMQ server. | 
Mechanism Configuration in the Server
The auth_mechanisms configuration key determines which of the
installed mechanisms are offered to connecting clients.
This variable should be a list of accepted values corresponding to mechanism names, for example, the following list
auth_mechanisms.1 = PLAIN
auth_mechanisms.2 = AMQPLAIN
auth_mechanisms.3 = ANONYMOUS
is used by default.
The server mechanisms are ordered in decreasing level of preference. See the configuration file documentation.
Mechanism Configuration in the Client
Applications must opt-in to use a different authentication mechanism
such as EXTERNAL.
Mechanism Configuration in Java
The Java client does not use
the javax.security.sasl package by default
since this can be unpredictable on non-Oracle JDKs and is
missing entirely on Android. There is a RabbitMQ-specific
SASL implementation, configured by
the SaslConfig interface. A
class DefaultSaslConfig is provided to make
SASL configuration more convenient in the common case. A
class JDKSaslConfig is provided to act as a
bridge to javax.security.sasl.
ConnectionFactory.getSaslConfig()
and ConnectionFactory.setSaslConfig(SaslConfig)
are the primary methods for interacting with authentication mechanisms.
Mechanism Configuration in .NET
The .NET client provides its own SASL mechanism
implementations based on the AuthMechanism
and AuthMechanismFactory
interfaces. The ConnectionFactory.AuthMechanisms
property is a list of authentication mechanism factories in
preference order.
Mechanism Configuration in Erlang
The Erlang client provides its own SASL mechanism
implementations in the amqp_auth_mechanisms
module. The #amqp_params record can be
provided with a list of authentication functions in
preference order for network connections.
Troubleshooting Authentication
This section covers several very common problems related to authentication. For authorization (permission) errors, see another section below.
Inspecting server logs is crucially important when investigating authentication-related issues.
Incorrect Permissions
A failed authentication attempt, that is, when incorrect credentials were specified by the client, will result in a server log message that looks like this:
2019-03-25 12:28:19.047 [info] <0.1613.0> accepting AMQP connection <0.1613.0> (127.0.0.1:63839 -> 127.0.0.1:5672)
2019-03-25 12:28:19.056 [error] <0.1613.0> Error on AMQP connection <0.1613.0> (127.0.0.1:63839 -> 127.0.0.1:5672, state: starting):
PLAIN login refused: user 'user2' - invalid credentials
2019-03-25 12:28:22.057 [info] <0.1613.0> closing AMQP connection <0.1613.0> (127.0.0.1:63839 -> 127.0.0.1:5672)
Authentication failures on connections that authenticate using X.509 certificates will be logged differently. See TLS Troubleshooting guide for details.
rabbitmqctl authenticate_user can be used to test authentication for a username and password pair:
- rabbitmqctl with bash
- rabbitmqctl with PowerShell
rabbitmqctl authenticate_user "a-username" "a/password"
# note that double quotes are required due to the & character
rabbitmqctl.bat authenticate_user 'a-username' '"a/p&assword"'
If authentication succeeds, it will exit with the code of zero. In case of a failure, a non-zero exit code will be used and a failure error message will be printed.
rabbitmqctl authenticate_user will use a CLI-to-node communication connection to attempt to authenticate
the username/password pair against an internal API endpoint.
The connection is assumed to be trusted. If that's not the case, its traffic can be encrypted using TLS.
Default User's Connection From Remote Hosts are Refused
Default user cannot connect from remote hosts for a reason. Enabling such connections can dramatically decrease the security of the cluster. Consider creating a unique user with generated credentials for all production clusters.
If credentials are reported as correct, connections from localhost succeed but remote connections fail, the issue is default user connectivity limitations.
In this case, the error will look like this:
2024-08-24 17:28:32.153698-04:00 [error] <0.1567.0> PLAIN login refused: user 'guest' can only connect via localhost
A separate user with generated credentials must be created for clients (including shovels and federation links) connecting from remote hosts.
Authentication Failure Notifications in AMQP 0-9-1
Per AMQP 0-9-1 spec, authentication failures should result in the server closing TCP connection immediately. However, with RabbitMQ clients can opt in to receive a more specific notification using the authentication failure notification extension to AMQP 0-9-1. Modern client libraries support that extension transparently to the user: no configuration would be necessary and authentication failures will result in a visible returned error, exception or other way of communicating a problem used in a particular programming language or environment.
Troubleshooting Authorisation
Inspecting server logs is crucially important when investigating authorization-related issues.
Missing Permissions
The most common scenario with authorization failures is when a user was created but not granted permissions for the virtual host the client tries to connect to.
In this case, the connection will be refused with a message that looks like this:
2019-03-25 12:26:16.301 [info] <0.1594.0> accepting AMQP connection <0.1594.0> (127.0.0.1:63793 -> 127.0.0.1:5672)
2019-03-25 12:26:16.309 [error] <0.1594.0> Error on AMQP connection <0.1594.0> (127.0.0.1:63793 -> 127.0.0.1:5672, user: 'user2', state: opening):
access to vhost '/' refused for user 'user2'
2019-03-25 12:26:16.310 [info] <0.1594.0> closing AMQP connection <0.1594.0> (127.0.0.1:63793 -> 127.0.0.1:5672, vhost: 'none', user: 'user2')
Insufficient Permissions
Another very common scenario is that the permissions are defined but they are insufficient for the operations that the client tries to perform.
In this case the connection will be accepted
rabbitmqctl list_permissions and rabbitmqadmin users permissions can be used to inspect a user's
permission in a given virtual host:
- rabbitmqctl with bash
- rabbitmqadmin with bash
- rabbitmqctl with PowerShell
- rabbitmqadmin with PowerShell
rabbitmqctl list_permissions --vhost /
# => Listing permissions for vhost "/" ...
# => user	configure	write	read
# => user2	.*	.*	.*
# => guest	.*	.*	.*
# => temp-user	.*	.*	.*
rabbitmqctl list_permissions --vhost gw1
# => Listing permissions for vhost "gw1" ...
# => user	configure	write	read
# => guest	.*	.*	.*
# => user2	^user2	^user2	^user2
# List permissions for all users in the default virtual host
rabbitmqadmin users permissions --vhost /
# List permissions for all users in a specific virtual host
rabbitmqadmin users permissions --vhost gw1
rabbitmqctl.bat list_permissions --vhost /
rabbitmqctl.bat list_permissions --vhost gw1
# List permissions for all users in the default virtual host
rabbitmqadmin.exe users permissions --vhost "/"
# List permissions for all users in a specific virtual host
rabbitmqadmin.exe users permissions --vhost "gw1"
Authorisation failures (permission violations) are logged with the following messages:
2019-03-25 12:30:05.209 [error] <0.1627.0> Channel error on connection <0.1618.0> (127.0.0.1:63881 -> 127.0.0.1:5672, vhost: 'gw1', user: 'user2'), channel 1:
operation queue.declare caused a channel exception access_refused: access to queue 'user3.q1' in vhost 'gw1' refused for user 'user2'