Citadel Realms

Citadel contains only the base operating system and the Gnome desktop, it does not include any applications. To be able to install and run applications Citadel can create spaces which are called Realms.

A Realm is a container similar to a Docker or LXC container in which any Linux distribution could be installed. We use a Debian based image but it would not be difficult to create an image for another Linux distribution.

The realm containers are launched with systemd-nspawn but this is a detail of how they are implemented and not something it is necessary to learn about in order to use them.

Citadel provides a command-line tool realms for creating, managing, and launching Realm instances.

The default realm

One realm is always selected to be the default realm. This realm starts automatically when the system boots. The realms utility can be used to change which realm is the default realm. Switching the default realm changes the symlink /realm/default.realm to point to a different realm instance directory.

citadel:~# realms default
Default Realm: main

citadel:~# realms default project
[+] default realm changed from 'main' to 'project'

citadel:~# realms default
Default Realm: project

The current realm

Multiple realms may be launched at once but the Gnome Desktop is only associated with one of the running realms. This realm is called the current realm.

When displaying applications available to launch from the desktop, Gnome will only be aware of applications that are installed in the realm which is set as current and any application launched from the desktop will run inside this current realm.

Setting another realm as current does not affect any applications that are already running. Changing the current realm only means that any further applications which are launched will now run in the newly chosen realm.

Changing or querying the current realm is done with the realms current command, and if you choose a realm which is not currently running it will be automatically started.

citadel:~# realms current
Current Realm: main

citadel:~ # realms current project
[+]: Started realm 'project'
[+]: Realm 'project' set as current realm

citadel:~ # realms current
Current Realm: project

Underneath the hood, this command just changes the symlink /run/realms/current.realm to point to a new realm. This directory is monitored for changes with inotify and when the symlink changes a new set of .desktop files is swapped into a temporary directory where Gnome will look for metadata about which applications are installed.

Creating a new realm

New realms are created with the command realms new <realm name>

When a new realm is created a btrfs snapshot of some application image is created at /realms/realm-$name/rootfs. By default it is the base image (base.appimg) which is cloned as a snapshot. Application images are described in detail in a later section.

citadel:~ # realms new project
[+]: Populating realm home directory with files from /realms/skel
Create a snapshot of '/storage/appimg/base.appimg' in '/realms/realm-project/rootfs'

A new empty home directory is also created for the realm instance. Any file which are placed into the /realm/skel directory will be copied into any newly created realm home directory.

Realms configuration file

All of the curretly supported configuration options are listed below with their default values assigned.

use-shared-dir = true
use-sound = true
use-x11 = true
use-wayland = true
use-gpu = false
use-kvm = false
use-network = true
network-zone = "clear"

If you wish to change any of these options to something other than what is listed above add the corresponding line to the file /realms/realm-$name/config

citadel:~ # echo "use-gpu = true" > /realms/realm-main/config

Option use-shared-dir

Set to false to disable mounting the shared directory /realms/Shared into this realm at /home/user/Shared.

Option use-sound

Set to false to prevent mounting pulse audio socket and sound device into this realm.

Option use-x11

Set to false to prevent mounting /tmp/.X11-unix into the realm. This is the socket for communicating with the XWayland X11 compatibility daemon.

Option use-wayland

Set to false to prevent mounting the wayland display server socket /run/user/1000/wayland-0 into the realm.

Option use-gpu

Set to true to mount the device /dev/dri/renderD128 into the realm. Adding this device will make hardware graphics acceleration available to applications running in the realm.

Option use-kvm

Set to true to mount the device /dev/kvm into the realm. This will make it possible to run Qemu and other KVM based tools with hardware virtualization inside the realm.

Option use-network

Set to false to disable configuring the realm with access to the internet. The realm instance will only have a localhost network interface.

Option network-zone

Setting a name here will create bridge device in citadel with the name vz-$name if it doesn't already exist and attach this realm instance to that bridge.

Realms base directory layout

The realms base directory is stored on the storage partition at /storage/realms and is bind mounted to /realms on the root filesystem for convenience.

/realms
    config
    /Shared
    /skel
    /default.realm -> realm-main
    /realm-main
    /realm-project
    /realm-testing

File /realms/config

This file is a template of the configuration file for individual realms. When a new realm is created this file in copied into the new realm instance directory. By modifying this file, the default configuration for new realm instances can be changed.

Directory /realms/Shared

This directory is bind mounted to /home/user/Shared of each running realm that has the option use-shared-dir enabled. It's a convenient way to move files between different realms and between citadel and realms.

Directory /realms/skel

Files which are added to this directory will be copied into the home directory of any newly created realm. The directory is copied as a tree of files and may contain subdirectories.

A symlink which points to a realm instance directory of the default realm. The default realm is the realm which starts when the system is booted.

Directory /realms/realm-$name

This is a realm instance directory, for a realm with $name as the realm name.

/realm-main
    config
    /home
    /rootfs
config

Configuration file for the realm instance copied from /realms/config or created by the user.

/home

Home directory for this realm. It will be mounted to /home/user in the realm instance.

/rootfs

The root filesystem of this realm. It is cloned from (a btrfs subvolume snapshot of) some application image.

Application Images

(Not to be confused with the AppImage packaging system)

The root filesystem for realms are called Application Images but we often use the shorter name appimg.

We have created a framework for building a Debian based images and we use this to build the default appimg that we ship.

We also encourage users to experiment with building their own custom images.

Tree Application Images are the only type of application image which are currently implemented for realms.

The rootfs is a tree of files on the filesystem, and it is also a btrfs subvolume which is cloned at zero cost (internally with btrfs subvolume snapshot) to use as the root filesystem of newly created realms.

Block Application Images (and also Sealed Application Images)

In the future we will add another type of application image called a Block Application Image. This type of image will be stored as a disk volume image file and will be mounted with a loop device rather than existing as a tree of files on the filesystem.

This will make it possible to enforce dm-verity verification over the image and ensure that no malicous or unintended modifications can be made to any of the the files on the root filesystem. Signature verification over the dm-verity root hash is done from the citadel rootfs image which is also secured with dm-verity. When enforcement of boot integrity is also implemented this will create a chain of cryptographic assurances that no component of the system has been tampered with.

Block images with signatures and dm-verify verification enabled are called Sealed Application Images

Updating an Application Image

To modify or update an application image run the realms update-appimg command. A container will be created for updating the image and a root shell session will open. From this session regular package management commands can be run. Any changes made will only affect future realms created from this appimg.

citadel:~ # realms update-appimg
[+]: Entering root shell on base appimg
root@base-appimg-update:/# apt update

[...]