ZFS
ZFS or OpenZFS is an open-source storage platform. It includes the functionality of both traditional file systems and volume manager.
This page has instructions for creating and auto-mounting an encrypted ZFS drive or partition on an existing encrypted alpine linux system, using ZFS's own encryption capabilities. To do fresh install of Alpine Linux with root partition on ZFS, see Root on ZFS with native encryption.
Installation
Install the necessary packages and utilities by using the command:
# apk add zfs zfs-lts
Ensure that the kernel modules are loaded and verify that the device nodes are present:
# modprobe zfs # mdev -s
Configuration
The system will be encrypted when powered off but will not require you to type an extra password at boot, since it uses a key stored on the encrypted root partition. Alternative options are also given, such as prompting for a password at boot rather than storing the key on the root drive. The example in this guide is modeled around creating a ZFS filesystem to be used as a user's home directory, but it can be trivially modified to create a filesystem for other purposes.
Create an encryption key
This section can be skipped if you intend to unlock the drive by typing a password rather than unlocking automatically. You should use a password instead if your root partition is not encrypted. The location of the file /etc/home.key can be anything.
# dd if=/dev/random of=/etc/home.key bs=32 count=1 # chmod 600 /etc/home.key
IMPORTANT: Make sure you don't lose this key by overwriting your root filesystem or anything like that. You might want to store a copy of it on an encrypted USB drive, for instance.
Create the zpool
Replace `/dev/sd...` with the name of the disk or partition where you would like to make the zfs filesystem, such as `/dev/nvme0n1` or `/dev/sda1`. If you would like to be prompted for a password at boot rather than using the key as generated above, then replace `-O keylocation=file:///etc/home.key -O keyformat=raw` with `-O keylocation=prompt -O keyformat=passphrase`. The name "homepool" can be anything.
# zpool create -o ashift=12 -O acltype=posixacl -O compression=lz4 \ -O dnodesize=auto -O normalization=formD -O relatime=on -O xattr=sa \ -O encryption=aes-256-gcm -O keylocation=file:///etc/home.key -O keyformat=raw \ -O mountpoint=none homepool /dev/sd...
After completing this, verifying that the pool has been created should return something like::
# zpool status pool: homepool state: ONLINE config: NAME STATE READ WRITE CKSUM homepool ONLINE 0 0 0 sd... ONLINE 0 0 0 errors: No known data errors
To create and mount the filesystem issue the command
# zfs create -o mountpoint=/home/username homepool/username # chown username:username /home/username
The last command i.e chown is likely unnecessary if not creating a homedir.
Service configuration
Setup the following services to auto-mount the new filesystem using OpenRC
To import existing zpools:
# rc-update add zfs-import
To load the encryption keys:
# rc-update add zfs-load-key
Finally to mount the filesystems:
# rc-update add zfs-mount
Reboot the system, so that the encrypted ZFS drive is automounted and becomes available for use.
Maintenance
On alpine Linux, there in no cron/script provided to scrub (and eventually trim) your pool(s) on a regular basis, like in other Linux distributions.
Scrub
Scrubbing (and eventually trim) your pool(s) on a Oracle regular basis is essential. The scrub examines all data in the specified pools to verify that it checksums correctly. For replicated (mirror, raidz, or draid) devices, ZFS automatically repairs any damage discovered during the scrub
When scrubbing a pool with encrypted filesystems the keys do not need to be loaded. However, if the keys are not loaded and an unrepairable checksum error is detected the file name cannot be included in the zpool status -v verbose error report.
A scrub is split into two parts: metadata scanning and block scrubbing. The metadata scanning sorts blocks into large sequential ranges which can then be read much more efficiently from disk when issuing the scrub I/O.
This script is taken from debian zfs scripts. It is used to list the pools, make sure they are online, and no scrub is being done at the time. The contents of the script placed at /usr/libexec/zfs/scrub is as follows:
Contents of /usr/libexec/zfs/scrub
Make the script executable:
# chmod +x /usr/libexec/zfs/scrub
It is recommended to scrub regularly to assure your pool(s) and datas are in good shape. So the scrub script will be launched once a month, on the 2nd sunday of the month using cron
Edit your crontabs using the command:
# crontab -e
and add these 2 lines
# zfs scrub the second sunday of every month 24 0 8-14 * * if [ $(date +\%w) -eq 0 ] && [ -x /usr/libexec/zfs/scrub ]; then /usr/libexec/zfs/scrub; fi
Trim
The command zpool trim
initiates an immediate on-demand TRIM operation for all of the free space in a pool. This operation informs the underlying storage devices of all blocks in the pool which are no longer allocated and allows thinly provisioned devices to reclaim the space.
A manual on-demand TRIM operation can be initiated irrespective of the autotrim pool property setting. The link shows the types of vdev devices which can be trimmed.
This script is taken from debian zfs scripts. It is used to list the pools, make sure they are online, build only with NVME ssd drive(s) and no trim is being done at the time. The contents of the script placed at /usr/libexec/zfs/trim is as follows:
Contents of /usr/libexec/zfs/trim
Make the script executable:
# chmod +x /usr/libexec/zfs/trim
It is recommended to trim regularly to assure your pool(s) and datas are in good shape. So the trim script will be launched once a month, on the 1st sunday of the month using cron
Edit your crontabs using the command:
# crontab -e
and add these 2 lines
# zfs trim the first sunday of every month 24 0 1-7 * * if [ $(date +\%w) -eq 0 ] && [ -x /usr/libexec/zfs/trim ]; then /usr/libexec/zfs/trim; fi