---
title: "ejabberd"
date: 2022-03-29
icon: 'ejabberd.png'
tags: ['service']
short_desc: "A chat server based on XMPP."
---
[Ejabberd](https://ejabberd.im) is a server for the XMPP protocol written in Erlang. It's more scalable, and easier to setup than [Prosody](/prosody) due to having most of its modules built-in and pre-configured by default.
## Prerequisites
### Subdomains
Ejabberd presumes that you have already created all the **required and optional subdomains** for its operation prior to running it.
Depending on the usecase, you may need any or all of the following domains for XMPP functionality:
- **example.org** - Your XMPP hostname
- **conference.example.org** - For `mod_muc`, Multi User Chats (MUCs)
- **upload.example.org** - For `mod_http_upload`, file upload support
- **proxy.example.org** - For `mod_proxy65`, SOCKS5 proxy support
- **pubsub.example.org** - For `mod_pubsub`, publish-subscribe support (A fancier RSS)
Only the **example.org** domain is required for basic, private chat usage.
If you do **not** wish to use a certain domain, just disable it's associated module and ejabberd won't complain when it can't find it's associated certificate.
For example, if you don't want [Publish-Subscribe](https://xmpp.org/extensions/xep-0060.html) support, just comment out the `mod_pubsub` config in `/etc/ejabberd.yml`:
```yml
## mod_pubsub:
## access_createnode: pubsub_createnode
## plugins:
## - flat
## - pep
## force_node_config:
## ## Avoid buggy clients to make their bookmarks public
## storage:bookmarks:
## access_model: whitelist
```
This guide will assume **all these subdomains** have been created.
#### Custom Subdomains
If you wish to customize any of these domains, edit `/etc/ejabberd.yml` and under every appropriate module that needs a subdomain, add the following setting:
```yml
mod_muc:
host: {{}}muc.example.org{{}}
```
## Installation
To get the latest version of ejabberd, you need to first setup the ejabberd apt repositories:
```sh
curl -o /etc/apt/sources.list.d/ejabberd.list https://repo.process-one.net/ejabberd.list
curl -o /etc/apt/trusted.gpg.d/ejabberd.gpg https://repo.process-one.net/ejabberd.gpg
```
Then update the repositories and install the `ejabberd` package:
```sh
apt update
apt install ejabberd
```
## Configuration
The ejabberd server is configured in `/etc/ejabberd/ejabberd.yml`.
Changes are only applied by restarting the ejabberd daemon in systemd:
```sh
systemctl restart ejabberd
```
### Hostnames
The **XMPP hostname** is specified in the `hosts` section of
`ejabberd.yml`:
```yml
hosts:
- {{}}example.org{{}}
```
### Certificates
Unlike [Prosody,](https://prosody.im) ejabberd doesn\'t come equipped
with a script that can automatically copy over the relevant certificates
to a directory where the ejabberd user can read them.
One way of organizing certificates for ejabberd is to have them stored
in `/etc/ejabberd/certs`, with each domain having a separate directory
for both the fullchain cert and private key.
Using certbot, this process can be easily automated with these commands:
```bash
DOMAIN={{}}example.org{{}}
# Set the domain names you want here
declare -a subdomains=("" "conference." "proxy." "pubsub." "upload.")
for i in "${subdomains[@]}"; do
certbot --nginx -d $i$DOMAIN certonly
mkdir -p /etc/ejabberd/certs/$i$DOMAIN
cp /etc/letsencrypt/live/$i$DOMAIN/fullchain.pem /etc/ejabberd/certs/$i$DOMAIN
cp /etc/letsencrypt/live/$i$DOMAIN/privkey.pem /etc/ejabberd/certs/$i$DOMAIN
done
```
*Note: Just like with Prosody, you might want to write this script to a file and setup a [cronjob](/cron) to run it periodically. This should help prevent your certificates from expiring.*
Make sure all the certificates are readable by the `ejabberd` user:
```sh
chown -R ejabberd:ejabberd /etc/ejabberd/certs
```
To enable the use of all these certificates in ejabberd, the following
configuration is necessary:
```yml
certfiles:
- "/etc/ejabberd/certs/*/*.pem"
```
### Admin User
The **admin user** can be specified in `ejabberd.yml` under the `acl`
section:
```yml
acl:
admin:
user: admin
```
This would make **admin@example.org** the user with administrator
privileges.
### File Uploads
To ensure full compliance with XMPP standards, add the following configuration to `mod_http_upload`:
```yaml
mod_http_upload:
put_url: https://@HOST@:5443/upload
docroot: {{}}/var/www/upload{{}}
custom_headers:
"Access-Control-Allow-Origin": "https://@HOST@"
"Access-Control-Allow-Methods": "GET,HEAD,PUT,OPTIONS"
"Access-Control-Allow-Headers": "Content-Type"
```
Make sure to create and give the `ejabberd` user ownership of `/var/www/upload` or any other directory you choose to use for file uploads:
```sh
chown -R ejabberd:ejabberd /var/www/upload
```
### Message Archives
The ejabberd server supports keeping archives of messages through its `mod_mam` module. This can be enabled by uncommenting the following lines:
```yml
mod_mam:
assume_mam_usage: true
default: always
```
## Database
### Why use a database?
We can find the following comment in the `mod_mam` section of `/etc/ejabberd.yml`:
```yml
mod_mam:
## Mnesia is limited to 2GB, better to use an SQL backend
## For small servers SQLite is a good fit and is very easy
## to configure. Uncomment this when you have SQL configured:
## db_type: sql
```
As these comments imply, an **SQL backend** is strongly recommended if you wish to use your ejabberd server for anything more than just testing. Ejabberd supports **MySQL, SQLite** and **PostgreSQL.** For the purpose of efficiency, this guide will use **PostgresSQL** because other server software like [Matrix](/matrix) and [PeerTube](/peertube) support it.
### Installing PostgreSQL
PostgreSQL is available in the Debian repositories:
```sh
apt install postgresql
```
In addition, you will have to install the **appropriate headers for Erlang,** the language ejabberd is written in, so it can actually interact with the PostgreSQL server:
```sh
apt install erlang-p1-pgsql
```
Start the PostgreSQL daemon to begin using it:
```sh
systemctl start postgresql
```
### Creating the Database
To create the database, first create a PostgreSQL user for ejabberd:
```sh
su -c "createuser --pwprompt ejabberd" postgres
```
Then, create the database and make `ejabberd` its owner:
```sh
su -c "psql -c 'CREATE DATABASE ejabberd OWNER ejabberd;'" postgres
```
### Importing Database Scheme
Ejabberd does **not** create the database scheme by default; It has to be imported into the database before use.
```sh
su -c "curl -s https://raw.githubusercontent.com/processone/ejabberd/master/sql/pg.sql | psql ejabberd" postgres
```
### Configuring ejabberd to use PostgreSQL
Finally, add the following configuration to `ejabberd.yml`:
```yml
sql_type: pgsql
sql_server: "localhost"
sql_database: "{{}}ejabberd{{}}"
sql_username: "{{}}ejabberd{{}}"
sql_password: "{{}}psql_password{{}}"
default_db: sql
```
That line at the end sets **every module's database** to default to the `sql` backend. This includes the `mod_mam` module, so all our data is being stored with PostgreSQL.
### Voice and Video Calls
Ejabberd supports the **TURN** and **STUN** protocols to allow internet users behind NATs to perform voice and video calls with other XMPP users. **This is enabled by default using [ejabberd_stun](https://docs.ejabberd.im/admin/configuration/listen#ejabberd-stun-1).**
**However,** if you plan on running ejabberd alongside **other applications** that require TURN and STUN, such as Matrix, then you'll have to setup your own external TURN server using Coturn.
#### Setup with Coturn and `mod_stun_disco`
Firstly, setup a TURN and STUN server with [Coturn,](/coturn) using an **authentication secret.**
Then, edit `mod_stun_disco` to contain the appropriate information for
your turnserver:
```yml
mod_stun_disco:
secret: "{{}}your_auth_secret{{}}"
services:
-
host: {{}}turn.example.org{{}}
type: stun
-
host: {{}}turn.example.org{{}}
type: turn
```
## Using ejabberd
### Registering the Admin User
To begin using ejabberd, firstly start the ejabberd daemon:
```sh
systemctl restart ejabberd
```
Then, using `ejabberdctl` as the ejabberd user, register the admin user which is set in `ejabberd.yml`:
```sh
su -c "ejabberdctl register {{}}admin example.org password{{}}" ejabberd
```
This will create the user **admin@example.org.**
### Using the Web Interface
By default, ejabberd has a web interface accessible from **http://example.org:5280/admin**. When accessing this interface, you will be prompted for the admin credentials:
{{< img src="/pix/ejabberd-login.webp" >}}
After signing in with the admin credentials, you will be able to manage
your ejabberd server from this web interface:
{{< img src="/pix/ejabberd-admin.webp" >}}
## Further Configuration
For a deeper look into all the modules and options, have a look at the following ejabberd documentation:
- Ejabberd's [Listen Modules](https://docs.ejabberd.im/admin/configuration/listen/) and [Listen Options](https://docs.ejabberd.im/admin/configuration/listen-options/)
- Ejabberd's [Top-Level Options](https://docs.ejabberd.im/admin/configuration/toplevel/)
- Ejabberd's [Modules' Options](https://docs.ejabberd.im/admin/configuration/modules/)
*And with that, you've successfully setup your ejabberd XMPP server!*
---
Written by [Denshi.](https://denshi.org)
Donate Monero at:
`48dnPpGgo8WernVJp5VhvhaX3u9e46NujdYA44u8zuMdETNC5jXiA9S7JoYMM6qRt1ZcKpt1J3RZ3JPuMyXetmbHH7Mnc9C`