Remote Access With Guacamole

Apache Guacamole is a fantastic software for those looking for an easy way to manage connections to a cluster of computers or those looking for a VDI that can run on a Chromebook. I use Apache Guacamole to manage access to my small cluster of Linux computers. In this guide, we'll explore setting up an Apache Guacamole server with Docker Compose, connecting computers, and adding an extension to Guacamole for improved logging with session recording.
Installing Docker Compose
Let's start by updating our system. To do this, run the following command:
sudo apt update && sudo apt upgrade -y
Now, let's remove old Docker packages:
for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg -y; done
Then, we can add Docker's apt repository to our sources:
sudo apt-get update
sudo apt-get install ca-certificates curl -y
sudo install -m 0755 -d /etc/apt/keyrings
sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc
sudo chmod a+r /etc/apt/keyrings/docker.asc
echo \
"deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \
$(. /etc/os-release && echo "$VERSION_CODENAME") stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
sudo apt-get update
Finally, let's install Docker's packages:
sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin -y
Once apt
has installed Docker, we'll need to start the Docker Engine. To do this, run the following command:
sudo systemctl start docker
Pulling Guacamole's Containers
With Docker now set up, we can pull Guacamole's containers from Docker Hub. This guide is not applicable for computers using ARM chips, as Apache Guacamole only provides AMD64
containers. As of this guide's writing, the latest tag is 1.5.5
, and I will be using MariaDB because it is easy to implement.
docker pull guacamole/guacamole:1.5.5
docker pull guacamole/guacd:1.5.5
docker pull mariadb
Initialize Database
For Guacamole to work correctly, we'll need to initialize our database. Luckily, Guacamole provides these commands in the file /opt/guacamole/bin/initdb.sh --mysql > initdb.sql
.
docker run --rm guacamole/guacamole:1.5.5 /opt/guacamole/bin/initdb.sh --mysql > initdb.sql
We'll now begin writing our docker-compose.yml
file. We'll start by spinning up our database.
services:
guacdb:
container_name: guacamole_db
image: mariadb
restart: unless-stopped
environment:
MYSQL_ROOT_PASSWORD: '<Please Pick A Good Password>'
MYSQL_DATABASE: 'guacamole_db'
MYSQL_USER: 'guacamole_user'
MYSQL_PASSWORD: '<Please Pick A Good Password>'
volumes:
- './db-data:/var/lib/mysql'
volumes:
db-data:
Save & start the container:
docker compose up -d
Now copy the SQL file from the machine to your container:
docker cp initdb.sql guacamole_db:/initdb.sql
Open the container's shell:
docker exec -it guacamole_db bash
Load the file into the database:
cat /initdb.sql | mariadb -u root -p guacamole_db
Exit the container once you're done with the exit
command, and turn it off by running :
docker compose down
Starting The Server
Open the previous docker-compose.yaml
file and replace its contents with the following:
services:
guacdb:
container_name: guacamole_db
image: mariadb
restart: unless-stopped
environment:
MYSQL_ROOT_PASSWORD: '<Please Pick A Good Password>'
MYSQL_DATABASE: 'guacamole_db'
MYSQL_USER: 'guacamole_user'
MYSQL_PASSWORD: '<Please Pick A Good Password>'
volumes:
- './db-data:/var/lib/mysql'
guacd:
container_name: guacd
image: guacamole/guacd:1.5.5
restart: unless-stopped
volumes:
- ./record:/var/lib/guacamole/recordings:rw
guacamole:
container_name: guacamole
image: guacamole/guacamole:1.5.5
restart: unless-stopped
ports:
- 80:8080
environment:
GUACD_HOSTNAME: "guacd"
MYSQL_HOSTNAME: "guacdb"
MYSQL_DATABASE: "guacamole_db"
MYSQL_USER: "guacamole_user"
MYSQL_PASSWORD: "<Please Pick A Good Password>"
TOTP_ENABLED: "true"
WEBAPP_CONTEXT: "ROOT"
GUACAMOLE_HOME: '/opt/guacamole_home'
RECORDING_SEARCH_PATH: '/var/lib/guacamole/recordings'
depends_on:
- guacdb
- guacd
volumes:
- ./guacamole_home:/opt/guacamole_home
- ./record:/var/lib/guacamole/recordings:ro
volumes:
db-data:
Save & start the containers:
docker compose up -d
Accessing Guacamole
To access your server, open your browser and enter the IP of your Guacamole server. If you don't know your server's IP, enter the following into your terminal:
ip addr
Once you navigate to your server's IP, you'll be greeted with the following screen:

The original username and password is guacadmin
and guacadmin
respectively. Since we added TOTP_ENABLED: "true"
, we'll need to scan the QR code displayed with an authenticator app of your choice. To make authentication only username & password, remove the line TOTP_ENABLED: "true"
.

Create a New Connection
Once inside, navigate to Settings
:

Then open the tab Connections
and click + New Connection
.

On this page, name your connection and set the protocol to RDP
. Side note: to install RDP
on Ubuntu run sudo apt install xrdp
.

Then edit the network information so Guacamole can connect to your server

And finally, set the screen recording path to ${HISTORY_PATH}/${HISTORY_UUID}
and enable Automatically create recording path
:

Then click Save
and navigate to the Home
page.

Then click on the connection you added:

Voila, an RDP connection in your browser:

Integrated Session Recording

When navigating to the server logs, we see nothing in the Logs
column. Let's fix that! Start by navigating to the release page of the current Guacamole release, for us it's: 1.5.5
. In the future, it will be different.
Then, find guacamole-history-recording-storage-#.#.#.tar.gz
and copy its link. Replace the link in the following command with the link that matches your Guacamole version. If you're using: 1.5.5
you can copy and paste the commands below without any problems.
wget "https://apache.org/dyn/closer.lua/guacamole/1.5.5/binary/guacamole-history-recording-storage-1.5.5.tar.gz?action=download" -O guacamole-history-recording-storage-1.5.5.tar.gz -q
Once downloaded, extract the file guacamole-history-recording-storage-1.5.5.jar
to guacamole_home/extensions
:
mkdir -p guacamole_home/extensions && tar -xvf guacamole-history-recording-storage-1.5.5.tar.gz -C guacamole_home/extensions && rm guacamole-history-recording-storage-1.5.5.tar.gz && mv guacamole_home/extensions/guacamole-history-recording-storage-1.5.5/guacamole-history-recording-storage-1.5.5.jar guacamole_home/extensions/guacamole-history-recording-storage-1.5.5.jar && rm -r guacamole_home/extensions/guacamole-history-recording-storage-1.5.5/
We'll need to get the Guacamole user ID to assign read and write permissions to the record directory. To do this, run the following:
ID1=$(docker exec guacd id -u)
ID2=$(docker exec guacamole id -g)
sudo chown -R "${ID1}:${ID2}" ./record
sudo chown -R "${ID1}:${ID2}" ./guacamole_home
Then restart the server:
docker compose down && docker compose up -d
Sign in to Guacamole and open the connection we made previously.

After waiting a few seconds, exit the connection and navigate to the logs section in Settings
. You'll notice that there is now a View
link in the logs section. Clicking it will open the recording in your browser.

Conclusion
You’ve successfully created an Apache Guacamole web server, configured an RDP
connection, and installed the connection history extension. With this configuration, Guacamole effectively facilitates managing and sharing access to computer clusters while centralizing access and simplifying usage tracking.
If you enjoyed this article, you're likely interested in system security. In that case, my guide on user auditing and account policies is a must-read. Linux is a cornerstone of any IT career, and mastering these systems is an essential skill that can kickstart your journey in cybersecurity and IT. Believe me, you won't want to overlook this guide, especially if you participate in CyberPatriot or IT. Just click here to read it now, and I’ll see you there shortly. Cheers!