OpenZFS
OpenZFS is a file system with volume management capabilities designed specifically for storage servers.
Some neat features of ZFS include:
- Aggregating multiple physical disks into a single filesystem.
- Automatically repairing data corruption.
- Creating point-in-time snapshots of data on disk.
- Optionally encrypting or compressing data on disk.
Usage⚑
Mount a pool as readonly⚑
zpool import -o readonly=on {{ pool_name }}
Mount a ZFS snapshot in a directory as readonly⚑
mount -t zfs {{ pool_name }}/{{ snapshot_name }} {{ mount_path }} -o ro
Mount a dataset that is encrypted⚑
If your dataset is encrypted using a key file you need to:
- Mount the device that has your keys
- Import the pool without loading the key because you want to override the keylocation attribute with zfs load-key. Without the -l option, any encrypted datasets won't be mounted, which is what you want.
- Load the key(s) for the dataset(s)
- Mount the dataset(s).
zpool import rpool # without the `-l` option!
zfs load-key -L file:///path/to/keyfile rpool
zfs mount rpool
Umount a pool⚑
zpool export pool-name
If you get an error of pool or dataset is busy
run the next command to see which process is still running on the pool:
lsof 2>/dev/null | grep dataset-name
List pools⚑
zpool list
List the filesystems⚑
zfs list
Clean the space of a ZFS pool⚑
It doesn't matter how big your disks are, you'll eventually reach it's limit before you can buy new disks. It's then time to clean up some space.
Manually remove data⚑
See which datasets are using more space for their data⚑
To sort the datasets on the amount of space they use for their backups use zfs list -o space -s usedds
Clean it up⚑
Then you can go dataset by dataset using ncdu
cleaning up.
See which datasets are using more space for their backups⚑
To sort the datasets on the amount of space they use for their backups use zfs list -o space -s usedsnap
See the differences between a snapshot and the contents of the dataset⚑
To compare the contents of a ZFS snapshot with the current dataset and identify files or directories that have been removed, you can use the zfs diff
command. Here's how you can do it:
- First, find the snapshot name using the following command:
zfs list -t snapshot dataset_name
- Then, compare the contents of the snapshot with the current dataset (replace
<snapshot_name>
with your snapshot name):
zfs diff <dataset>@<snapshot_name> <dataset>
For example:
zfs diff tank/mydataset@snap1
The output will show files and directories that have been removed (-
), modified (M
), or renamed (R
). Here's an example:
- 4 /path/to/removedfile.txt
If you want to see only the deleted files, you can pipe the output through grep
:
zfs diff <dataset>@<snapshot_name> | grep '^-'
This will help you identify which files or directories were in the snapshot but are no longer in the current dataset.
Get read and write stats from pool⚑
zpool iostat -v {{ pool_name }} {{ refresh_time_in_seconds }}
Get all properties of a pool⚑
zpool get all {{ pool_name }}
Get all properties of a filesystem⚑
zfs get all {{ pool_name }}
Set zfs module parameters or options⚑
Most of the ZFS kernel module parameters are accessible in the SysFS /sys/module/zfs/parameters
directory. Current values can be observed by
cat /sys/module/zfs/parameters/PARAMETER
Many of these can be changed by writing new values. These are denoted by Change|Dynamic in the PARAMETER details below.
echo NEWVALUE >> /sys/module/zfs/parameters/PARAMETER
If the parameter is not dynamically adjustable, an error can occur and the value will not be set. It can be helpful to check the permissions for the PARAMETER
file in SysFS.
In some cases, the parameter must be set prior to loading the kernel modules or it is desired to have the parameters set automatically at boot time. For many distros, this can be accomplished by creating a file named /etc/modprobe.d/zfs.conf
containing a text line for each module parameter using the format:
# change PARAMETER for workload XZY to solve problem PROBLEM_DESCRIPTION
# changed by YOUR_NAME on DATE
options zfs PARAMETER=VALUE
Some parameters related to ZFS operations are located in module parameters other than in the zfs kernel module. These are documented in the individual parameter description. Unless otherwise noted, the tunable applies to the zfs kernel module. For example, the icp
kernel module parameters are visible in the /sys/module/icp/parameters
directory and can be set by default at boot time by changing the /etc/modprobe.d/icp.conf
file.
Get compress ratio of a filesystem⚑
zfs get compressratio {{ filesystem }}
Rename or move a dataset⚑
NOTE: if you want to rename the topmost dataset look at rename the topmost dataset instead. File systems can be renamed by using the zfs rename
command. You can perform the following operations:
- Change the name of a file system.
- Relocate the file system within the ZFS hierarchy.
- Change the name of a file system and relocate it within the ZFS hierarchy.
The following example uses the rename
subcommand to rename of a file system from kustarz
to kustarz_old
:
zfs rename tank/home/kustarz tank/home/kustarz_old
The following example shows how to use zfs rename
to relocate a file system:
zfs rename tank/home/maybee tank/ws/maybee
In this example, the maybee
file system is relocated from tank/home
to tank/ws
. When you relocate a file system through rename, the new location must be within the same pool and it must have enough disk space to hold this new file system. If the new location does not have enough disk space, possibly because it has reached its quota, rename operation fails.
The rename operation attempts an unmount/remount sequence for the file system and any descendent file systems. The rename command fails if the operation is unable to unmount an active file system. If this problem occurs, you must forcibly unmount the file system.
You'll lose the snapshots though, as explained below.
Rename the topmost dataset⚑
If you want to rename the topmost dataset you need to rename the pool too as these two are tied.
$: zpool status -v
pool: tets
state: ONLINE
scrub: none requested
config:
NAME STATE READ WRITE CKSUM
tets ONLINE 0 0 0
c0d1 ONLINE 0 0 0
c1d0 ONLINE 0 0 0
c1d1 ONLINE 0 0 0
errors: No known data errors
To fix this, first export the pool:
$ zpool export tets
And then imported it with the correct name:
$ zpool import tets test
After the import completed, the pool contains the correct name:
$ zpool status -v
pool: test
state: ONLINE
scrub: none requested
config:
NAME STATE READ WRITE CKSUM
test ONLINE 0 0 0
c0d1 ONLINE 0 0 0
c1d0 ONLINE 0 0 0
c1d1 ONLINE 0 0 0
errors: No known data errors
Now you may need to fix the ZFS mountpoints for each dataset
zfs set mountpoint="/opt/zones/[Newmountpoint]" [ZFSPOOL/[ROOTor other filesystem]
Rename or move snapshots⚑
If the dataset has snapshots you need to rename them too. They must be renamed within the same pool and dataset from which they were created though. For example:
zfs rename tank/home/cindys@083006 tank/home/cindys@today
In addition, the following shortcut syntax is equivalent to the preceding syntax:
zfs rename tank/home/cindys@083006 today
The following snapshot rename operation is not supported because the target pool and file system name are different from the pool and file system where the snapshot was created:
$: zfs rename tank/home/cindys@today pool/home/cindys@saturday
cannot rename to 'pool/home/cindys@today': snapshots must be part of same
dataset
You can recursively rename snapshots by using the zfs rename -r
command. For example:
$: zfs list
NAME USED AVAIL REFER MOUNTPOINT
users 270K 16.5G 22K /users
users/home 76K 16.5G 22K /users/home
users/home@yesterday 0 - 22K -
users/home/markm 18K 16.5G 18K /users/home/markm
users/home/markm@yesterday 0 - 18K -
users/home/marks 18K 16.5G 18K /users/home/marks
users/home/marks@yesterday 0 - 18K -
users/home/neil 18K 16.5G 18K /users/home/neil
users/home/neil@yesterday 0 - 18K -
$: zfs rename -r users/home@yesterday @2daysago
$: zfs list -r users/home
NAME USED AVAIL REFER MOUNTPOINT
users/home 76K 16.5G 22K /users/home
users/home@2daysago 0 - 22K -
users/home/markm 18K 16.5G 18K /users/home/markm
users/home/markm@2daysago 0 - 18K -
users/home/marks 18K 16.5G 18K /users/home/marks
users/home/marks@2daysago 0 - 18K -
users/home/neil 18K 16.5G 18K /users/home/neil
users/home/neil@2daysago 0 - 18K -
Repair a DEGRADED pool⚑
First you need to make sure that it is in fact a problem of the disk. Check the dmesg
to see if there are any traces of reading errors, or SATA cable errors.
A friend suggested to mark the disk as healthy and do a resilver on the same disk. If the error is reproduced in the next days, then replace the disk. A safer approach is to resilver on a new disk, analyze the disk when it's not connected to the pool, and if you feel it's safe then save it as a cold spare.
Replacing a disk in the pool⚑
If you are going to replace the disk, you need to bring offline the device to be replaced:
zpool offline tank0 ata-WDC_WD2003FZEX-00SRLA0_WD-xxxxxxxxxxxx
Now let us have a look at the pool status.
zpool status
NAME STATE READ WRITE CKSUM
tank0 DEGRADED 0 0 0
raidz2-1 DEGRADED 0 0 0
ata-TOSHIBA_HDWN180_xxxxxxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWN180_xxxxxxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWN180_xxxxxxxxxxxx ONLINE 0 0 0
ata-WDC_WD80EFZX-68UW8N0_xxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWG180_xxxxxxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWG180_xxxxxxxxxxxx ONLINE 0 0 0
ata-WDC_WD2003FZEX-00SRLA0_WD-xxxxxxxxxxxx OFFLINE 0 0 0
ata-ST4000VX007-2DT166_xxxxxxxx ONLINE 0 0 0
Sweet, the device is offline (last time it didn't show as offline for me, but the offline command returned a status code of 0).
Time to shut the server down and physically replace the disk.
shutdown -h now
When you start again the server, it’s time to instruct ZFS to replace the removed device with the disk we just installed.
zpool replace tank0 \
ata-WDC_WD2003FZEX-00SRLA0_WD-xxxxxxxxxxxx \
/dev/disk/by-id/ata-TOSHIBA_HDWG180_xxxxxxxxxxxx
zpool status tank0
pool: main
state: DEGRADED
status: One or more devices is currently being resilvered. The pool will
continue to function, possibly in a degraded state.
action: Wait for the resilver to complete.
scan: resilver in progress since Fri Sep 22 12:40:28 2023
4.00T scanned at 6.85G/s, 222G issued at 380M/s, 24.3T total
54.7G resilvered, 0.89% done, 18:28:03 to go
NAME STATE READ WRITE CKSUM
tank0 DEGRADED 0 0 0
raidz2-1 DEGRADED 0 0 0
ata-TOSHIBA_HDWN180_xxxxxxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWN180_xxxxxxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWN180_xxxxxxxxxxxx ONLINE 0 0 0
ata-WDC_WD80EFZX-68UW8N0_xxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWG180_xxxxxxxxxxxx ONLINE 0 0 0
ata-TOSHIBA_HDWG180_xxxxxxxxxxxx ONLINE 0 0 0
replacing-6 DEGRADED 0 0 0
ata-WDC_WD2003FZEX-00SRLA0_WD-xxxxxxxxxxxx OFFLINE 0 0 0
ata-TOSHIBA_HDWG180_xxxxxxxxxxxx ONLINE 0 0 0 (resilvering)
ata-ST4000VX007-2DT166_xxxxxxxx ONLINE 0 0 0
The disk is replaced and getting resilvered (which may take a long time to run (18 hours in a 8TB disk in my case).
Once the resilvering is done; this is what the pool looks like.
zpool list
NAME SIZE ALLOC FREE EXPANDSZ FRAG CAP DEDUP HEALTH ALTROOT
tank0 43.5T 33.0T 10.5T 14.5T 7% 75% 1.00x ONLINE -
If you want to read other blogs that have covered the same topic check out 1.
Check the health of the degraded disk⚑
Follow these instructions.
RMA the degraded disk⚑
Follow these instructions.
Installation⚑
Install the required programs⚑
OpenZFS is not in the mainline kernel for license issues (fucking capitalism...) so it's not yet suggested to use it for the root of your filesystem.
To install it in a Debian device:
- ZFS packages are included in the
contrib
repository, but thebackports
repository often provides newer releases of ZFS. You can use it as follows.
Add the backports repository:
vi /etc/apt/sources.list.d/bullseye-backports.list
deb http://deb.debian.org/debian bullseye-backports main contrib
deb-src http://deb.debian.org/debian bullseye-backports main contrib
vi /etc/apt/preferences.d/90_zfs
Package: src:zfs-linux
Pin: release n=bullseye-backports
Pin-Priority: 990
- Install the packages:
apt update
apt install dpkg-dev linux-headers-generic linux-image-generic
apt install zfs-dkms zfsutils-linux
BE CAREFUL: if root doesn't have sbin
in the PATH
you will get an error of loading the zfs module as it's not signed. If this happens to you reinstall or try the debugging I did (which didn't work).
Create your pool⚑
First read the ZFS storage planning article and then create your main
pool with a command similar to:
zpool create \
-o ashift=12 \
-o autoexpand=on \
main raidz /dev/sda /dev/sdb /dev/sdc /dev/sdd \
log mirror \
/dev/disk/by-id/nvme-eui.e823gqkwadgp32uhtpobsodkjfl2k9d0-part4 \
/dev/disk/by-id/nvme-eui.a0sdgohosdfjlkgjwoqkegdkjfl2k9d0-part4 \
cache \
/dev/disk/by-id/nvme-eui.e823gqkwadgp32uhtpobsodkjfl2k9d0-part5 \
/dev/disk/by-id/nvme-eui.a0sdgohosdfjlkgjwoqkegdkjfl2k9d0-part5 \
Where:
-o ashift=12
: Adjusts the disk sector size to the disks in use./dev/sda /dev/sdb /dev/sdc /dev/sdd
are the rotational data disks configured in RAIDZ1- We set two partitions in mirror for the ZLOG
- We set two partitions in stripe for the L2ARC
If you don't want the main pool to be mounted use zfs set mountpoint=none main
.
If you don't want replication use zpool create main sde sdf sdg
Create your filesystems⚑
Once we have the pool you can create the different filesystems. If you want to use encryption with a key follow the next steps:
mkdir /etc/zfs/keys
chmod 700 /etc/zfs/keys
dd if=/dev/random of=/etc/zfs/keys/home.key bs=1 count=32
Then create the filesystem:
zfs create \
-o mountpoint=/home/lyz \
-o encryption=on \
-o keyformat=raw \
-o keylocation=file:///etc/zfs/keys/home.key \
main/lyz
If you want to use a passphrase instead [you can use the zfs create -o encryption=on -o keylocation=prompt -o keyformat=passphrase
command.
I'm assuming that compression
was set in the pool.
You can check the created filesystems with zfs list
Enable the autoloading of datasets on boot⚑
It is possible to automatically unlock a pool dataset on boot time by using a systemd unit. For example create the following service to unlock any specific dataset:
/etc/systemd/system/zfs-load-key.service
[Unit]
Description=Load encryption keys
DefaultDependencies=no
After=zfs-import.target
Before=zfs-mount.service
[Service]
Type=oneshot
RemainAfterExit=yes
ExecStart=/usr/bin/zfs load-key -a
StandardInput=tty-force
[Install]
WantedBy=zfs-mount.service
systemctl start zfs-load-key.service
systemctl enable zfs-load-key.service
reboot
Configure NFS⚑
With ZFS you can share a specific dataset via NFS. If for whatever reason the dataset does not mount, then the export will not be available to the application, and the NFS client will be blocked.
You still must install the necessary daemon software to make the share available. For example, if you wish to share a dataset via NFS, then you need to install the NFS server software, and it must be running. Then, all you need to do is flip the sharing NFS switch on the dataset, and it will be immediately available.
Install NFS⚑
To share a dataset via NFS, you first need to make sure the NFS daemon is running. On Debian and Ubuntu, this is the nfs-kernel-server
package.
sudo apt-get install nfs-kernel-server
Further, with Debian and Ubuntu, the NFS daemon will not start unless there is an export in the /etc/exports
file. So, you have two options: you can create a dummy export, only available to localhost, or you can edit the init script to start without checking for a current export. I prefer the former. Let's get that setup:
echo '/tmp localhost(ro)' >> /etc/exports
$ sudo /etc/init.d/nfs-kernel-server start
$ showmount -e hostname.example.com
Export list for hostname.example.com:
/mnt localhost
With our NFS daemon running, we can now start sharing ZFS datasets. The sharenfs
property can be on
, off
or opts
, where opts
are valid NFS export options. So, if you want to share the pool/srv
dataset, which is mounted to /srv
to the 10.80.86.0/24
network you could run:
# zfs set sharenfs="rw=@10.80.86.0/24" pool/srv
# zfs share pool/srv
# showmount -e hostname.example.com
Export list for hostname.example.com:
/srv 10.80.86.0/24
/mnt localhost
If you need to share to multiple subnets, you would do something like:
sudo zfs set sharenfs="rw=@192.168.0.0/24,rw=@10.0.0.0/24" pool-name/dataset-name
If you need root
to be able to write to the directory enable the no_root_squash
NFS option
root_squash — Prevents root users connected remotely from having root privileges and assigns them the user ID for the user nfsnobody. This effectively "squashes" the power of the remote root user to the lowest local user, preventing unauthorized alteration of files on the remote server. Alternatively, the no_root_squash option turns off root squashing. To squash every remote user, including root, use the all_squash option. To specify the user and group IDs to use with remote users from a particular host, use the anonuid and anongid options, respectively. In this case, a special user account can be created for remote NFS users to share and specify (anonuid=,anongid=), where is the user ID number and is the group ID number.
You should now be able to mount the NFS export from an NFS client. Install the client with:
sudo apt-get install nfs-common
And then mount it with:
mount -t nfs hostname.example.com:/srv /mnt
To permanently mount it you need to add it to your /etc/fstab
, check this section for more details.
Configure a watchdog⚑
Watchdogs are programs that make sure that the services are working as expected. This is useful for example if you're suffering the ZFS pool is stuck error.
- Install Python bindings for systemd to get logging functionality.
sudo apt install python3-systemd
- Create the next script:
#!/usr/bin/env python3
"""
ZFS watchdog service
"""
import os
import socket
import time
import sys
import signal
from datetime import datetime
from pathlib import Path
from systemd import journal
def log(message: str) -> None:
"""Log message to journal."""
journal.send(message)
def get_notify_socket() -> socket.socket:
"""Connect to the notify socket."""
notify_socket = os.environ.get("NOTIFY_SOCKET", None)
assert notify_socket is not None, "NOTIFY_SOCKET needs to be defined"
internal_notify_socket = socket.socket(
socket.AF_UNIX, socket.SOCK_DGRAM | socket.SOCK_CLOEXEC
)
internal_notify_socket.connect(notify_socket)
return internal_notify_socket
def get_wait_period() -> int:
"""Deduce the wait period between checks.
Get WATCHDOG_USEC, convert it to seconds and get ~0.75 of it (update watchdog every 3s if WATCHDOG_USEC is 5s).
With a minimum of 1 second.
"""
watchdog_usec = os.environ.get("WATCHDOG_USEC", None)
assert watchdog_usec is not None, "WATCHDOG_USEC needs to be defined"
return max(int(int(watchdog_usec) / 1000000 * 0.75), 1)
def send_socket_message(message: bytes) -> None:
"""Send message over socket"""
notify_socket = get_notify_socket()
# log(f"Sending message {message} to the socket")
notify_socket.send(message)
def socket_notify_ready() -> None:
"""Mark service as ready"""
send_socket_message(b"STATUS=Ready")
send_socket_message(b"READY=1")
log("ZFS watchdog is ready")
def socket_notify_stop(*args) -> None:
"""Mark service as stopping and exit.
We need the args argument to be compatible with the signal.signal signature.
"""
send_socket_message(b"STATUS=Stopping")
send_socket_message(b"STOPPING=1")
log("ZFS watchdog is stopped")
sys.exit(0)
def socket_notify_running() -> None:
"""Mark service as running"""
send_socket_message(
("STATUS=Running, checking every %is" % get_wait_period()).encode("utf8")
)
log("ZFS watchdog is running")
def watchdog_continue(
last_journal_update: datetime, journal_update_frequency: int = 600
) -> datetime:
"""Update watchdog timestamp."""
# log("Sending the message to the watchdog")
send_socket_message(b"WATCHDOG=1")
now = datetime.now()
if (now - last_journal_update).seconds > journal_update_frequency:
log("ZFS watchdog is still alive")
return now
return last_journal_update
def watchdog_skip() -> None:
log("Watchdog timestamp not updated.")
def check_condition() -> bool:
"""Check ZFS is working as expected.
We're able to:
- Write a new file
- Read from the file
- Remove the file.
In a decent amount of time. If any of the operations gets hanged up for longer than
the WATCHDOG_USEC the server will be restarted.
"""
check_path = os.environ.get("ZFS_FILE_CHECK_PATH", None)
assert check_path is not None, "ZFS_FILE_CHECK_PATH needs to be defined"
check_path = Path(check_path)
# log(f"Checking path {check_path}.")
# Write a new file
# log(f"Writing to path {check_path}.")
check_path.write_text("Alive")
# Read the file
# log(f"Reading from path {check_path}.")
assert check_path.read_text() == "Alive"
# Remove the file
# log(f"Removing path {check_path}.")
os.remove(check_path)
# log(f"Check passed.")
return True
def wait() -> None:
wait_period = get_wait_period()
time.sleep(wait_period)
if __name__ == "__main__":
log(f"Using socket {os.environ.get('NOTIFY_SOCKET', None)}")
for send_signal in (signal.SIGINT, signal.SIGABRT, signal.SIGTERM):
signal.signal(send_signal, socket_notify_stop)
socket_notify_ready()
socket_notify_running()
last_journal_update = datetime.now()
while True:
if check_condition():
last_journal_update = watchdog_continue(last_journal_update)
# log(f"Last journal update was: {last_journal_update")
wait()
If you're creating another python watchdog, you can copy-paste the whole script and only tweak the check_condition
to check whatever you want to check.
If you need to debug the watchdog script, run it manually with python and exporting the needed variables, otherwise you run the risk of entering and never ending restart loop
- Create a systemd service
systemctl edit --full -force zfs_watchdog
and add:
[Unit]
Description=ZFS watchdog
Requires=zfs.target
After=zfs.target
[Service]
ExecStart=/usr/bin/zfs_watchdog.py
WatchdogSec=60s
Restart=on-failure
NotifyAccess=all
StartLimitInterval=5min
StartLimitBurst=4
StartLimitAction=reboot
Environment="ZFS_FILE_CHECK_PATH=/path/to/your/mountpoint/zfs_test"
[Install]
WantedBy=multi-user.target
If you're debugging still the script use StartLimitAction=
instead so that you don't get unexpected reboots. - Start the service with systemctl start zfs_watchdog
. - Check that it's working as expected with journalctl -feu zfs_watchdog
- Once you're ready everything is fine enable the service systemctl enable zfs_watchdog
Monitor the watchdog⚑
If you're using Prometheus with the Node exporter in theory if the watchdog fails it will show up as a failed service. For the sake of redundancy we can create a Loki alert that checks that the watchdog is still alive, if the watchdog fails or if it restarts the server.
groups:
- name: zfs_watchdog
rules:
- alert: ZFSWatchdogIsDeadError
expr: |
(count_over_time({unit="zfs_watchdog.service"} |= `is still alive` [11m]) or on() vector(0)) == 0
for: 0m
labels:
severity: critical
annotations:
summary: "The ZFS watchdog is dead at {{ $labels.hostname}}"
- alert: ZFSWatchdogError
expr: |
count_over_time({job="systemd-journal", syslog_identifier="systemd"} |= `Watchdog timeout` [10m]) > 0
for: 0m
labels:
severity: warning
annotations:
summary: "The ZFS watchdog did not complete successfully at {{ $labels.hostname}}"
- alert: ZFSWatchdogRebooting
expr: |
count_over_time({job="systemd-journal", syslog_identifier="systemd"} |= `Rebooting: unit zfs_watchdog.service failed`[10m]) > 0
for: 0m
labels:
severity: info
annotations:
summary: "The server {{ $labels.hostname}} is being rebooted by the ZFS watchdog"
Configure the deadman failsafe measure⚑
ZFS has a safety measure called the zfs_deadman_failmode. When a pool sync operation takes longer than zfs_deadman_synctime_ms
, or when an individual I/O operation takes longer than zfs_deadman_ziotime_ms
, then the operation is considered to be "hung". If zfs_deadman_enabled
is set, then the deadman behavior is invoked as described by zfs_deadman_failmode
. By default, the deadman is enabled and set to wait which results in "hung" I/O operations only being logged. The deadman is automatically disabled when a pool gets suspended.
zfs_deadman_failmode
configuration can have the next values:
wait
: Wait for a "hung" operation to complete. For each "hung" operation a "deadman" event will be posted describing that operation.continue
: Attempt to recover from a "hung" operation by re-dispatching it to the I/O pipeline if possible.panic
: Panic the system. This can be used to facilitate automatic fail-over to a properly configured fail-over partner.
Follow the guides under Set zfs module parameters or options to change this value.
Backup⚑
Please remember that RAID is not a backup, it guards against one kind of hardware failure. There's lots of failure modes that it doesn't guard against though:
- File corruption
- Human error (deleting files by mistake)
- Catastrophic damage (someone dumps water onto the server)
- Viruses and other malware
- Software bugs that wipe out data
- Hardware problems that wipe out data or cause hardware damage (controller malfunctions, firmware bugs, voltage spikes, ...)
That's why you still need to make backups.
ZFS has the builtin feature to make snapshots of the pool. A snapshot is a first class read-only filesystem. It is a mirrored copy of the state of the filesystem at the time you took the snapshot. They are persistent across reboots, and they don't require any additional backing store; they use the same storage pool as the rest of your data.
If you remember ZFS's awesome nature of copy-on-write filesystems, you will remember the discussion about Merkle trees. A ZFS snapshot is a copy of the Merkle tree in that state, except we make sure that the snapshot of that Merkle tree is never modified.
Creating snapshots is near instantaneous, and they are cheap. However, once the data begins to change, the snapshot will begin storing data. If you have multiple snapshots, then multiple deltas will be tracked across all the snapshots. However, depending on your needs, snapshots can still be exceptionally cheap.
ZFS snapshot lifecycle management⚑
ZFS doesn't though have a clean way to manage the lifecycle of those snapshots. There are many tools to fill the gap:
sanoid
: Made in Perl, 2.4k stars, last commit April 2022, last release April 2021- zfs-auto-snapshot: Made in Bash, 767 stars, last commit/release on September 2019
- pyznap: Made in Python, 176 stars, last commit/release on September 2020
- Custom scripts.
It seems that the state of the art of ZFS backups is not changing too much in the last years, possibly because the functionality is covered so there is no need for further development. So I'm going to manage the backups with sanoid
despite it being done in Perl because it's the most popular, it looks simple but flexible for complex cases, and it doesn't look I'd need to tweak the code.
Remove all snapshots of a dataset⚑
zfs list -t snapshot -o name path/to/dataset | tail -n+2 | tac | xargs -n 1 zfs destroy -r
Manually create a backup⚑
To create a snapshot of tank/home/ahrens
that is named friday
run:
zfs snapshot tank/home/ahrens@friday
Restore a backup⚑
You can list the available snapshots of a filesystem with zfs list -t snapshot {{ pool_or_filesystem_name }}
, if you don't specify the pool_or_filesystem_name
it will show all available snapshots.
You have two ways to restore a backup:
mount -t zfs main/lyz@autosnap_2023-02-17_13:15:06_hourly /mnt
To umount the snapshot run umount /mnt
.
- Rolling back the filesystem to the snapshot state: Rolling back to a previous snapshot will discard any data changes between that snapshot and the current time. Further, by default, you can only rollback to the most recent snapshot. In order to rollback to an earlier snapshot, you must destroy all snapshots between the current time and that snapshot you wish to rollback to. If that's not enough, the filesystem must be unmounted before the rollback can begin. This means downtime.
To rollback the "tank/test" dataset to the "tuesday" snapshot, we would issue:
$: zfs rollback tank/test@tuesday
cannot rollback to 'tank/test@tuesday': more recent snapshots exist
use '-r' to force deletion of the following snapshots:
tank/test@wednesday
tank/test@thursday
As expected, we must remove the @wednesday
and @thursday
snapshots before we can rollback to the @tuesday
snapshot.
See how much space do your snapshots consume⚑
When a snapshot is created, its space is initially shared between the snapshot and the file system, and possibly with previous snapshots. As the file system changes, space that was previously shared becomes unique to the snapshot, and thus is counted in the snapshot’s used
property.
Additionally, deleting snapshots can increase the amount of space that is unique for use by other snapshots.
Note: The value for a snapshot’s space referenced property is the same as that for the file system when the snapshot was created.
You can display the amount of space or size that is consumed by snapshots and descendant file systems by using the zfs list -o space
command.
# zfs list -o space -r rpool
NAME AVAIL USED USEDSNAP USEDDS USEDREFRESERV USEDCHILD
rpool 10.2G 5.16G 0 4.52M 0 5.15G
rpool/ROOT 10.2G 3.06G 0 31K 0 3.06G
rpool/ROOT/solaris 10.2G 3.06G 55.0M 2.78G 0 224M
rpool/ROOT/solaris@install - 55.0M - - - -
rpool/ROOT/solaris/var 10.2G 224M 2.51M 221M 0 0
rpool/ROOT/solaris/var@install - 2.51M - - - -
From this output, you can see the amount of space that is:
- AVAIL: The amount of space available to the dataset and all its children, assuming that there is no other activity in the pool.
-
USED: The amount of space consumed by this dataset and all its descendants. This is the value that is checked against this dataset's quota and reservation. The space used does not include this dataset's reservation, but does take into account the reservations of any descendants datasets.
The used space of a snapshot is the space referenced exclusively by this snapshot. If this snapshot is destroyed, the amount of
used
space will be freed. Space that is shared by multiple snapshots isn't accounted for in this metric. * USEDSNAP: Space being consumed by snapshots of each data set * USEDDS: Space being used by the dataset itself * USEDREFRESERV: Space being used by a refreservation set on the dataset that would be freed if it was removed. * USEDCHILD: Space being used by the children of this dataset.
Other space properties are:
- LUSED: The amount of space that is "logically" consumed by this dataset and all its descendents. It ignores the effect of
compression
andcopies
properties, giving a quantity closer to the amount of data that aplication ssee. However it does include space consumed by metadata. - REFER: The amount of data that is accessible by this dataset, which may or may not be shared with other dataserts in the pool. When a snapshot or clone is created, it initially references the same amount of space as the filesystem or snapshot it was created from, since its contents are identical.
See the differences between two backups⚑
To identify the differences between two snapshots, use syntax similar to the following:
$ zfs diff tank/home/tim@snap1 tank/home/tim@snap2
M /tank/home/tim/
+ /tank/home/tim/fileB
The following table summarizes the file or directory changes that are identified by the zfs diff
command.
File or Directory Change | Identifier |
---|---|
File or directory has been modified or file or directory link has changed | M |
File or directory is present in the older snapshot but not in the more recent snapshot | — |
File or directory is present in the more recent snapshot but not in the older snapshot | + |
File or directory has been renamed | R |
Create a cold backup of a series of datasets⚑
If you've used the -o keyformat=raw -o keylocation=file:///etc/zfs/keys/home.key
arguments to encrypt your datasets you can't use a keyformat=passphrase
encryption on the cold storage device. You need to copy those keys on the disk. One way of doing it is to:
- Create a 100M LUKS partition protected with a passphrase where you store the keys.
- The rest of the space is left for a partition for the zpool.
WARNING: substitute /dev/sde
for the partition you need to work on in the next snippets
To do it: - Create the partitions:
fdisk /dev/sde
n
+100M
n
w
- Create the zpool
zpool create cold-backup-01 /dev/sde2
Sync an already created cold backup⚑
Mount the existent pool⚑
Imagine your pool is at /dev/sdf2
:
- Connect your device
- Check for available ZFS pools: First, check if the system detects any ZFS pools that can be imported:
sudo zpool import
This command will list all pools that are available for import, including the one stored in /dev/sdf2
. Look for the pool name you want to import.
- Import the pool: If you see the pool listed and you know its name (let's say the pool name is
mypool
), you can import it with:
sudo zpool import mypool
- Import the pool from a specific device: If the pool isn't showing up or you want to specify the device directly, you can use:
sudo zpool import -d /dev/sdf2
This tells ZFS to look specifically at /dev/sdf2
for any pools. If you don't know the name of the pool this is also the command to run.
This should list any pools found on the device. If it shows a pool, import it using:
sudo zpool import -d /dev/sdf2 <pool_name>
- Mount the pool: Once the pool is imported, ZFS should automatically mount any datasets associated with the pool. You can check the status of the pool with:
sudo zpool status
Additional options:
- If the pool was exported cleanly, you can use
zpool import
without additional flags. - If the pool wasn’t properly exported or was interrupted, you might need to use
-f
(force) to import it:
sudo zpool import -f mypool
Monitorization⚑
Monitor the ZFS events⚑
You can see the ZFS events using zpool events -v
. If you want to be alerted on these events you can use this service to ingest them into Loki and raise alerts.
Monitor the dbgmsg
file⚑
If you use loki remember to monitor the /proc/spl/kstat/zfs/dbgmsg
file:
- job_name: zfs
static_configs:
- targets:
- localhost
labels:
job: zfs
__path__: /proc/spl/kstat/zfs/dbgmsg
Troubleshooting⚑
To debug ZFS errors you can check:
- The generic kernel logs:
dmesg -T
,/var/log/syslog
or where kernel log messages are sent. - ZFS Kernel Module Debug Messages: The ZFS kernel modules use an internal log buffer for detailed logging information. This log information is available in the pseudo file
/proc/spl/kstat/zfs/dbgmsg
for ZFS builds where ZFS module parameterzfs_dbgmsg_enable = 1
ZFS pool is stuck⚑
Symptom: zfs or zpool command appear hung, does not return, and is not killable
Likely cause: kernel thread hung or panic
If a kernel thread is stuck, then a backtrace of the stuck thread can be in the logs. In some cases, the stuck thread is not logged until the deadman timer expires.
The only way I've yet found to solve this is rebooting the machine (not ideal). I even have to use the magic keys -.- . A solution would be to reboot server on kernel panic but it's not the kernel who does the panic but a module of the kernel, so that solution doesn't work.
You can monitor this issue with loki using the next alerts:
groups:
- name: zfs
rules:
- alert: SlowSpaSyncZFSError
expr: |
count_over_time({job="zfs"} |~ `spa_deadman.*slow spa_sync` [5m])
for: 1m
labels:
severity: critical
annotations:
summary: "Slow sync traces found in the ZFS debug logs at {{ $labels.hostname}}"
message: "This usually happens before the ZFS becomes unresponsible"
And to patch it you can use a software watchdog that reproduces the error and automatically resets the server. Another patch can be to Configure the deadman failsafe measure to panic
.
kernel NULL pointer dereference in zap_lockdir⚑
There are many issues open with this behaviour: 1, 2
In my case I feel it happens when running syncoid
to send the backups to the backup server.
Clear a permanent ZFS error in a healthy pool⚑
Sometimes when you do a zpool status
you may see that the pool is healthy but that there are "Permanent errors" that may point to files themselves or directly to memory locations.
The idea is that you analyze the problem and check your hard drives health. Once you've done that clear the errors with zpool clear my_pool
and run two scrubs zpool scrub my_pool
to clear the errors.
It takes a long time to run, so be patient.
If you want to stop a scrub run:
zpool scrub -s my_pool
If you're close to the event that made the error you can check the zpool events -v
to shed some light.
A few notes:
- repaired errors are shown in the counters, but won't elicit a "permanent errors" message
- "permanent" isn't the best word to use, because all of the error data is historical, based on time when an I/O operation failed. In some cases, transient errors (eg cable unplugged for a few minutes, or iSCSI network reconfiguration) can result in a "permanent" error message, but the data can be accessible and fine.
- scrubbing can fix problems, but so can reading or writing. So the difference between a read and a scrub is the scrub goes through everything and reads all copies, sides of mirrors, and parity blocks for verification. For example, if a mirrored data is read, but one side of the mirror was corrupted, then the other side will be read, verified, and if successfully verified the repair is made.
- on large systems, scrubs can take days, weeks, or months, so it isn't practical to run a scrub just to make an error notice go away. Of particular note, you don't want to delete the previous error data at the start of a scrub that can take a month.
zpool status
is not a replacement for async system monitoring. For the latter, use zpool events or configure a zed monitor (zed watches the events)- similarly, if you want detailed information on a specific failure, zpool events -v shows detailed information about both correctable and uncorrectable errors.
You can read this long discussion if you want more info.
ZFS pool is in suspended mode⚑
Probably because you've unplugged a device without unmounting it.
If you want to remount the device you can follow these steps to symlink the new devfs entries to where zfs thinks the vdev is. That way you can regain access to the pool without a reboot.
So if zpool status says the vdev is /dev/disk2s1, but the reattached drive is at disk4, then do the following:
cd /dev
sudo rm -f disk2s1
sudo ln -s disk4s1 disk2s1
sudo zpool clear -F WD_1TB
sudo zpool export WD_1TB
sudo rm disk2s1
sudo zpool import WD_1TB
If you don't care about the zpool anymore, sadly your only solution is to reboot the server. Real ugly, so be careful when you umount zpools.
Cannot receive incremental stream: invalid backup stream Error⚑
This is usually caused when you try to send a snapshot that is corrupted.
To solve it: - Look at the context on loki to identify the snapshot in question. - Delete it - Run the sync again
This can be monitored with loki through the next alert:
- alert: SyncoidCorruptedSnapshotSendError
expr: |
count_over_time({syslog_identifier="syncoid_send_backups"} |= `cannot receive incremental stream: invalid backup stream` [15m]) > 0
for: 0m
labels:
severity: critical
annotations:
summary: "Error tryig to send a corrupted snapshot at {{ $labels.hostname}}"
message: "Look at the context on loki to identify the snapshot in question. Delete it and then run the sync again"
Learning⚑
I've found that learning about ZFS was an interesting, intense and time consuming task. If you want a quick overview check this video. If you prefer to read, head to the awesome Aaron Toponce articles and read all of them sequentially, each is a jewel. The docs on the other hand are not that pleasant to read. For further information check JRS articles.