Skip to content

Latest commit

 

History

History
3647 lines (2366 loc) · 153 KB

File metadata and controls

3647 lines (2366 loc) · 153 KB

Secure and reproducible Arch Linux

Introduction

Goals

This guide has the following goals:

  • Automate secure installation of Arch Linux.
  • Document security best practices and processes around performing backups, using YubiKeys, operating Secure Boot etc. in a single place.
  • Provide optimal security against unauthorized access to online services you use.
  • Provide optimal security against unauthorized access to data on your devices.

Audience

This guide is mainly targets developers and system administrators using Linux as daily operating system on their workstations, which either use or would like to use Arch Linux.

Motivation

The main motivation for this project is to gather and combine various best practices around secure use of YubiKeys, Secure Boot, TPM, GPG and other security-oriented technologies implemented using Arch Linux distribution. There is many guides describing each of those technologies, but no so many which combines them to create a complete solution, which satisfy certain assumptions and protects against specific threat models. This guide aims to give you thorough understanding of security practices described here to raise your confidence in them, so you can proof under what conditions you remain secure and to highlight which conditions you should avoid, as your threat model does not protect against them.

In addition, this project was created to automate and document usually manual and complex processes of generating and maintaining used secrets in secure way, installing and configuring Arch Linux as well as managing the data and backups on used machines. Every person dealing with their own data should be aware how to manage and secure it. Every person dealing with secrets should know how to manage them in secure manner. Every Arch Linux user should have automated installation and data restoration process to avoid problems when OS installation gets corrupted, hardware breaks or data gets removed. This guide proposes the approach for handling those processes and describes them in detail.

Many tutorial about YubiKey management also provide misleading information on how many secrets there are to configure in YubiKey or say "This PUN is important, make sure you write it down somewhere safe". This guide provides a framework for this kind of secret management, ensuring that the secrets you use are managed in a safety manner.

Status

At the time of writing, Assumptions, Bootstrapping and Hardware bootstrapping sections are mostly complete, as they are the base of the project.

Glossary is partially written.

Miscellaneous section contains many topics documented as well.

OS Recovery Volume profile, Arch Linux installation automation and Day-2 Operations are not documented yet.

Table of Contents

Assumptions

This section describes used security measures in designed setup and possible attack scenarios against the data which this guide tries to protect.

Each of the protected resource is broken down into "regular" authentication method, special authentication method (if applicable) and possible attack vectors.

Security of your Daily Password Manager

Regular authentication methods for Daily Password Manager

In the regular circumstances, your Daily Password Manager will be protected by the following factors:

This data will be stored on the following devices:

  • Any of your daily devices like laptop or mobile phone.
  • Your local backup device.
  • Your remote backup storage.

This password should never be written down or stored directly to make it harder to leak. Harder does not mean impossible, as for example, if you type your password on a compromised device (e.g. with software keylogger) or via hardware keylogger, your password will effectively be compromised. This is why it is important to only type your Master Password on trusted devices.

Last factor required to get access to your password manager is a physical possession and physical presence of your Hardware Security Device, in this case YubiKey.

E.g. if someone takes over your machine using remote control software, they won't be able to open your locked password manager.

+ During regular operation, only possession of your YubiKey make it possible to unlock the password database.
+ Alternatively, one must additionally posses the secret used to configure the YubiKey.
+ In case of brute-force attack, using HSM increases the entropy of the encryption key for the database.

Attack vectors for Daily Password Manager

An attacker may have additional ways of getting information from your password manager. This includes scenarios like:

If an attacker manages to trick you into pulling rogue software update or running some rogue software on your machine, so they can take it over, they may then remotely access data in your password manager.

If you leave your machine unattended and unlocked, your password manager can be easily compromised.

Security of your online services

Regular authentication methods for online services

The security of online services you use will differ from service to service. All services are usually protected with the following measures:

Using password manager allows you to use unique password for every service, with optimal level of entropy to make used password impossible to brute-force.

+ Using unique password for each service ensures, that even if one password leaks in any way, only single service is affected by this. It prevents attacker from gaining access to your other accounts using the same password.
+ Using high-entropy passwords makes it almost impossible to brute-force. If attacker gets access to hashed version of your password, they won't be able to make use of it anyway.

Some services over MFA authentication using the following mechanism:

Most popular 2nd authentication factor (2FA). In this guide, TOTP secrets are stored on YubiKey and obtaining them requires physical touch of the device. This means if someone compromises your machine remotely, they won't be able to obtain codes required for logging in into the services.

+ Using TOTP baked by hardware key protects your accounts if your machine gets compromised via software.

Some services may also offer FIDO2 authentication. In such case, the protection is based on 2 factors:

YubiKey offers FIDO2 authentication, where secret key is stored on your YubiKey and touch is required to confirm physical presence.

As a second factor, PIN is used so even if someone steals your security key, they won't be able to use it to impersonate you.

Special authentication methods for online services

In case you lose access to both of your YubiKeys, you should have special MFA recovery codes backed up on Offline Backup Volume, which should allow you to re-gain access to your account.

Attack vectors for online services

When attacker wants to take over your online account, they have the following attack vectors:

If an attacker gets remote access to your Daily Password Manager, they can then access all the services, which are not protected by MFA. Services protected with OATH-TOTP or FIDO2 should remain safe.

If you leave your machine unattended, including security key plugged in, an attacker with physical access may access all your accounts, including ones protected with OATH-TOTP. Only accounts protected using FIDO2 should remain safe, as they require Master PIN every time you authenticate.

Do note, that an attacker with physical access may install a keylogger on your machine and trick you to provide your Master PIN, then again access your machine unattended and compromise all your accounts.

If targeted online service itself is vulnerable to some attacks, exploiting such vulnerability may allow an attacker to take over your account in there.

If network environment you work in is controlled by attacker, they may trick you into visiting their own version of service to trick you to send them your password. In such scenario, again, accounts using MFA should remain secure.

In case of MITM attacks, an attacker may also try to steal your browser cookies, which will allow to access your account, bypassing authentication requirement.

Using VPN provides good protection against MITM attacks.

An attacker may try to trick you into telling them you own password. This scenario is also safe for accounts using MFA.

If online service is vulnerable to XSS vulnerability, they may try to trick you into clicking a link which opening will result in sending your session cookies to the attacker, so they can impersonate you bypassing the authentication.

One can try to protect themselves against such attacks by disabling or conditionally enabling JavaScript on the websites you visit.

Security of your data

Security of your data will depend on where the data is stored. For each data storage, we will again identify protection measures and attack vectors.

With this guide we identify the following storages for your data:

  • PC disk
  • Mobile phone built-in storage
  • Local backup storage
  • Remote backup storage
  • Offline Backup Volume

PC disk

With this guide all your disks are protected using Full Disk Encryption done with LUKS.

Regular authentication methods for PC disks

This may sound obvious, but for accessing your encrypted data, either physical presence or authenticated network access must be satisfied to get access to your data.

For each of your disk there will be unique encryption key generated with entropy level, which makes brute-forcing such key infeasible. In regular use, this key will be additionally protected with GPG encryption.

As your disk encryption key is encrypted, your GPG encryption key stored on YubiKey will be needed every time you need to decrypt your PC disk and in order to use your GPG encryption key, you must provide your Master PIN, which should only be stored in your memory.

As your disk encryption key is encrypted, your GPG encryption key stored on YubiKey will be needed every time you need to decrypt your PC disk.

In addition to the encryption key, there will be a secret generated, which will be stored in TPM of your device. This ensures, that your data can only be accessed on your machine running your, trusted software.

Special authentication methods for PC disks

In addition to the authentication methods above, there will be secondary encryption key generated, which will be stored in Offline Backup Volume. It can be used in the following situations:

  • Your machine BIOS gets updated.
  • You lose your encryption key.
  • Your TPM gets wiped.
  • Your machine gets damaged (e.g. access to TPM, but not HDD).

In order to access the Recovery Key, you must first have access to Offline Backup Volume described below.

Attack vectors for PC disks

If your machine gets compromised via software, an attacker will effectively bypass the authentication and gets access to all your data.

If you leave your machine unlocked, an attacker with physical access to it may access and dump all the data stored on it.

Mobile phone built-in storage

With mobile devices, we cannot have as sophisticated security settings as with PC disks, so this section describes what is used for mobile devices.

Regular authentication methods for mobile built-in storage

Mobile devices cannot be decrypted remotely, so physical access to the device is required to get access to data stored on it.

Mobile device will have encryption password set, which is stored in the security module on device itself, which should protect it against brute-force attacks.

Attack vectors for mobile built-in storage

If your device is running vulnerable software and it's turned on, it might get exploited which will give an attacker remote access to your data.

Qualified attacker may also extract encryption keys from your device RAM, assuming it has physical access to your device, then use this encryption key to get access to your data.

If your device gets stolen locked and attacker will posses your fingerprint, they may get access to data stored on your device.

- The fingerprint can be taken from the device itself, making this attack relatively easy.

If an attacker manages to posses your mobile device unlocked (e.g. stealing it), they immediately get access to all your data and all online services you are logged into.

Local backup storage

Your local backup storage might be less secure than your PC disk, due to the fact that it might be accessed from multiple devices. Depending on architecture of your local storage, you will use different authentication methods.

Regular authentication methods for local backup storage

If your local backup storage is a portable disk, similar to PC disk protection, it will be protected using encrypted encryption key, which should be stored on all devices accessing your disk.

If your local backup storage is a portable disk, similar to PC disk protection, to decrypt encryption key of your device, you must have physical access to your YubiKey.

If your local backup storage is a portable disk, similar to PC disk protection, to decrypt encryption key of your device, you must know your Master PIN.

If your local backup storage is a remote system (e.g. local server), then you will use your Hardware Security Module and Master PIN to authenticate to this remote system.

Special authentication methods for local backup storage

This step is optional, but if you wish, you may generate a special recovery key which will be stored on your Offline Backup Volume.

Attack vectors for local backup storage

If you use local server as local backup storage, compromising this device will effectively give an attacker access to your backups and possibly data inside it.

If an attacker compromises your machine, either via software or by physical presence, they may access your local backup storage, which may give them access to data from your other devices.

Remote backup storage

With this guide, BorgBackup is used for backups, but you may as well use restic. Depending on the supported storage protocol you may favor one tool or another.

Regular authentication methods for Borg

As Borg performs remote backups over SSH protocol, access to backups will be authenticated the same way as backup to your server, which is, via GPG key stored on your YubiKey.

In addition to your YubiKey, which stores private key, Master PIN is needed to be able to use the PIN. This ensures, that even if someone steals your YubiKey, they won't be able to use it.

In addition to authentication mechanism for accessing remote storage, backups are also protected with static password, which is used to protect encryption keys of your backups. This means even if an attacker get access to your storage, they won't be able to read your backups.

Special authentication methods for Borg

If for any reason both your YubiKeys becomes unusable, you can still access your backups using copy of your authentication key in Master GPG keyring stored in Offline Backup Volume.

Depending on how you configure your remote server, you may have a special SSH key used only for configuring your server. If this is the case, make sure this SSH key is secure, as it will also allow accessing your backups.

Attack vectors for remote backup storage with Borg

Attack vectors for remote backup storage with Borg are omitted, as they would be too complex to be feasible. An attacker would need to bypass almost all of your security measures to get access to your backups, it would probably be easier to get access to your personal device instead.

Summary

To summarize all describe attack vectors, here is the list of recommended best practices to follow:

  • Never leave your devices unattended unlocked.
  • If you leave your machine for longer period, hibernate it to remove disk encryption keys from memory.
  • Do not use your Master Password or Master PIN on untrusted hardware.
  • Keep your software up to date to protect it from known vulnerabilities.
  • Use VPN service on your devices, especially when working in untrusted network environment (e.g. Hotels).

The list above lists behavior practices. Other best practices are more one-time, like:

  • Using unique passwords with at least 256 bits of entropy.
  • Using FIDO2 where possible.
  • Using U2F where possible.
  • Using GPG keys where possible.

Requirements

This section describes what hardware/software is required to be able to make full use of this guide.

Detailed list

Following hardware is required:

1 x (or more) x86 Computer with UEFI support (and recommended Secure Boot and TPM support)

This guides assumes you have personal machine which will be used for daily operation. The guide also supports customized installation for each machine if you have more than one. The machine can be as well a virtual machine, though this is not recommended from security point of view, as host machine may not be trusted.

It is also helpful to actually have more than one machine available, so you can use other machine to help yourself with debugging if provisioning process of first machine fails for any reason.

- NOTE: Support for Secure Boot and TPM is highly recommended, without it your configuration will be less secure.

(Optional) 1 x 4 ports USB hub

For bootstrapping process, a lot of operations happens on USB devices, so to safe you some time and troubles with connecting, disconnecting, mounting and unmounting the devices, a passive 4 ports USB hub is a nice thing to have, so you can have the following devices plugged at the same time:

  • Arch Linux bootable ISO
  • Temporary Volume
  • Offline Backup Volume
  • YubiKey

1 x Temporary Computer running Linux, Windows or macOS

For bootstrapping process, this guide requires you to have a machine running modern OS, from which you will be able to create a bootable USB stick with Arch Linux ISO.

Target machine can be used as well so long you already have some operating system running on it.

2 x Hardware security device with PIV, OpenPGP and OATH - HOTP support like YubiKey 5 Series

The main point of this guide is to make use of hardware security device as much as possible. Two devices are required so one can be kept in secure place as a backup in case the primary one is lost, stolen or damaged.

2 x Dedicated removable storage device (e.g. pendrive) for offline backup of master keys and passwords

Some secrets generated as part of this guide (e.g. GPG master key) should be kept on always-offline volume to maximize their security. To ensure redundancy for those volumes, it is recommended to have at least 2 copies of them, to be able to handle situation when one of them gets corrupted or simply breaks. Those volumes may simply be encrypted pendrives.

2 x Dedicated removable storage device (e.g. pendrive) for a OS recovery volume

This guide assumes data stored on daily computer is either version controlled, reproducible or regularly backed up into a local storage. This means it should be possible and it is recommended to be able to re-install the OS on machine at any time in a timely manner. To be able to do that, one should always keep a dedicated OS recovery volume, which is capable of performing fully automated OS installation. Having this possibility allows not treating your workstation as snowflake, giving you confidence, that when workstation breaks or gets stolen, you can quickly recover from it.

- NOTE: While 1 device should be sufficient for installation, it is recommended to have 2, so in case you have issues with new version of recovery volume, you can fallback to use the old one.

1 x Temporary removable storage device (e.g. pendrive) for temporary storage

As part of bootstrapping process and in case of some incidents, it is recommended to have one storage devices from which can be formatted to be able to store files when rebooting machine from OS connected to the internet to offline mode.

1 x Dedicated removable storage device (e.g. USB HDD, network disk) for local backups

Following 3-2-1 Backup Rule and to be able to quickly restore your last backup (restoring from local disk will be faster than restoring from internet), it is recommended to have a local storage available, which do not require internet access.

1 x Dedicated remote storage server for remote backups (with SSH support for Borg or S3-like support for restic)

Again, following 3-2-1 Backup Rule, to keep your backups geographically secure, it is recommended to have a remote storage server for keeping another copy of your backups, in case when both active copy and local backup gets damaged (e.g. in apartment fire).

Summary

To summarize, following hardware is required for bootstrapping process:

  • 1 x x86 machine currently running modern OS
  • 2 x YubiKey 5 Series
  • 5 x 8GB+ pendrive (2 x backup + 2 x recovery + 1 x temporary)
  • (Optional) 1 x 4 ports USB hub

And additionally for day-2 operations:

  • 1 x Dedicated removable local storage device
  • 1 x Dedicated remote storage with either SSH or S3-like support

Bootstrapping

This section explains steps, which needs to be performed once to start using practices defined in this guide. It includes tasks like generating GPG Master Key, creating Master Password, generating Secure Boot Platform Key (PK) etc.

At the end of this section, you will have:

  • The following secrets generated and backed up on 2 encrypted pendrives:
    • Master Password
    • PIN
    • GPG AdminPIN
    • PIV PUK
    • YubiKey Challenge-Response secret
    • Secure Boot Platform Key (PK)
    • Secure Boot Key Encryption Key (KEK)
    • (Optional) Password salt
  • Both YubiKeys initialized with:
    • Signing key (DB) for Secure Boot via PIV applet.
    • GPG signing, encryption and authentication sub-keys via OpenPGP applet.
    • Challenge-Response on 2nd slot.
  • Recovery USB stick with personalized Arch Linux installer.

Preparation

Before we start creating secrets etc, we must prepare some dependencies which will be needed later. Follow the steps below.

Getting Arch Linux installation medium

First step of bootstrapping is to get a Arch Linux USB Stick created. We will use it without network configured for secrets generation.

For this step, following items are required from Requirements section:

  • 1 x Temporary removable storage device (e.g. pendrive)
  • 1 x Temporary computer running Windows, Linux or macOS

With items above prepared, head to USB flash installation medium and prepare your USB stick.

Before rebooting, make sure you remember an address of this guide, so you can continue following it there.

Once rebooted, make sure you configure your network before proceeding.

Configuring Arch Linux ISO

Connecting to the Wi-Fi

If you know the name of your device and your network SSID, you can use the command below to authenticate to the network. Once done, connection should be established automatically and DHCP client should configure the rest automatically.

iwctl station <device> connect <ssid>

To find your wireless device names, run:

iwctl device list

To scan and list available networks, run:

iwctl station <device> scan && iwctl station <device> get-networks
Expanding available disk space

To make some space if you need to install more packages, run the following command:

mount -o remount,size=4G /run/archiso/cowspace

NOTE: All packages will be installed into RAM, so make sure you have enough of it available. Depending on the age of your ISO, you may also need to install updates to make things working.

NOTE: 4GB should be sufficient to install all updates, Gnome Shell and Firefox.

(Optional) Installing and running graphical interface

If you want to have a graphical interface during bootstrapping, for example to be able to easily copy-paste the commands which needs to be executed, run the commands below:

# Ignore linux package (kernel update) to avoid removing kernel modules, which will be required for example for mounting ext4 filesystem.
# - firefox - For browsing Web and guide online.
# - gnome-control-center - For adjusting GUI settings like resolution or mouse.
# - gnome-shell - Main GUI package.
# - gnome-terminal - For Terminal in GUI.
# - keepassxc - For testing password strength.
# - nautilus - For visually checking files and mounts.
pacman -Syyu --ignore linux firefox gnome-control-center gnome-shell gnome-terminal keepassxc nautilus
XDG_SESSION_TYPE=wayland dbus-run-session gnome-session

NOTE: This is only required to be done manually during bootstrapping process. Later on, your customized ISO may have it included out of the box.

Fetching required resources into temporary volume

With Arch Linux USB stick running, we can fetch this repository, verify it's signature and run a script, which will pull all required dependencies into a temporary volume, so you can continue following bootstrapping process without the internet access, to make sure generated secrets are not exposed to the internet.

Fetching repository

Run the following command to import GPG signing public key, which is used to sign releases in this repository. This will allow to verify the signature of downloaded code.

curl https://github.com/invidian.gpg -o invidian.gpg
gpg --import invidian.gpg

Then, run the commands below to fetch and verify this repository:

export VERSION=bootstrap-v0.1.0-rc1
curl -L https://github.com/invidian/secure-and-reproducible-arch-linux/releases/download/${VERSION}/${VERSION}.tar.gz.asc -o ${VERSION}.tar.gz.asc
curl -L https://github.com/invidian/secure-and-reproducible-arch-linux/archive/${VERSION}.tar.gz -o ${VERSION}.tar.gz
gpg --verify ${VERSION}.tar.gz.asc ${VERSION}.tar.gz

If everything worked, you should see the output similar to the following:

gpg: Signature made Fri Jan 22 23:22:10 2021 UTC
gpg:                using RSA key C79F76DAB29245AE262EC790CEBABB44587E3AE2
gpg: Good signature from "Mateusz Gozdek <mgozdekof@gmail.com>" [unknown]

The output will also include the following:

gpg: WARNING: This key is not certified with a trusted signature!
gpg:          There is no indication that the signature belongs to the owner.
Primary key fingerprint: 297C 1601 AF63 2225 7066  7925 9718 7FA1 271E C224
     Subkey fingerprint: C79F 76DA B292 45AE 262E  C790 CEBA BB44 587E 3AE2

Despite this warning, the repository you downloaded is still correct according to the signing key that you downloaded.

To learn more about this warning, read Tails documentation about verifying images.

Let's also save information where the repository has been downloaded, so we can later on copy it into Temporary Volume:

export REPOSITORY_PATH=$(pwd)

Finally, unpack the repository and make it your working directory, so we can use scripts from it in the next steps:

export TARGET_DIR=secure-and-reproducible-arch-linux
mkdir -p ./${TARGET_DIR} && tar zxvf ${VERSION}.tar.gz -C ${TARGET_DIR} --strip-components=1
cd ${TARGET_DIR}/
(Optional) Format temporary volume

Once running Arch, plug your USB devices which will serve as a temporary volume, then use the command below to identify plugged devices:

ls -d /dev/disk/by-id/* | grep -v -E -- '(-part[0-9]+|by-id/(dm|lvm|wwn|nvme-eui|ata)-)'

If you are unsure, which of your device is which, in care of removable devices, you can run the following command:

journalctl -fk | grep 'SerialNumber:'

Then unplug and plug your device. It should print you the serial number, which you can find in the list created by the command above.

Once you identified your device, run the following command to make be able to automate remaining commands:

export TMP_DEVICE=/dev/disk/by-id/<your device>
export TMP_ID=tmp

Now, run the command below to examine the script which will create a new GPT partition table on your selected device and create one big partition on it:

cat ./scripts/partition-offline-backup-volume.sh

Once you confirm, that the script is safe to run, run it:

OBV_DEVICE=$TMP_DEVICE OBV_ID=$TMP_ID ./scripts/partition-offline-backup-volume.sh

Now, create a ext4 filesystem on the newly created partition using the following command:

export DEVICE=/dev/disk/by-partlabel/$TMP_ID
test -b $DEVICE && mkfs.ext4 -L $TMP_ID $DEVICE
Mount temporary volume

If your temporary volume was already formatted, so you did not identify it yet, do it as described above, then export information about it using commands below:

export TMP_DEVICE=/dev/disk/by-id/<your device>
export TMP_ID=tmp

And run commands below to mount it:

export PARTITION=${TMP_DEVICE}-part1
export MOUNTPOINT=/mnt/$TMP_ID
mkdir -p $MOUNTPOINT && mount $PARTITION $MOUNTPOINT

Finally, make temporary volume your working directory:

cd $MOUNTPOINT

The /mnt/tmp mountpoint will be used in the next steps.

Move repository into Temporary Volume

With Temporary Volume mounted, copy downloaded repository and associated GPG key into Temporary Volume by running the commands below:

cp -r $REPOSITORY_PATH/${TARGET_DIR}/ ./
Fetching packages

Use Terminal opened in previous step or make sure you're in the temporary volume as a working directly and run the following commands to download the packages, which we will install once we go into offline mode.

# Breakdown of dependencies:
# - archiso - To automatically pull all dependencies of archiso script.
# - ccid - Smart card driver.
# - efitools - For transforming Secure Boot signature lists.
# - git - For versioning and syncing data on Offline Backup Volumes.
# - gnupg-pkcs11-scd - For using PIV applet for GPG.
# - hopenpgp-tools - Fecommended by https://github.com/drduh/YubiKey-Guide for linting your GPG key.
# - libp11 - PKCS#11 engine for sbsign, requires to work with YubiKey.
# - opensc - Smart card tools required for p11tool to detect the YubiKey as smartcard.
# - sbsigntools - For testing Secure Boot signing using YubiKey.
# - softhsm - In case you use PIV applet for GPG sub-keys, but you want to use GPG without YubiKey.
# - tpm2-tools - For interating with TPM2.0 on your hardware.
# - yubikey-manager - For configuring YubiKey.
pacman -Syyw archiso ccid efitools git gnupg-pkcs11-scd hopenpgp-tools libp11 opensc sbsigntools softhsm tpm2-tools yubikey-manager
mkdir -p packages
cp /var/cache/pacman/pkg/* ./packages/
(Optional) Building ssss

In Master Password Recovery section, which we will get to later, there is a procedure described for recovering your Master Password in case you forget it. If you plan to make use of it, we need to build Shamir's Secret Sharing package for Arch Linux. This can be simply done using one of the helpers script in the repository.

First, remember to examine the script before running it to make sure it's safe to execute. This can be done using the command below:

cat ./secure-and-reproducible-arch-linux/scripts/build-ssss.sh

After confirming the script, execute it as below:

./secure-and-reproducible-arch-linux/scripts/build-ssss.sh

The script will build the ssss package and place it in packages directory on Temporary Volume, so we can install it offline.

Fetching recommended GPG client configuration

Next, download hardended GPG configuration we will use when generating GPG keys:

curl https://raw.githubusercontent.com/drduh/config/master/gpg.conf -o gpg.conf
Unmounting Temporary Volume

Before we reboot, we should safely unmount Temporary Volume. To do so, run the commands below:

cd $REPOSITORY_PATH && \
sync -f /mnt/tmp/* && \
umount /mnt/tmp

Rebooting into offline mode

With all dependencies from the internet pulled, we can now reboot to make sure our OS has not been tampered and to make sure we stay in offline mode.

Ensure you will be offline

Before you boot, ensure your Ethernet cable is unplugged to make sure you don't get network configured automatically.

Mounting temporary volume

After the reboot, repeat the steps from Mount temporary volume to make your Temporary Volume available and your working directory.

Ensuring disk space

To make sure you have enough space in RAM filesystem, to install all downloaded packages, run the following command:

mount -o remount,size=4G /run/archiso/cowspace

NOTE: All packages will be installed into RAM, so make sure you have enough of it available.

Installing dependencies offline

With temporary volume as your working directory, run the command below to install all required packages:

pacman -U ./packages/*

Creating secrets

Now we can proceed with the secrets generation.

Master Password

First thing to do is to create and memorize your personal Master Password. It will be used to protect your Daily Password Manager and to decrypt Offline Backup Volumes and this is the only password you will have to remember.

This password should never be written down or saved to minimize the risk of it leaking.

If you have GUI installed, you can run KeePassXC and use Password Generator feature there to verify strength of your Master Password.

If you forget your Master Password, you will loose access to the secrets mentioned above, which might have severe consequences. To avoid having your Master Password as a single point of failure, later on as part of the bootstrapping process, Master Password Recovery section describes how to split your Master Password securely, so in case of emergency, you will be able to recover it.

Backup volumes

Identifying plugged devices

With your new Master Password, plug your USB devices which will serve as a Offline Backup Volumes, then use the command below to identify plugged devices:

ls -d /dev/disk/by-id/* | grep -v -E -- '(-part[0-9]+|by-id/(dm|lvm|wwn|nvme-eui|ata)-)'

If you are unsure, which of your device is which, in care of removable devices, you can run the following command:

journalctl -fk | grep 'SerialNumber:'

Then unplug and plug your device. It should print you the serial number, which you can find in the list created by the command above.

Formatting

Once you identified your device, run the following command to make be able to automate remaining commands:

export OBV_DEVICE=/dev/disk/by-id/<your device>
export OBV_ID=OBV1 # Partition label is limited to 16 characters.

Now, run the command below to examine the script which will create a new GPT partition table on your selected device and create one big partition on it:

cat ./secure-and-reproducible-arch-linux/scripts/partition-offline-backup-volume.sh

Once you confirm, that the script is safe to run, run it:

./secure-and-reproducible-arch-linux/scripts/partition-offline-backup-volume.sh

Now, let's create a LUKS container on partition we created using the command below:

export PARTITION=/dev/disk/by-partlabel/$OBV_ID
test -b $PARTITION && \
  cryptsetup luksFormat --verbose --verify-passphrase --label $OBV_ID $PARTITION

Now we need to open created LUKS container to create a filesystem on it. This can be done with the command below:

cryptsetup open $PARTITION $OBV_ID

Now, create a ext4 filesystem using the following command:

export DEVICE=/dev/mapper/$OBV_ID
test -b $DEVICE && mkfs.ext4 -L $OBV_ID $DEVICE

And now mount it:

export MOUNTPOINT=/mnt/$OBV_ID
mkdir -p $MOUNTPOINT && mount $DEVICE $MOUNTPOINT

Now, you can repeat the steps above for your another recovery device.

Safe removal

If you need to unplug first volume to plug second one, follow this step to remove your device safely before unplugging. Run the following commands:

sync && \
umount /mnt/$OBV_ID && \
rmdir /mnt/$OBV_ID && \
cryptsetup close /dev/mapper/$OBV_ID
Mounting secure volume again

If you want to make your Offline Backup Volume available again, run the following commands to identify the right device:

ls -d /dev/disk/by-id/* | grep -v -E -- '(-part[0-9]+|by-id/(dm|lvm|wwn|nvme-eui|ata)-)'
journalctl -fk | grep 'SerialNumber:'

Now plug the device and export information about it:

export OBV_DEVICE=/dev/disk/by-id/<your device>
export OBV_ID=OBV1

And run commands below to decrypt and mount it:

export PARTITION=/dev/disk/by-partlabel/$OBV_ID
cryptsetup open $PARTITION $OBV_ID
export MOUNTPOINT=/mnt/$OBV_ID
mkdir -p $MOUNTPOINT && mount /dev/mapper/$OBV_ID $MOUNTPOINT
Summary

When you are finished, leave at least one device plugged, decrypted, mounted and make it your working directory, so we can save some files there.

Run command below to make your most recent Offline Backup Volume your working directory:

cd $MOUNTPOINT

Master PIN

Similar to Master Password, you should create and memorize digits-only PIN number, which will be used while performing actions with your YubiKey. Similar to Master Password, this PIN number should never be written down or typed on insecure machines.

PIN will be used for:

  • GPG Smartcard functionality
  • FIDO2 access

Configuring YubiKeys

Configuring applications

First thing to do when you configure your YubiKey is to ensure that all required interfaces it can expose are enabled and that they are secure.

Generating configuration lock code

First, using the command below, we generate the configuration lock code:

xxd -ps -l 16 /dev/urandom > yubikey-configuration-lock-code
Setting configuration lock code

Now, set the lock code using the command below:

ykman config set-lock-code --new-lock-code $(cat yubikey-configuration-lock-code)
Enabling required interfaces

This guide requires OTP application to be enabled over USB for KeePassXC HMAC-SHA1 Challenge-Response access, so enable it using the command below:

ykman config usb --enable OTP --lock-code $(cat yubikey-configuration-lock-code)

Now, if you are going to access your Daily Password Manager from mobile device over NFC, enable OTP over NFC as well:

ykman config nfc --enable OTP --lock-code $(cat yubikey-configuration-lock-code)

If you plan to store TOTP codes on your YubiKey, enable OATH application:

ykman config usb --enable OATH --lock-code $(cat yubikey-configuration-lock-code)

And similarly, if you plan to use this feature from mobile device over NFC, enable it with NFC:

ykman config nfc --enable OATH --lock-code $(cat yubikey-configuration-lock-code)

For common web authentication protocols, enable U2F and FIDO2:

ykman config usb --enable U2F --enable FIDO2 --lock-code $(cat yubikey-configuration-lock-code)

If you plan to use them over NFC, enable that:

ykman config nfc --enable U2F --enable FIDO2 --lock-code $(cat yubikey-configuration-lock-code)

Next, enable PIV applet for Secure Boot signing and possible PKCS#11 usage (e.g. S/MIME email encryption):

ykman config usb --enable PIV --lock-code $(cat yubikey-configuration-lock-code)

And for NFC if you plan to use it:

ykman config nfc --enable PIV --lock-code $(cat yubikey-configuration-lock-code)

Finally, enable OpenPGP applet for native GPG support:

ykman config usb --enable OPGP --lock-code $(cat yubikey-configuration-lock-code)

And if you plan to use it over NFC, for example with OpenKeychain:

ykman config nfc --enable OPGP --lock-code $(cat yubikey-configuration-lock-code)

NOTE: ykman mode is a simplified interface for ykman config [usb|nfc]. It manages the same settings, but do not support lock code.

Setting up HMAC-SHA1 Challenge-Response
Swapping slots

This guide use only one slot out of two available of OTP applet in YubiKey. First slot functionality is triggered by short touch and second slot is triggered by long touch.

By default first slot is configured with Yubico OTP, so one-time challenge will be typed into your computer when you touch the YubiKey button.

Even though OTP challenge is not confidential, accidental typing it might be annoying, so it is recommended to swap the slots, so first one will be used for HMAC-SHA1 Challenge-Response.

We swap slots instead of wiping or overriding first slot to retain a factory secret programmed by Yubico, so if you need to use Yubico OTP service with production cc prefix, it is still possible.

To swap the slots, run the following command:

ykman otp swap

NOTE: If you have access code configured on any of the slots, swapping will fail! You must remove access code from both slots to be able to swap them.

This command is non destructive and you can run it as many times as you want. Slots will be swapped each time.

To see currently programmed slots, run:

ykman otp info

**NOTE: Make sure you run the swap command mentioned above for each of your YubiKey. **

Checking existing Challenge-Response configuration

If you want to find out if your device has already Challenge-Response configured on some slot, run the following command:

ykchalresp -1 'Sample #2'

Then, try running the command below:

ykchalresp -2 'Sample #2'

If any of the command succeeds and prints a hash, it means on the used slot Challenge-Response is already configured.

Generating Access Code

To avoid accidental removal of your OTP configuration, it is recommended to set up an access code, which will be required for each configuration change. We will generate random Access Code and store it on Offline Backup Volumes.

To generate Access Code and write it to the file, run the command below:

(LC_ALL=C tr -dc '[:digit:]' < /dev/urandom | head -c12; echo) > yubikey-slot-1-hmac-sha1-challenge-response-access-code

Now, because we have 2 slots on each device, we generate one more Access Code by running the command below:

(LC_ALL=C tr -dc '[:digit:]' < /dev/urandom | head -c12; echo) > yubikey-slot-2-hmac-sha1-challenge-response-access-code
Configuring access code

To configure Access Code on your device, run the commands below:

ykman otp settings --new-access-code $(cat yubikey-slot-1-hmac-sha1-challenge-response-access-code) 1
ykman otp settings --new-access-code $(cat yubikey-slot-2-hmac-sha1-challenge-response-access-code) 2
Verifying access code

To verify that your access code is functional, you can run the command below:

ykman otp settings --delete-access-code 1

It should fail with the error similar to below:

Error: Failed to write to the YubiKey. Make sure the device does not have restricted access.
Generating Challenge-Response key

To be able to maintain redundancy between two keys, we must generate Challenge-Response key manually, then program it on each of YubiKeys.

To generate the key and write it to file, run the command below:

xxd -ps -l 20 /dev/urandom > yubikey-slot-1-hmac-sha1-challenge-response-secret

This command generates secret key from /dev/urandom in hex format with length of 40 characters, which will be suitable as Challenge-Response key.

Setting Challenge-Response key on the devices

With configuration generated, plug each of your YubiKey and run the following command to apply generated configuration:

ykman otp --access-code $(cat yubikey-slot-1-hmac-sha1-challenge-response-access-code) chalresp 1 -t $(cat yubikey-slot-1-hmac-sha1-challenge-response-secret)
Verifying Challenge-Response configuration

To verify, that both of your YubiKeys are configured the same way, you can run the command below for each of them. If everything is correct, they should both produce the same output:

ykchalresp -1 'Sample #2'

Additionally, if you like, you can reproduce the YubiKey functionality using the command below:

echo -n 'Sample #2'  | openssl dgst -sha1 -mac HMAC -macopt hexkey:$(cat yubikey-slot-1-hmac-sha1-challenge-response-secret) | awk '{print $2}'
hmac-lt64 configuration flag

By default, YubiKey Challenge-Response take challenges between 0 and 63 bytes and calculates response for them. With ykpersonalize, you can disable hmac-lt64 configuration flag, which makes YubiKey always require 64 bytes challenge. Note, that this mode may not be compatible with some applications.

See this GitHub issue for more details.

Setting up PIV applet

Next on the list is generation of the PIV applet, where we will store key pair with certificate, which will be used to sign the kernels for Secure Boot.

Generating Management Key

With YubiKey 5, you can either have only PIN configured, or PIN, PUK and Management Key. Since we want to have PUK functionality to make PIV block after unsuccessful PIN attempts, we need to generate the Management Key and PUK.

Run the command below to generate the Management Key:

export LC_CTYPE=C; dd if=/dev/urandom 2>/dev/null | tr -d '[:lower:]' | tr -cd '[:xdigit:]' | fold -w48 | head -1 > yubikey-piv-applet-management-key
Configuring Management Key

To use generated Management Key make sure you have pcscd.service unit running by executing the following command:

systemctl start pcscd.service

You can check it's status using the command below:

systemctl status pcscd.service

When the service is running, run the command below for each YubiKey plugged:

ykman piv change-management-key --management-key 010203040506070801020304050607080102030405060708 --new-management-key $(cat yubikey-piv-applet-management-key)

NOTE: 010203040506070801020304050607080102030405060708 Management Key used above is a default Management Key.

Generating PUK

Now, we need to generate PUK, which will be used, when you type your PIN incorrectly 3 times. PUK will be stored on Offline Backup Volume, so once you lock your PIV applet, you will need access to it to be able to unlock it.

To generate PUK, run the command below:

export LC_CTYPE=C; dd if=/dev/urandom 2>/dev/null | tr -cd '[:digit:]' | fold -w8 | head -1 > yubikey-piv-applet-puk

Commands above are taken from https://developers.yubico.com/yubico-piv-tool/YubiKey_PIV_introduction.html

Setting PUK

To set PUK, run the command below with each of your YubiKeys plugged:

ykman piv change-puk --puk 12345678 --new-puk $(cat yubikey-piv-applet-puk)
Setting PIN

Finally, we can set up PIN, which will be used to unlock the PIV applet after plugging it in. For PIN, you should use Master PIN mentioned above.

Run the command below to configure PIN on each YubiKey:

ykman piv change-pin -P 123456
Summary

For now, we do not generate any secrets using PIV applet. This will be done at later stage when we generate Secure Boot keys.

(Optional) Setting up OpenPGP applet

If you plan to use PIV certificates for GPG sub-keys and not utilize OpenPGP applet in YubiKey, you can skip this section.

This guide use GPG keys to protect main part of disk encryption keys, as well for signing Git objects. This means we need to configure OpenPGP applet on YubiKeys to act as a GPG smartcard.

Setting Admin PIN

Similarly to PUK for PIV applet, OpenPGP applet has Admin PIN, which is used when you exceed your PIN attempts limit. To generate Admin PIN for your YubiKeys, run the command below:

NOTE: This command will print your Admin PIN to the screen, so you can copy-paste it in the next step when configuring it. Make sure your display output is secure.

export LC_CTYPE=C; dd if=/dev/urandom 2>/dev/null | tr -cd '[:digit:]' | fold -w8 | head -1 | tee yubikey-gpg-smartcard-admin-pin

If you use graphical interface, make sure you copy your Admin PIN to clipboard before proceeding. It will be required for the next step.

To configure Admin PIN, run the following command:

gpg --change-pin

Now, select option 3. to change Admin PIN.

You will be prompted for the existing Admin PIN, which by default is 12345678.

Repeat this step for each of your YubiKeys.

Setting PIN

Now, you can configure your regular PIN using Master PIN. To do that, run the command below:

gpg --change-pin

And select option 1. to change your PIN.

You will be prompted for the existing PIN, which by default is 123456.

Repeat this step for each of your YubiKeys.

Enabling touch requirement

It is recommended to enable touch requirement for any of GPG operation, so in case when your machine gets compromised, attacker won't be able to use your GPG keys.

To enable it for all operations, run the following commands:

ykman openpgp set-touch --admin-pin $(cat yubikey-gpg-smartcard-admin-pin) aut on
ykman openpgp set-touch --admin-pin $(cat yubikey-gpg-smartcard-admin-pin) sig on
ykman openpgp set-touch --admin-pin $(cat yubikey-gpg-smartcard-admin-pin) enc on

Repeat this step for each of your YubiKeys.

Setting Reset Code

This guide do not make use of Reset Code. Reset Code in GPG applet is used only for resetting PIN capability. If you would like to generate and use Reset Code, generate it in similar way as Admin PIN was generated.

See this link for more details.

Configuring and securing other YubiKey applets

YubiKey has also other applets available to use, like OATH-TOTP or FIDO which supports protecting with credentials. If you plan to use those applets and you prefer to have them secured, follow the steps similar to above to generate unique keys/passwords and store them on Offline Backup Volume.

This guide will later on use OATH-TOTP for TPM authenticity validation, but it does not require password protection.

Generating GPG keys

Prepare GPG configuration

Open Terminal and change working directory to one of Offline Backup Volumes. Then run the following commands:

export GNUPGHOME=$(pwd)/gnupg-workspace
mkdir -p "$GNUPGHOME"
chmod 0700 "$GNUPGHOME"

Now, copy previously downloaded gpg.conf file into new GPG home directory by running command like:

export TEMPORARY_VOLUME=/mnt/tmp
cp $TEMPORARY_VOLUME/gpg.conf $GNUPGHOME/

Now you can run the command below to initialize the GPG database to verify the configuration:

gpg --list-keys

If successful, it should create few new files in gnupg-workspace directory.

Generate keys
Generating Master Key

First thing to do is to generate Master Key, which will be stored only on Offline Backup Volume. To do that, run the command below:

gpg --expert --full-generate-key

Now, select option (11) ECC (set your own capabilities):

11

We use ECC as it is recommended algorithm nowadays.

Now, type S to leave only Certify capability selected:

S

Then type Q to confirm the selection:

Q

Next, select 1 elliptic curve option. YubiKey supports Curve 25519 and secp256k1 and Curve 25519 is more commonly used:

(1) Curve 25519

Choose that Master Key should never expire:

0
y

Next, type your real name which will be used as an human-readable identifier for your key.

John Doe

Then type your email address:

john.doe@example.com

Now leave the Comment section empty, as this is considered best practice.

If everything you typed is correct, confirm with o:

o

Now, GPG will ask you to provide the passphrase for the key. Considering that by following this guide your GPG Master Key will be always kept offline and protected using your Master Password, it should be OK to leave the password blank.

After confirmation of empty password, your GPG key ID should be printed.

Now, run the command below to export information about your newly generated key to next commands:

export KEYID=0x$(gpg --list-secret-keys --with-colons | head -n1 | cut -d: -f5)
(Optional) Adding extra identities to your Master Key

If you have more than one username or email address, which you would like to associate with your new Master Key, you can add them using the command below:

gpg --edit-key $KEYID

Now, type adduid:

adduid

Type the details as explained above.

Finally, confirm with o:

o

You can repeat this process for as many identities as you have.

Note: Added identities will be available to everyone, so make deliberate decision if you like to have those IDs being public information.

(Optional) Selecting primary identity

Now, with gpg, by default last identity added will be marked as default. This will be marked with a . next to the identity number.

[ultimate] (1)  John Doe <john.doe@example.com>
[ unknown] (2). John Doe <john@sample.com>

If you want to switch your primary identity, select it using uid command. In the example, we will switch the primary UID back to first one:

uid 1

GPG should now show you that identity has been selected:

[ultimate] (1)* John Doe <john.doe@example.com>
[ unknown] (2). John Doe <john@sample.com>

Now, run primary to make selected identity primary. Then unselect the uid to confirm that first one is now primary:

primary
uid 1

You should now see the output similar to the following:

[ultimate] (1). John Doe <john.doe@example.com>
[ unknown] (2)  John Doe <john@sample.com>

Finally, type save to commit the changes:

save
(Optional) Generating Signing key

NOTE: If you would like to use PIV keys for GPG, you can skip this step.

Next key we need to generate is a Signing key, which will be used to sign information you generate like emails or git objects to proof they are coming from you. This key will be stored on YubiKey, but to be able to use the same key on both YubiKeys for redundancy, it must be generated offline, then transferred into both YubiKeys.

To generate the key, run the command below:

gpg --expert --edit-key $KEYID

Then, type addkey:

addkey

Select (10) ECC (sign only) kind of key:

10

And curve (1) Curve 25519 as explained above:

1

Now, it is recommended to renew your keys yearly, to ensure you still have access to your Master Key, so select 1 year of expiration time:

1y

Verify the expiration date and confirm:

y

And confirm again:

y

Now, GPG will ask you to provide the passphrase for the key. Considering that by following this guide your key will be stored only offline and on YubiKey, it should be OK to leave the password blank.

Finally, commit the result by using:

save

Signing key can also be saved without password, as it will be transferred into YubiKey anyway.

(Optional) Generating Encryption key

NOTE: If you would like to use PIV keys for GPG, you can skip this step.

Next key to generate is Encryption key, which will be used to encrypt the data which only you will be able to decrypt. The procedure is almost identical to generating Signing key.

To generate the key, run the command below:

gpg --expert --edit-key $KEYID

Then, type addkey:

addkey

Select (12) ECC (encrypt only) kind of key:

12

And curve (1) Curve 25519 as explained above:

1

Again, select 1y expiration time as explained above:

1y

Verify the expiration date and confirm:

y

And confirm again:

y

Now, GPG will ask you to provide the passphrase for the key. Considering that by following this guide your key will be stored only offline and on YubiKey, it should be OK to leave the password blank.

Finally, commit the result by using:

save

Encryption key can also be saved without password, as it will be transferred into YubiKey anyway.

(Optional) Generating Authentication Key

NOTE: If you would like to use PIV keys for GPG, you can skip this step.

Last key we will generate is Authentication key. This key is used when you log in into remote servers to prove your identity. This can can be used for example for logging in using SSH. GPG also supports TLS authentication.

To generate the key, run the command below:

gpg --expert --edit-key $KEYID

Then, type addkey:

addkey

Now, depending on the services you are going to use your authentication key, you may select either ECC key (recommended) or older RSA key. Note, that some cloud providers like AWS do not support ECC key types at the time of writing. However, if you still prefer to use ECC key with GPG, you can generate additional RSA key using PIV applet, which you can use for "legacy" services.

If you select (11) ECC (set your own capabilities) kind of key, type:

11

Enable Authenticate capability:

A

Disable Sign capability:

S

Confirm selection:

Q

Then select curve (1) Curve 25519 as explained above:

1

If you select (8) RSA (set your own capabilities), type:

8

Enable Authenticate capability:

A

Disable Sign capability:

S

Disable Encrypt capability:

E

Confirm selection:

Q

Now choose 4096 as key length:

4096

Similarly to other keys, select 1y expiration time:

1y

Verify the expiration date and confirm:

y

And confirm again:

y

Now, GPG will ask you to provide the passphrase for the key. Considering that by following this guide your key will be stored only offline and on YubiKey, it should be OK to leave the password blank.

Finally, commit the result by using:

save

Encryption key can also be saved without password, as it will be transferred into YubiKey anyway.

(Optional) Signing new Master Key with your existing key

If you already use GPG and you want to switch to use key which is stored on YubiKey, it is recommended to sign your new Master Key with your existing Master Key to proof to people trusting your existing key that this newly generated key belongs to you.

To do that, export your new public key using the command below:

gpg --export --armor $KEYID > $TEMPORARY_VOLUME/$KEYID.gpg

Now, unmount your temporary volume, plug it to machine where you have your existing master key available and use the command below to sign new key:

export NEW_KEY_ID=<KEYID from other machine>
gpg --sign-with EXISTING_KEY_ID --sign-key $NEW_KEY_ID

When it's done, you can export your new master key with new signature and copy it back onto temporary volume, so it can be imported on Offline Backup Volume.

gpg --export --armor $NEW_KEY_ID > TEMPORARY_VOLUME/$NEW_KEY_ID-signed.gpg

You can now also upload your new Master Key to key server, with included signature from old key. To do this, run the command below:

gpg --send-keys $NEW_KEY_ID

Mount back your temporary volume on Secure OS with Offline Backup Volume available and run:

gpg --import $TEMPORARY_VOLUME/${KEYID}-signed.gpg

Generating Secure Boot keys

This guide use Secure Boot to protect your machines against bootloader-level malware and to prevent unauthorized code execution on your machine. To use Secure Boot, you must generate your own set of keys for Secure Boot, which key used for signing the kernels will be stored on YubiKey. For that, you need PIV applet configured as explained in Setting up PIV applet section.

Checking if YubiKey PIV slot is in use

If you used your YubiKey before, it is good to check if there are any existing certificates on the key to avoid overriding it.

Checking if YubiKey PIV slot has certificate

To check if your YubiKey PIV slot has some certificate generated on a given slot, run the command below. In this case, we check slot 9c:

ykman piv export-certificate 9c -

If there is a certificate already installed, certificate will be printed. Sample output:

$ ykman piv export-certificate 9c -
-----BEGIN CERTIFICATE-----
MIICuDCCAaCgAwIBAgIUcLwNBpwgDIWAJSLpCKVny6O1kHwwDQYJKoZIhvcNAQEL
BQAwFjEUMBIGA1UEAwwLbWF0LXRlc3RpbmcwHhcNMjEwMzA3MTEzMTEwWhcNMjIw
MzA3MTEzMTEwWjAWMRQwEgYDVQQDDAttYXQtdGVzdGluZzCCASIwDQYJKoZIhvcN
AQEBBQADggEPADCCAQoCggEBAJ/XJPOSP7gm3pm7NuRGRdtf4tLTUoywJZ3LJxXk
lQmuGRo91/SJ2wnxFLosfSkHuYK6tURgLuIfknXUezExT2IUC0ZRme/oPsvLQ1ij
IUbupzmbbbLFDjFUsRbyZ3xxSjxIH2GaTzh8K3YlKmr1F5kI6OdiDnCGhXHBlYr0
QeT091h3FG3P7A78mwv4+Rn8ifdL37uYXlWVuW78dWkiLMdFHyjRTrQMlj2TIoMd
Pi0+GfmTokziTgM2sDAkRhbkGQDghlFMjFahj//nQkYWWxVjE1KwvgFEwth4iIrl
IMOV0WNubSYP9dJm2bTjDO5WublxLe2MxMTnRaZVmSkrXi0CAwEAATANBgkqhkiG
9w0BAQsFAAOCAQEAjq3BgqrddMAo3f+W1t8/ZqdWmxdsNs5RUs8Mhgp9PgNxa0aP
zDIcuD7JIBrwmXNrobPZRmmPJbKmHP3230Gyq8DyeAoEHVxICKf7p3OTFA/adAWw
ZXmxP7tfW/P35cSsZvNjL50gVGc33eirRWibbVkDTn+3FhTLyQe3wmXTbrdZFTJN
2j58AiARQ2MX2av6hDDmwEPC1yGImRWkzzToOc8R3xt5oqy+RJ0b60UwRLWArfuJ
LpJ8ZEJ6ltTjZGgyMRHn7VamijkeAgoAHwkRTwKT9DshLwmC2oVIOhPVm7Wbdaox
6cIwjyZypyBw0Pawr0GlA0Sl4oo1v0kWL458xw==
-----END CERTIFICATE-----

If there is no certificate, you should get error similar to below:

$ ykman piv export-certificate 9d -
Usage: ykman piv export-certificate [OPTIONS] SLOT CERTIFICATE
Try 'ykman piv export-certificate -h' for help.

Error: No certificate found.
Checking if YubiKey PIV slot has key pair

Checking if slot has a certificate is recommended before checking if slot has a private key, as certificate generation attempt is used to determine if slot already has a key pair generated.

To check if slot has key pair generated, run the command below:

** NOTE: This command may wipe the certificate from your slot. Use with caution! It is recommended to check for certificate existence before running this command. **

openssl genrsa -out /tmp/private.pem 2048
openssl rsa -in /tmp/private.pem -outform PEM -pubout | ykman piv generate-certificate -s "testing" 9d -

If slot already has some key, this command should return no output.

If slot is empty and there is no private key, you should see output similar to the following:

$ ykman piv generate-certificate 9c testing.pem -s foo-testing
Enter PIN:
Enter a management key [blank to use default key]:
Usage: ykman piv generate-certificate [OPTIONS] SLOT PUBLIC-KEY
Try 'ykman piv generate-certificate -h' for help.

Error: Certificate generation failed.
Touch your YubiKey...
Generating keys

Platform Key and Key Exchange Key for Secure Boot will be stored on Offline Backup Volume, as they are not required for daily operation. They will only be used during Hardware Bootstrapping process to roll out Database public key.

As some BIOSes may not support 4096 bit keys, we use 2048 bit keys.

As Secure Boot verification process to not take certificate expiry time into account, make it expire after one day.

As Secure Boot keys generation consist of several commands, we will use helper script. As usual, first examine the script:

cat /mnt/tmp/secure-and-reproducible-arch-linux/scripts/generate-secure-boot-keys.sh

Now, run it with your personal identifier as an argument to generate the keys:

/mnt/tmp/secure-and-reproducible-arch-linux/scripts/generate-secure-boot-keys.sh "John Doe's"

The content of the script is taken from Gentoo wiki.

Transferring Database Key into YubiKeys

With Secure Boot keys generate, now we can transfer Database key into the YubiKey.

First, make directory with generates secrets your working directory.

Then, we will import private key using the command below:

ykman piv import-key --management-key $(cat yubikey-piv-applet-management-key) --pin-policy ALWAYS --touch-policy ALWAYS 9c secureboot/db.key

Next, we can import the certificate:

ykman piv import-certificate --management-key $(cat yubikey-piv-applet-management-key) --verify 9c secureboot/db.crt

You can verify the import process by running:

ykman piv info

This process must be repeated for each of your YubiKeys.

Verifying signing capabilities

To verify, that both your YubiKeys can generate valid Secure Boot signature, we must configure PKCS#11 engine.

Run the commands below to configure it:

mkdir -p /etc/pkcs11/modules
echo "module: opensc-pkcs11.so" > /etc/pkcs11/modules/opensc-pkcs11.module

And make sure your pcscddaemon is running by executing the command below:

systemctl start pcscd.service

Now, run p11tool to verify it's able to detect your YubiKey:

p11tool --list-all-privkeys

You should get output similar to the following:

$ p11tool --list-all-privkeys
warning: no token URL was provided for this operation; the available tokens are:

pkcs11:model=p11-kit-trust;manufacturer=PKCS%2311%20Kit;serial=1;token=System%20Trust
pkcs11:model=p11-kit-trust;manufacturer=PKCS%2311%20Kit;serial=1;token=Default%20Trust
pkcs11:model=PKCS%2315%20emulated;manufacturer=piv_II;serial=cf397c0faff98e2d;token=%20kernel-signing%20key

Now, we can generate test Secure Boot signature using the command below:

sbsign --engine pkcs11 --key 'pkcs11:manufacturer=piv_II;id=%02' --cert secureboot/db.crt --output test-secure-boot-signature --detached /run/archiso/bootmnt/arch/boot/x86_64/vmlinuz-linux

This command will ask you for PIN twice and you need to finalize the process by touching the YubiKey.

NOTE: If you do not have db.crt file around anymore, you can generate it by running ykman piv export-certificate 9c db.crt, assuming you have already generated and transferred the certificate into your YubiKey.

Now, to verify generated signature, run:

sbverify --cert secureboot/db.crt --detached test-secure-boot-signature /run/archiso/bootmnt/arch/boot/x86_64/vmlinuz-linux

If everything went well, you should get the following output:

Signature verification OK

To clean up, you can now remove test signature file using command below:

rm test-secure-boot-signature

Master Password Recovery

It is recommended to prepare for a situation, where you cannot remember the Master Password anymore or if anything happens to you, your family is able to get access to your data on your behalf.

To do that securely, this guide recommends using Shamir's Secret Sharing implemented with ssss CLI tool, which allows you to split your Master Password into 3 or 5 shares, which you can then distribute across your trusted family members, friends, lawyer or a bank. Only after gathering the majority of the shares, one will be able to reveal your Master Password.

While Shamir's Secret Sharing has mixed opinions in terms of security and in certain scenarios it might be insecure, for such a rare use as recovery of the Master Password, it should be sufficient.

Sharding

Shards generation commands use a helper script to verify that you shard correct input. Examine the script before running it:

cat /mnt/tmp/secure-and-reproducible-arch-linux/scripts/safe-input-password.sh

Next, export the identifier for shards, which will be appended at the beginning of each shard:

export SHARD_IDENTIFIER=john-doe-master-password

You can generate your shares using the following command:

(set -o pipefail && /mnt/tmp/secure-and-reproducible-arch-linux/scripts/safe-input-password.sh | ssss-split -w $SHARD_IDENTIFIER -t 3 -n 5 -q | tr \\n ' ' && echo) > master-password-shards

You can see generated shards by inspecting master-password-shards file:

less master-password-shards

Now, we will encrypt each shard separately using your GPG key, so you can securely distribute it later.

First, select right recipient:

export RECIPIENT=$(gpg --with-colons --list-secret-keys $KEYID | grep '^uid' | head -n1 | cut -d: -f10)

Then, run the command below to encrypt each shard:

for i in {1..5}; do cat master-password-shards | awk -v i=$i '{print $i}' | gpg --encrypt --armor --recipient "${RECIPIENT}" > "master-password-shard-${i}.gpg"; done

This will generate 5 GPG encrypted files, which we will later on copy into temporary volume.

To verify that shards are properly encrypted and functional, use the command below:

(for i in 1 2 5; do gpg -d master-password-shard-${i}.gpg; done) | ssss-combine -t 3 | less

It should show you your Master Password.

You can now copy encrypted shards to Temporary Volume for distribution:

cp master-password-shard-*.gpg /mnt/tmp/
Distribution

When distributing the shards with trusted parties, make sure you do not have access to more than threshold (default is 3) - 1 shards at a time, for example when you decrypt and print them, to minimize the chance for any middle-ware (e.g. printer, your daily OS) to have access to number of shards required to restore the Master Password.

When sharing the shards with trusted parties, assume they may not be as technical as you, so make it clear what is a document you give them, under what conditions it should be used and how to use it. It may also include a list of all the parties who posses a shard to make a recovery process easier.

Periodic verification

Periodically, for example annually, you should verify with all the parties, that they still have access to their shards and that their contact information is actual. This is recommended to avoid situation, where in case of a need for recovery, you find out that majority of the shards is lost or certain parties are not reachable anymore.

Rotating Master Password

In case when you need to change your Master Password, all shards must be re-generated and re-distributed to all parties. Old shards should be destroyed in such scenario.

Combining shards

You can test the recovery process by executing the following command:

ssss-combine -t 3 -n 5

Now, provide 3 of 5 shards. As a result, you should see your Master Password being printed.

(Optional) Generating Password Salt

If you plan to use Password Salt, it should be safe to save it on Offline Backup Volume in case you forget it.

Generating secrets for your first hardware

Part of Hardware Bootstrapping process is generation of Disk Encryption Recovery Key and BIOS password, which will be stored on Offline Secure Volume.

To save some time on rebooting and mounting volumes, we will generate them right now.

Encrypted version of Disk Encryption Recovery Key will also be stored in encrypted version in first Recovery Volume, so we can perform regular automated Arch Linux installation.

Head to Selecting hardware hostname step of Generating hardware secrets section and follow the steps from there to generate required hardware secrets, then return here.

(Optional) Generating PIV authentication certificate

If you decided to generate ED25519 GPG authentication key, but you sometimes need RSA key e.g. for popular cloud providers like AWS or Azure, you can generate additional RSA key pair which will be stored on 9a authentication slot in PIV applet of your YubiKeys, so you can use them for SSH.

To generate new RSA private key, run the command below:

openssl genrsa -out yubikey-piv-applet-9a-private-key.pem 2048

Now, to generate public key, which will be used for PIV operations, run the command below:

openssl rsa -in yubikey-piv-applet-9a-private-key.pem -outform PEM -pubout -out yubikey-piv-applet-9a-public-key.pem

Then, we will import private key using the command below:

ykman piv import-key --management-key $(cat yubikey-piv-applet-management-key) --pin-policy ONCE --touch-policy ALWAYS 9a yubikey-piv-applet-9a-private-key.pem

If you didn't generate a certificate yet, we are going to do this right now. We will generate a self-signed certificate using imported RSA key, so PIV/PKCS #11 lib is able to extract the public-key from the smartcard. For that, first export desired certificate identifier:

export IDENTIFIER="John's Doe"

Then, run the command below to generate the certificate:

test -f yubikey-piv-applet-9a-certificate.pem || ykman piv generate-certificate --management-key $(cat yubikey-piv-applet-management-key) --subject "${IDENTIFIER} Authentication Key"  --valid-days 1 9a yubikey-piv-applet-9a-public-key.pem

Once finished, we can export the certificate so we can import it into other YubiKeys:

ykman piv export-certificate 9a yubikey-piv-applet-9a-certificate.pem

Now, if you already generated the certificate, run the command below to import it into YubiKey:

test -f yubikey-piv-applet-9a-certificate.pem && ykman piv import-certificate --management-key $(cat yubikey-piv-applet-management-key) --verify 9a yubikey-piv-applet-9a-certificate.pem

This way, both of your YubiKeys will have the same certificate and you will be able to create identical copy in case your YubiKey gets lost or destroyed.

You can verify the import process by running:

ykman piv info

(Optional) Generating PIV encryption certificate

If you plan to use PKCS#11 as a backend for GPG, it is recommended to generate separate encryption certificate on PIV applet, similarly to what you would do with OpenPGP applet.

This certificate will be stored on dedicated 9d slot.

Note, that while using PKCS#11 for GPG is more convenient, as GPG's scdaemon and opensc clients do not compete against exclusive access to pcscd daemon, it might be considered less secure due to fact, that at the time of writing OpenSC do not support RSA 4096 bits key nor ED25519. Only RSA 2048 bits key are supported.

To generate new RSA private key, run the command below:

openssl genrsa -out yubikey-piv-applet-9d-private-key.pem 2048

Now, to generate public key, which will be used for PIV operations, run the command below:

openssl rsa -in yubikey-piv-applet-9d-private-key.pem -outform PEM -pubout -out yubikey-piv-applet-9d-public-key.pem

Then, we will import private key using the command below:

ykman piv import-key --management-key $(cat yubikey-piv-applet-management-key) --pin-policy ONCE --touch-policy ALWAYS 9d yubikey-piv-applet-9d-private-key.pem

If you didn't generate a certificate yet, we are going to do this right now. We will generate a self-signed certificate using imported RSA key, so PIV/PKCS #11 lib is able to extract the public-key from the smartcard. For that, first export desired certificate identifier:

export IDENTIFIER="John's Doe"

Then, run the command below to generate the certificate:

test -f yubikey-piv-applet-9d-certificate.pem || ykman piv generate-certificate --management-key $(cat yubikey-piv-applet-management-key) --subject "${IDENTIFIER} Encryption Key"  --valid-days 1 9d yubikey-piv-applet-9d-public-key.pem

Once finished, we can export the certificate so we can import it into other YubiKeys:

ykman piv export-certificate 9d yubikey-piv-applet-9d-certificate.pem

Now, if you already generated the certificate, run the command below to import it into YubiKey:

test -f yubikey-piv-applet-9d-certificate.pem && ykman piv import-certificate --management-key $(cat yubikey-piv-applet-management-key) --verify 9d yubikey-piv-applet-9d-certificate.pem

This way, both of your YubiKeys will have the same certificate and you will be able to create identical copy in case your YubiKey gets lost or destroyed.

You can verify the import process by running:

ykman piv info

(Optional) Adding PIV certificates into GPG Master Key

If you generated all 3 PIV certificates (signing for Secure Boot, authentication and encryption), you can add them as a sub-keys to your GPG Master key to use them instead of natively generated ones. As explained in section above, this has advantages and disadvantages.

Configuring gpg-agent and gnupg-pkcs11-scd

First, we need to configure GPG to use gnupg-pkcs11-scd instead of shipped by default scdaemon, which allows using PIV certificates (tokens) with GPG. This can be done using the commands below:

echo "scdaemon-program /usr/bin/gnupg-pkcs11-scd" > ${GNUPGHOME}/gpg-agent.conf
cat <<EOF > ${GNUPGHOME}/gnupg-pkcs11-scd.conf
providers p1
provider-p1-library /usr/lib/opensc-pkcs11.so
EOF

Now, stop running gpg-agent using the command below, so it can pick up new configuration on new start:

gpgconf --kill all

If you run the command below, you should see that now GPG sees empty smartcard:

gpg --card-status
Detecting key grips

Now, run the command below to verify that gpg-agent can see keys from PIV applet:

echo "SCD LEARN" | gpg-agent --server gpg-connect-agent

If you get a lot of output and it includes CNs of your PIV certificates, you can extract the information about the keys using commands below:

AUTHENTICATION_KEY=""; until test -n "${AUTHENTICATION_KEY}"; do AUTHENTICATION_KEY=$(echo "SCD LEARN" | gpg-agent --server gpg-connect-agent 2>&1 | grep Authentication | grep '^S' | awk '{print $3}' | tail -n1); done; echo "Authentication keygrip: $AUTHENTICATION_KEY"
SIGNING_KEY=""; until test -n "${SIGNING_KEY}"; do SIGNING_KEY=$(echo "SCD LEARN" | gpg-agent --server gpg-connect-agent 2>&1 | grep Signing | grep '^S' | awk '{print $3}' | tail -n1); done; echo "Signing keygrip: $SIGNING_KEY"
ENCRYPTION_KEY=""; until test -n "${ENCRYPTION_KEY}"; do ENCRYPTION_KEY=$(echo "SCD LEARN" | gpg-agent --server gpg-connect-agent 2>&1 | grep Encryption | grep '^S' | awk '{print $3}' | tail -n1); done; echo "Encryption keygrip: $ENCRYPTION_KEY"

This information will be needed in next steps when adding keys to GPG Master Key.

Adding keys to Master Key

With gnupg-pkcs11-scd fully configured, we can now add PIV keys into our GPG Master key. To start, run the command below:

gpg --edit-key --expert $KEYID
Adding authentication key

Add new key:

addkey

Select (13 Existing key):

13

Now, paste first key ID printed above, which should be authentication key.

Then, deselect Sign and Encrypt capabilities and select Authenticate instead:

s
e
a

And confirm:

q

Set expiration time to 1y, similar to native GPG keys:

1y
y
y
Adding signing key

Add new key:

addkey

Select (13 Existing key):

13

Now, paste second key ID printed above, which should be signing key.

Then, deselect Encrypt capability to leave only Sign:

e

And confirm:

q

Set expiration time to 1y, similar to native GPG keys:

1y
y
y
Adding encryption key

Add new key:

addkey

Select (13 Existing key):

13

Now, paste last key ID printed above, which should be encryption key.

Then, deselect Sign capability to leave only Encrypt:

s
e
a

And confirm:

q

Set expiration time to 1y, similar to native GPG keys:

1y
y
y

Finally, save everything you've done:

save

At this point, you should be able to use GPG to sign, encrypt and authenticate.

Saving and synchronizing data between Offline Backup Volumes

At this point, you should have all secrets created or generated, which are covered by the general bootstrap process. There will be more secrets to generate, which will be done for each machine you have, described in Hardware Bootstrapping section, but this will be done later.

At this point, all secrets are stored on a single Offline Backup Volume and we need to sync them into the other volume. To do that, we will be using Git, as this allows:

  • Easy synchronization between multiple copies of Offline Backup Volume.
  • Versioning of generated secrets.
  • Improved integrity with Git checksums.

Initializing repository

First, we will initialize Git repository, so we can add some local configuration, which will be persistent on Offline Backup Volume.

git init

Configuring Git

Generate basic Git configuration, which will use GPG. This can be done automatically using the command below:

export NAME=$(gpg --with-colons --list-keys $KEYID | grep '^uid' | head -n1 | cut -d: -f 10 | cut -d\< -f1 | sed 's/ $//g')
export EMAIL=$(gpg --with-colons --list-keys $KEYID | grep '^uid' | head -n1 | cut -d: -f 10 | cut -d\< -f2 | sed -E 's/( |>)$//g')
export SIGNING_KEY=$(gpg --with-colons --list-keys $KEYID | grep :s: | cut -d: -f5)
cat <<EOF | tee .git/config
[user]
        email = ${EMAIL}
        name = ${NAME}
        signingkey = ${SIGNING_KEY}!
[commit]
        gpgsign = true
EOF

To make sure git can ask you for PIV PIN, run the command below:

export GPG_TTY=$(tty)

Committing all data

With basic configuration, we can now commit all generated files into the Git repository by running the commands below:

git add --all
git commit -s -m "Initial commit"

You can now also verify, that Git signing is functional by running the command below:

git verify-commit HEAD

Do note, that signing commits here does not have much value, as we store Signing private key in this Git repository itself and it's not password protected. However, it proofs, that your Git configuration is functional.

Initial sync into other Offline Backup Volumes

Creating a copy of data into another Offline Backup Volume is as simple as running the command below:

export OBV_ID=OBV2
export MOUNTPOINT=/mnt/$OBV_ID
rm -rf ${MOUNTPOINT}/lost+found
git clone $(pwd) $MOUNTPOINT

Syncing further changes

If you rotate, modify or add new secrets in the future, to synchronize 2 Offline Backup Volumes, commit your changes in one copy, for example by running:

git add --all # To add all new, untracked files.
git commit -s

Then, change your working directory to the other Offline Backup Volume, for example by running:

export OBV_ID=OBV2
export MOUNTPOINT=/mnt/$OBV_ID
git -C $MOUNTPOINT pull --ff-only origin master

And if you want to pull from 2nd copy into the first one, make sure you configure the right origin URL by running the command as below:

git remote add origin /mnt/$OBV_ID

And also run:

git pull --ff-only origin master

(Optional) Transferring GPG keys into YubiKeys

If you use PIV certificates for your GPG keys, you can skip this section.

With all generates secrets synchronized and versioned, you can now transfer generated GPG keys into all your YubiKeys. As keytocard GPG command removes the keys from your local GPG keyring, we will use git restore functionality to restore them to be able to transfer them into backup YubiKey.

Before you start, make sure you remember or copy your GPG Admin PIN into clipboard. You can print your Admin PIN using the command below:

cat yubikey-gpg-smartcard-admin-pin

To transfer keys into YubiKey, start by running the command below:

gpg --edit-key $KEYID

Now, select the Signature key:

key 1

It should change ssb in the line with key with usage: S to ssb*, marking the key as selected.

Now, initiate the transfer of the key by running:

keytocard

Select (1) Signature key slot for storing the key:

1

Now, gpg will ask you for Admin PIN. It may actually ask you twice in this step.

Next, deselect Signature key and select Encryption key:

key 1
key 2

And initiate transfer to the card:

keytocard

This time, select only available (2) Encryption key slot:

2

Now, gpg will ask you for Admin PIN.

Next, deselect Encryption key and select authentication key:

key 2
key 3

Initiate transfer to card:

keytocard

Select (3) Authentication key slot:

3

Now, gpg will ask you for Admin PIN.

Finally, confirm everything:

save

You can verify that your GPG keys has been transferred to YubiKey by running the command below:

gpg --card-status

Now, run the command below to restore your GPG keyring to original state:

git restore gnupg-workspace/

Unplug your current YubiKey and plug the backup one, then repeat the procedure described above, including git restore command.

Copying public keys and certificates into Temporary Volume

As a last step in Offline Mode, before unplugging Offline Backup Volumes, we need to copy some of the generated secrets from it into a temporary volume, so we can embed them into a Recovery Volume we are going to build.

First, export your GPG public key, so we can upload it to key server:

gpg --export --armor $KEYID > /mnt/tmp/gpg-public-key.gpg

Next, copy Secure Boot certificates in various formats, to make sure we have format available, which is compatible with your BIOS, which will be needed for Hardware Bootstrapping:

mkdir -p /mnt/tmp/secureboot
cp secureboot/compound_db.* secureboot/compound_KEK.* secureboot/db.{auth,cer,crt,esl} secureboot/KEK.{auth,cer,crt,esl} secureboot/PK.{auth,cer,crt,esl} /mnt/tmp/secureboot/

Now, you can unmount both Temporary Volume and Offline Backup Volumes by running the commands below:

cd && sync && umount /mnt/*

Once finished, make sure you unplug your Offline Backup Volumes. Remember to never plug them into the machine connected to the network.

Creating first Arch Linux bootable USB device

In this step, we will prepare your first Recovery Volume, which is Arch Linux bootable USB device. This process has more details described in Recovery Volume maintenance section. Here, we are going to focus to create a basic Arch Linux ISO, which will allow you to perform automated Arch Linux installation over network.

Reboot your machine now and boot it again with Arch Linux ISO. Follow Configuring Arch Linux ISO and Mount Temporary Volume again to prepare your environment.

Archiso

The repository you downloaded at the beginning of bootstrap process and which is now stored on temporary volume includes a modified copy of archiso utility, which is used for building official Arch Linux ISO images. This guide uses modified version of it, which supports both Secure Boot and dm-verity, which is used to dynamically validate that data on the Recovery Volume has not been tempered.

By default archiso use systemd-boot to manage boot entries and GPG for validation the inner data image in ISO image. However, they do not provide any support for Secure Boot. The GPG signature validation is not very secure either without the Secure Boot, as the signature can simply be replaced with different one and you wouldn't even notice it. Also, GPG validation do not cover entire content of the ISO image (though files which are ignored are not really important). GPG validation may also be a bit slow on older hardware and is definitely slower than dm-verity, which is performed dynamically when OS accesses specific data blocks on the partition. dm-verity also offers protection during entire time the operating system is running. If file on verity device would be modified, reading such file will fail.

Modified version of archiso in this guide use Unified Kernel Images as a method for booting the ISO image. This method offers simplicity of setup, as there is no intermediate bootloader involved in the boot process, as Kernel image is built into the UEFI binary, which is executed by default by UEFI. One less component in boot process makes it easier to secure, as in this case, we build UEFI binary including kernel, kernel command line parameters and other information, then we sign it directly using Signing Key, which is stored in Secure Boot Signature Database (DB).

Limitations

The consequence of such setup is limited flexibility, as it is only possible to boot one image and set of parameters, as there is no boot loader to allow us to select what to boot. For example, booting memtest86+ from Recovery Volume is not possible, though it is acceptable for this guide, as the Recovery Volumes are designed to be used "often", depending how often you would like to reinstall your system, or how often your installation breaks and needs to be manually repaired, while memtest86+ is usually used only if your hardware is possible defected. On the other hand, once you install the OS, you can place memtest86+ and other UEFI binaries onto your EFI partition, sign them with Signing key and configure your UEFI to be able to boot them using efibootmgr, in addition to your default boot option.

Summary

You can find modified mkarchiso script here.

Default archiso profile

This repository also provides a customized version of Archiso releng profile, which adds tools requires by this guide and support for the features in used customized version of Archiso.

The shipped profile is addition-only and tries to only ship files which are required. A lot of files in this profile is actually not used (e.g. syslinux or efiboot configuration), but they are kept to allow easy synchronization with upstream Archiso profiles.

It is recommended, that each user creates their own copy (or fork) of this profile and customize it for their needs by adding hardware profiles etc.

You can find modified profile in archiso-recovery-volume-profile directory.

Customizing your Arch Linux ISO profile

Network setup
SSH Access
GUI

Saving your Arch Linux ISO profile

Building Arch Linux ISO using mkarchiso

Writing your Arch Linux ISO into Recovery Volume

Summary

Hardware Bootstrapping

When you purchase new hardware, there are some one-time steps required to adopt the hardware into your build system and to "own" them by rolling your own BIOS password and Secure Boot keys. This section explains those steps.

Generating hardware secrets

This guide handles BIOS password and Disk Encryption Recovery Keys as secrets unique to each hardware you own. Unique secrets per device minimize the impact if any of those gets compromised, an attacker won't be able to use it against your other hardware pieces.

Booting Recovery Volume in Offline mode

To generate the secrets, boot your Recovery Volume and make sure it stays in Offline mode by stopping systemd-networkd.service unit in first 30 seconds after boot. You should do it using the command below:

systemctl stop systemd-networkd.service

Mounting Offline Backup Volume

Next, make your Offline Backup Volume available. Use the following commands to identify the right device:

ls -d /dev/disk/by-id/* | grep -v -E -- '(-part[0-9]+|by-id/(dm|lvm|wwn|nvme-eui|ata)-)'
journalctl -fk | grep 'SerialNumber:'

Now plug the device and export information about it:

export OBV_DEVICE=/dev/disk/by-id/<your device>
export OBV_ID=OBV1

And run commands below to decrypt and mount it:

export PARTITION=/dev/disk/by-partlabel/$OBV_ID
cryptsetup open $PARTITION $OBV_ID
export MOUNTPOINT=/mnt/$OBV_ID
mkdir -p $MOUNTPOINT && mount /dev/mapper/$OBV_ID $MOUNTPOINT

Finally, make your Offline Backup Volume your working directory using the command below:

cd $MOUNTPOINT

Selecting hardware hostname

Now, you need to decide how your machine will be named. This name will be used to identify device-unique secrets and as a hostname in your OS installation.

Run the command below to learn about allowed characters, length etc of the hostname:

man hostname.7

Once you decide on something, export information about it using this command:

export TARGET_HOSTNAME=dellxps15johndoe

Generate Disk Encryption Recovery Key

If your Secure Boot with TPM setup breaks for some reason, you will be able to use unique Disk Encryption Recovery Key for each of your device, which will be stored on Offline Backup Volume or encrypted in Recovery Volume, to access your data and possibly fix your installation.

It is recommended, that your Disk Encryption Recovery Key is at least 55 characters long, consisting of lower and upper case letters and numbers, which should give you at least 256 bits of entropy using zxcvbn algorithm. This should be sufficient to provide good protection (more described in AES-256 Security section) and be short enough it can be typed by hand if needed.

To generate Disk Encryption Recovery Key, run the command below:

export LC_CTYPE=C; dd if=/dev/urandom 2>/dev/null | tr -cd '[:alnum:]' | fold -w55 | head -1 > ${TARGET_HOSTNAME}-disk-encryption-recovery-key

Generate BIOS password

This guide assumes each of your device has unique BIOS password configured. Due to difficulty of cracking the BIOS password and simplicity of simply resetting it, BIOS password can be short as it only provides the basic protection to your machine configuration.

However, it is still good to password to be unique per hardware and stored in your Daily Password Manager, so you don't have to remember it and you can access it easily.

Different BIOS-es have different password policies, so you may need to check if your BIOS is not limited to shorter maximum length than the default one used below. Usually 8 characters is acceptable, sometimes it's a limit, and sometimes it's a minimum length for "strong" password.

Command below will generate a BIOS password for your hardware and print it to the screen. If you have your Daily Password Manager available, you can save the password there.

If you only have one hardware available, write down printed BIOS password on a piece of paper, so you can use it in the next step, then destroy the piece of paper.

export BIOS_PASSWORD_LENGTH=8
export LC_CTYPE=C; dd if=/dev/urandom 2>/dev/null | tr -cd '[:alnum:]' | fold -w${BIOS_PASSWORD_LENGTH} | head -1 | tee ${TARGET_HOSTNAME}-bios-password

BIOS password security is described with more details in BIOS Password section.

Copying generated Disk Encryption Recovery Key into a Temporary Volume

If you want to make your Offline Backup Volume available again, run the following commands to identify the right device:

ls -d /dev/disk/by-id/* | grep -v -E -- '(-part[0-9]+|by-id/(dm|lvm|wwn|nvme-eui|ata)-)'
journalctl -fk | grep 'SerialNumber:'

Now plug the device and export information about it:

export TMP_DEVICE=/dev/disk/by-id/<your device>
export TMP_ID=tmp

And run commands below to mount it:

export PARTITION=${TMP_DEVICE}-part1
export MOUNTPOINT=/mnt/$TMP_ID
mkdir -p $MOUNTPOINT && mount $PARTITION $MOUNTPOINT

Then, create encrypted copy of generated Disk Encryption Recovery Key:

export GNUPGHOME=$(pwd)/gnupg-workspace
export RECIPIENT=$(gpg --with-colons --list-secret-keys $KEYID | grep '^uid' | head -n1 | cut -d: -f10)
gpg --encrypt --armor --recipient "$RECIPIENT" < ${TARGET_HOSTNAME}-disk-encryption-recovery-key > ${MOUNTPOINT}/${TARGET_HOSTNAME}-disk-encryption-recovery-key.gpg

Sync data between your copies of Offline Backup Volume

Make sure you follow Syncing data between Offline Backup Volumes to keep your Offline Backup Volumes synchronized, so you always have multiple actual copies of your secrets.

Summary

If you are following Bootstrapping process right now, head back to next section of it, which is Saving and synchronizing data between Offline Backup Volumes.

If you are going to shut down the machine now, run this commands below before unplugging the volumes to remove them securely from the system:

cd && sync -f /mnt/* && umount /mnt/*

Build Recovery Volume with new hardware profile

With new encrypted Disk Encryption Recovery Key on Temporary Volume, you should now access your Recovery Volume profile repository and add the recovery key there, together with new hardware profile for the hostname you previously selected.

New profile for your hardware can be very basic. You can install it, then figure out all required hardware-specific quirks on live system and move them back to Recovery Volume profile for reproducibility.

BIOS Setup

Next step of preparing your hardware for installing the OS is BIOS configuration.

Enter BIOS

Follow your device manual to enter the BIOS. Usually it's done by pressing one of F12, DEL or F2 keys right after turning on the device.

Setting password

Once you entered the BIOS, first thing to do is to setup new password, which we generated in the previous step. Follow your device manual to do that.

Entering Secure Boot Setup Mode

Next thing we need to do right now is to find the Secure Boot settings in your BIOS and remove pre-installed keys from there, which will make machine enter "Setup Mode", which allows rolling in your own keys.

Follow your device manual to do that.

Clearing TPM

As part of initial hardware setup, you may also want to check the TPM settings on your device and clear it, to make sure there is no pre-installed objects there.

Again, follow your device manual to do that.

Summary

With everything configured, jump to OS Installation section to install OS on your new hardware.

Day-2 Operations

This section documents various processes, which are needed in daily use, like Updating Kernel, OS Installation or handling Lost YubiKey.

Machine Maintenance

Booting up

OS Installation

Updating system configuration and rebuilding Recovery ISO image

Updating Kernel

Updating BIOS

GPG

Swapping GPG to backup YubiKey

If you try to use your backup YubiKey with GPG, you will probably and up with an error similar to the following:

Please remove the current card and insert the one with serial number:

  3131 XXX

This happens, because GPG stores the PKCS11 URLs to the private keys stored on YubiKey PIV applet, which includes the serial field, which is part of the CHUID of PIV applet, which is unique to each YubiKey can cannot be configured, so 2 YubiKeys are identical.

Due to this, when you want to switch from using one YubiKey to another for GPG, you need to remove the private key references from your GPG keyring, so GPG can re-detect them under new PKCS11 URLs.

To make your backup YubiKey usable with GPG, run the commands from Detecting key grips section to export information about your keygrips. Then, the easiest way to make a switch is to remove a private keys references from your GPG keyring using the obtained keygrips. You can do this using the command below:

rm ~/.gnupg/private-keys-v1.d/${AUTHENTICATION_KEY}.key
rm ~/.gnupg/private-keys-v1.d/${SIGNING_KEY}.key
rm ~/.gnupg/private-keys-v1.d/${ENCRYPTION_KEY}.key

Now, run gpg --card-status to trigger smartcard detection.

If you try using GPG now, it should work again.

Using GPG via PKCS11 without YubiKey

One of the downsides of using GPG sub-keys via PIV applet described in section (Optional) Adding PIV certificates into GPG Master Key is, that using GPG sub-keys without YubiKey becomes not trivial, but fortunately, not impossible either. Usage of Master Key remains unaffected and usable.

If you want to use your GPG keys using Offline Backup Volume without a YubiKey or if you lose both of your YubiKeys and you need to use GPG, this section describes how to configure it.

To satisfy your GPG Master Key, we will use SoftHSM software, which allows to create a file-based PKCS11 backend. You will initialize the SoftHSM token on your Offline Backup Volume, import your PIV certificates there and reconfigure gnupg-pkcs11-scd to use SoftHSM instead of OpenSC for performing private keys operations.

To do that, you need to access your Offline Backup Volume by booting Recovery Volume in offline mode.

With Offline Backup Volume as a working directory, first we will create SoftHSM2 configuration and a helper script, which will make it easier for you to use it in the future. Run the commands below to create them:

cp /etc/softhsm2.conf ./
cat <<EOF > ./softhsm.sh
#!/bin/bash
export SOFTHSM2_CONF=\$(pwd)/softhsm2.conf
sed -i '/tokendir/d' softhsm2.conf
SOFTHSM2_PATH=\$(pwd)/softhsm2
mkdir -p $SOFTHSM2_PATH
echo "directories.tokendir = ${SOFTHSM2_PATH}" >> $SOFTHSM2_CONF
EOF
chmod +x softhsm.sh
. ./softhsm.sh

Now, you can initialize SoftHSM using the command below:

softhsm2-util --init-token --free --label GPG

The command will ask you to configure PUK and PIN. For PUK, I suggest default Admin PIN, so 12345678, as you will never use it and for PIN, I suggest using your Master PIN, for convenience.

You can now confirm that p11tool lists your token using this command:

p11tool --list-tokens

Next step is to import the certificates into SoftHSM Token. Before we do that, we will generate Signing Key (from Secure Boot) RSA public key, as this was not done as part of bootstrapping and it's required for this step. It can be done using the command below:

openssl rsa -in secureboot/db.key -outform PEM -pubout > secureboot/db.pem

Now, import all certificates into the token:

export TOKEN=$(p11tool --list-tokens | grep SoftHSM | awk '{print $2}')
p11tool --load-privkey yubikey-piv-applet-9a-private-key.pem --write --id 01 --label "Authentication Key " $TOKEN
p11tool --load-pubkey yubikey-piv-applet-9a-public-key.pem --write --id 01 --label "Authentication Key " $TOKEN
p11tool --load-certificate yubikey-piv-applet-9a-certificate.pem --write --id 01 --label "Authentication Key " $TOKEN
p11tool --load-privkey secureboot/db.key --write --id 01 --label "Signing Key " $TOKEN
p11tool --load-pubkey secureboot/db.pem --write --id 01 --label "Signing Key " $TOKEN
p11tool --load-certificate secureboot/db.crt --write --id 01 --label "Signing Key " $TOKEN
p11tool --load-privkey yubikey-piv-applet-9d-private-key.pem --write --id 03 --label "Encryption Key " $TOKEN
p11tool --load-pubkey yubikey-piv-applet-9d-public-key.pem --write --id 03 --label "Encryption Key " $TOKEN
p11tool --load-certificate yubikey-piv-applet-9d-certificate.pem --write --id 03 --label "Encryption Key " $TOKEN

Finally, reconfigure gnupg-pkcs11-scd using the commands below to use SoftHSM:

sed -i '/provider-p1-library/d' gnupg-workspace/gnupg-pkcs11-scd.conf
echo "provider-p1-library /usr/lib/softhsm/libsofthsm2.so" >> gnupg-workspace/gnupg-pkcs11-scd.conf

And apply the configuration:

gpgconf --kill all

You will also need to follow Swapping GPG to backup YubiKey steps to remove the existing private keygrips from your GPG keyring, so GPG can detect new PKCS11 URLs pointing to SoftHSM. After that, GPG should be usable.

Remember to commit all configuration changed and newly generated files and synchronize the content to your other Offline Backup Volumes.

YubiKey Maintenance

Adding new OATH-TOTP secret

YubiKey 5 supports up to 32 OATH-TOTP secrets, which can be managed either using ykman or GUI Yubico Authenticator.

Checking alternative protocols

When you enable MFA in some service, first check if they support some standards like WebAuthn or U2F, as those are better in terms of usability, as they don't need to store any secrets on your YubiKey.

Checking for number of devices supported

If OATH-TOTP is the only MFA option available, check if your service supports adding multiple devices. As you want to make sure that your other YubiKey is a fully functional copy of your primary YubiKey, you must add the secret there as well, so there can actually be 2 different secrets on each device, with the same functionality.

Handling single-device support with redundant YubiKeys

If your service supports only one OATH-TOTP device, you can add the same secret to both of your YubiKeys. If you use Yubico Authenticator, simply add new secret to your first key, then unplug it and plug your backup key and add it again.

Adding secrets to backup YubiKey later

If it happens, that you register to some service and enable MFA there and you do not have access to your backup YubiKey, make sure to get the text version of secret used for OATH-TOTP. encrypt it using your GPG key and store it in some temporary safe location (e.g. cloud storage). When you get access to your backup YubiKey, you can manually add the new secret in Yubico Authenticator or using ykman oath command.

Provisioning new YubiKey

Unlocking locked GPG Admin PIN

Unlocking locked PIV PIN

Decommissioning old YubiKey device

If you no longer want to use your YubiKey or you want to replace it with the new model, here is recommended list of steps, which will securely wipe all your secrets from your existing YubiKey.

Using Offline Backup Volume

After initial bootstrapping, the following information is stored on your Offline Backup Volume:

  • GPG Master Key
  • Secure Boot Platform Key
  • Secure Boot Key Encryption Key

Accessing

Syncing data between Offline Backup Volumes

Signing someone else's GPG key

Storing MFA recovery tokens

When you register to new service where you use MFA and service generates recovery tokens for you, follow the following procedure:

  • Encrypt file containing recovery tokens using GPG with you as an recipient
  • Ensure you can decrypt encrypted data
  • Remove original file with plain-text recovery tokens
  • Transfer encrypted recovery tokens on temporary volume
  • Boot up Offline Recovery Volume
  • Plug in both of your Offline Backup Volume and access them
  • Store recovery tokens on both Offline Backup Volumes, either encrypted or decrypted

Extending expiry time of your GPG keys

Rotating Master Password

Incident responses

Compromised Daily Password Manager copy

Compromised Daily Password Manager content

Compromised Offline Backup Volume

Compromised machine running unauthorized code

Compromised Master Password

Compromised Master PIN

Destroyed single Offline Backup Volume

Destroyed both Offline Backup Volumes

If both of your Offline Backup Volumes gets destroyed...

Destroyed single YubiKey

If one of your YubiKeys gets stolen or destroyed...

Destroyed all YubiKeys

If all your YubiKeys gets stolen or destroyed...

Destroyed Disk encryption header

If you damage your disk encryption header, e.g. using dd writing to your encrypted partition...

Destroyed Disk encryption key

If you remove your disk encryption key from boot partition...

Destroyed TPM Secret

If your machine gets damaged OR you accidentally wipe your TPM Secret...

Revoking your GPG sub-keys

Restoring your Master Password from shards

A password gets compromised

FAQ

This section contains various question answered, which might arise while reading the guide, which has not been described earlier.

How do you protect booting Recovery OS? Does it need to be protected?

  • Recovery OS uses Secure Boot, which guarantees it's Authenticity (identity).
  • Recovery OS uses dm-verity to generate hash of root filesystem, then is then added to the kernel command line parameters with EFISTUB and signed using Secure Boot Database Key. This guarantees it's integrity.
  • Booting from USB device usually requires providing BIOS password or enabling USB booting in the BIOS, so an attacker which posses your OS Recovery Volume still won't be able to boot it on your machine.
  • What about possible credentials stored on the image? Private keys for WireGuard installation network? WiFi password? If you encrypt it, how do you decrypt?
    • There should be no plaintext secrets stored on the image except the WiFi password, which is required to enable installation over local wireless network. An attacker with physical access to your Recovery Image, may have physical access to your Ethernet network anyway, if you do not have 802.1X configured on Ethernet level or additional physical protection on the Ethernet ports, as:
      • Limiting/disabling DHCP does not offer any real security.
      • MAC address filtering does not offer good security either, as MAC addresses can be spoofed. It may also be difficult to roll out on some devices.
      • Security measures should be also used on higher levels than MAC addresses (Layer 2) anyway to provide additional security for your network, like IP-based firewalls, authentication etc.
    • This guide stores GPG-encrypted Disk Encryption Recovery Keys on Recovery Volume to make them available during installation. For secrets which are not required for unattended boot (e.g. WiFi password), you can use this method to securely store them on Recovery Volume.
    • Storing other types of plaintext secrets in Recovery Image is not supported right now with this guide. If this is what you really need, perhaps you need to replace the archiso in Recovery Image build process to perform regular Arch Linux installation, including disk encryption, similar to what we will be finally using, then configuring overlayfs combined with tmpfs or similar configuration and mount your root partition in read-only mode, to get ISO like environment which cannot be permanently modified while used.

Why does Secure Boot keys do not require respecting expiry time?

BIOS clock might be tampered (changed or reset), which would make Secure Boot to always fail, or BIOS clock can be corrected to always allow of use old Secure Boot keys. So to summarize, as clock cannot be trusted, it does not provide any real security, so it is ignored.

Why Secure Boot keys do not require periodic rotation

Even though Secure Boot use X.509 certificates, it's doing so to be able to identify certificates using UUID and to attach additional metadata to the certificates using e.g. CommonName field.

Validity time fields are not used by Secure Boot implementations.

In addition to that:

  • Both PK and KEK private keys are kept offline, so they compromise risk is greatly limited.
  • When encrypting data is transported over insecure medium (e.g. Internet, WiFi), it is recommended to periodically rotate the encryption keys, so if attacker listening and recording the transmission manages to break the encryption key, they will only be able to access part of the transported data.
  • Signature private key is stored on Hardware Security Module, which eliminates the risk of private key being stolen.

What's the point of using multiple GPG sub-keys?

Why Arch Linux?

Miscellaneous

This section contains useful information and notes not mentioned in the sections above.

Simultaneous access to YubiKey applets (especially OpenSC and GPG via pcscd daemon)

Trying out this guide in virtualized environment

You can try out most of the best practices described in this guide in virtualized environment e.g. using QEMU.

Here is the breakdown of supported features:

  • QEMU supports UEFI boot with Secure Boot via OVMF.
  • QEMU supports emulating both TPM 1.2 and TPM 2 via tpm2-tss
  • Emulation of YubiKey PIV applet is currently not supported. As an alternative, you can either:
    • Do USB pass-through of your real YubiKey device to the virtual machine and use some unused slots for testing. YubiKey has 24 slots for certificates available, this guide uses only one for Secure Boot DB key pair.
    • Use tpm2-pkcs11 as a backend for PKCS#11 engine used by sbsign.
  • Emulation of YubiKey GPG smartcard applet is currently not supported either. For testing, you can:
    • Do USB pass-through of your real YubiKey device to the virtual machine. This guide use all 3 GPG slots on YubiKey, so make sure you have a backup of your existing keys.
    • Create a virtual disk which will be acting as your portable GPG keyring. Minimal extra steps will be required to get it to work, e.g. making sure virtual disk is mounted at time when GPG is required + additional environment variable to point to the right $GNUPGHOME.

Block-based backups vs File-based backups

This guide prefers file-based backups.

Credentials which shouldn't be stored in Daily Password Manager

This section contains list of credentials which are recommended to not be stored in Daily Password Manager, as storing them there may have security implications if password manager gets compromised.

MFA Recovery Codes/Tokens

It is not recommended to keep MFA Recovery Codes in your Daily Password Manager, because if content of your Daily Password Manager leaks, it allows an attacker to bypass the requirement of your YubiKey to log in into 2FA enabled services. See Storing MFA recovery tokens section to learn how to safely store MFA Recovery Tokens.

Password Salt

If you add "salt" to passwords stored in Daily Password Manager for extra security, make sure the salt is not stored there too. This also applies when you re-use salt for some other purposes e.g. as a PIN for GPG/PIV.

API Keys for services with MFA enabled

If you protect access to certain service using MFA, but you store API keys with full privileges in your Daily Password Manager, this effectively breaks the purpose of using MFA, as obtaining the API key will effectively give an attacker full access to the service.

Private keys

Whenever possible, Hardware Security Module should be used to store your private keys instead of Daily Password Manager. As an alternative, you may also use TPM to safely store your private keys. You can generate them using Offline Recovery Volume and then transfer to any machine where you are going to need them.

Master Password and Master PIN

Considering, that this information should never be digitally saved, it is best practice to not put those in your Password Manager either.

This allows to treat those authorization factors really as "something you know".

GPG AdminPIN and PIV PUK

If one has access to your GPG AdminPIN or PIV PUK, they can try to brute-force your PIN and they may modify settings on your smartcard in case GPG AdminPIN.

Also, if your Password Manager gets compromised (e.g. you leave unlocked machine with unlocked password manager), having a PUK or AdminPIN allows an evil actor to change the PIN on your security token, which effectively gives them access to your security token.

However, if you lock your YubiKey by entering wrong PIN multiple times, the only way to unlock it is when you get access to your Offline Backup Volume, which might not be very convenient.

Disk encryption recovery keys

Similar to situation with GPG AdminPIN and PIV PUK, storing Disk encryption recovery keys in your Daily Password Manager trades security for convenience. If an attacker compromises your Daily Password Manager (e.g. you leave unlocked machine with unlocked password manager), then they can look up your disk encryption recovery keys for other devices and get access on them later.

In the real-world scenario, you will need recovery keys in the following situation:

  • Corrupted Kernel update - If something goes wrong with kernel update process, then you need a Recovery Volume and recovery key to be able to fix your existing installation.
  • Accidental BIOS update - If you update your BIOS without suspending the TPM requirement for disk encryption, then after the update your system will refuse to decrypt due to differences in PCR values in TPM.
  • Accessing data on another device - If your device gets damaged and you wish to access your data using e.g. USB adapter, then the only way to decrypt the disk is using recovery key.
  • Accessing data from insecure OS - If you wish to access your data from e.g. bootable USB device which is not Secure Boot signed for your machine, then you must use the recovery key.

NOTE: The recovery key should be at least 55 uppercase, lowercase and number characters long to provide enough entropy against brute-force attacks, which might be a trouble for type by hand sometimes, though from my experience it's doable. If not, see Old Android phone as Offline Password Manager.

TPM Passwords

Depending on TPM version on your hardware, you will either need to manage one or more passwords to configure TPM on each of your devices.

For TPM1.2, there is only single owner password which is required for all TPM operations.

With TPM 2.0, each hierarchy (group of objects) can be managed using separate password. This guide only stores few secrets in TPM.

Credentials which can be stored in Daily Password Manager

Depending on your preference, here are the credentials, which you probably can store in your Daily Password Manager and it shouldn't in a significant way degrade your level of security.

BIOS Password

If an attacker has access to your machine, they may try to replace your initramfs with malicious one. However, Secure Boot protects against such scenario, as UEFI will refuse to boot unsigned initramfs as long as Secure Boot is enabled.

If an attacker has both access to your machine and manages to bypass the BIOS password, they can disable Secure Boot and let their initramfs execute when you boot the machine again. This will allow them to hide from you, that Secure Boot has been disabled, which will allow them to obtain your Master PIN, as this is what you type into the machine when decrypting the hard drive.

To protect against compromised BIOS, tpm2-totp project is used, which will generate TOTP code using secret stored in TPM and will display it on encryption password prompt while booting. It is then user's responsibility to verify the code using other device.

The secret for tpm2-totp will be sealed against TPM PCR bank 7, which stores hash of the current state of Secure Boot configuration. If you replace any of Secure Boot keys like KEK or PK, the value will change, so you won't be able to generate valid TOTP code anymore. This means, if attacker replaces your Secure Boot keys with yours, you will notice that, as boot process won't be able to calculate valid TOTP code anymore.

Technology used by this guide

  • TPM2 TOTP - To assert the unalteredness and trustworthiness of your device.
  • Secure Boot - To prevent injecting and executing malicious code on your device at the bootloader/initramfs level.

What this guide does not protect from

This guide is designed to provide high level of security to rather low-profile people, which most likely won't be targeted by some organizations, so this section gives couple of examples in which scenarios this guide most likely won't provide sufficient security.

Operating on compromised machine

Considering the following facts:

  • Most likely Linux desktop machines are not a common target for malware.
  • Minimizing the damage from operating on a compromised machine is very inconvenient.
  • On a desktop machine, user data is most likely more valuable than system data.

This guide only slightly reduces the damage in situation, where your machine gets compromised.

Assuming that your machine gets compromised (e.g. via RCE vulnerability + privilege escalation), the following things happen:

  • All unencrypted data stored on this machine should be considered public.
  • All online services which do not use MFA should be considered compromised.
  • All online services which active sessions you have opened on your machine (browser, API keys etc), should be considered compromised.
  • Your Master Password and Master PIN are most likely compromised, as attacker can record your keystrokes.
  • Other systems where your machine has unattended access (e.g. Kubernetes clusters).

Following information remains safe:

  • Your GPG identity, as it is protected by physical touch.
  • All online services which you do not have active sessions opened.

Apartment theft

If one breaks in into your apartment, ties you up and beats you up until you reveal your credentials to them, this guide most likely won't help you.

AES-256 security

In this guide, following data volumes are protected by Master Password:

  • Daily Password Manager
  • Offline Backup Volume

Compromising any of this data volumes allows attacker to perform an offline attack on them, to eventually get access to the information inside.

The time window to successfully attack such encrypted volume is a time where information inside is still considered safe. However, you may not know that the encrypted volume has leaked or you don't know what information an attacker already have. For instance, an attacker may already have some idea what form your Master Password is, which reduces the attack window. Or an attacker may try to trick you to leak your Master Password, for example by using hardware keylogger.

If those volumes were not encrypted, attacker would have access to information inside immediately (in 0 time).

In this case, both data volumes are encrypted using AES-256. Assuming that the password has high enough entropy (AES-256 use 256 bits keys, so password should have at least 256 bits of entropy), the data should be considered safe from brute-force attacks, even if you consider growing computing power efficiency (doubles over ~3-5 years, so complexity decreases from 2^256 to 2^255 over ~3-5 years), raise of quantum computing (initial algorithms can reduce required computation complexity by half, so from 2^256 to 2^128) etc.

In case of KeepassXC, which use HMAC-SHA1 slot in YubiKey, it generates additional ~150-170 bits of entropy added to your password, which is protected by hardware key.

During time window, where attacker can try to brute-force the password, it is recommended to rotate all your secrets stored on the volumes listed above, so when an attacker finally cracks them, they will no longer be useful. However, as time window size in such circumstances is enormously long, not rotating your secrets should not have an impact on your security.

Old Android phone as Offline Password Manager

Credentials which shouldn't be stored in Daily Password Manager section highlighted which credentials shouldn't be stored in your Daily Password Manager from safety reasons. The only way to access those secrets is to boot Offline Recovery Volume, plug your Offline Backup Volume and get password from there. This might be not very convenient if you have limited number of devices available.

As an alternative for Offline Recovery Volume, you may consider using old Android phone without network configured and without SIM card as a simple offline password manager, which would have additional copy of your passwords you keep offline, protected by the same Master Password which is used to protect Offline Backup Volume. This should make it more convenient to access passwords stored there when needed.

You could even try and modify your Android phone to act as a USB keyboard or use device like InputStick to avoid typing your password yourself.

Glossary

Password manager, which you use for every day logging in on your end devices like laptop or smartphone. This guide use KeePassXC, as it supports YubiKeys as a factor for unlocking the password database.

Password manager, stored on encrypted volume, which you only access from Live USB, which has no internet connection. It holds all critical credentials, which are not required during daily operation like AdminPIN, PUK, YubiKey PIV ManagementKey, Secure Boot Platform Key etc.

Backup rule saying you should always have at least 3 copies of your data, store 2 backup copies on different devices or storage media and keep at least 1 backup copy off-site.

Snowflake is servers/machines, which configuration has drifted from original or automated configuration. Such drift is not desired, as configuration of such machines are hard to reproduce, which makes the recovery process harder when machine breaks.

Recovery volume booted without network support.

GPG Master Key is a main source of trust for your GPG identity. This key has only signing capabilities, it is used only to sign and revoke your GPG sub-keys and it should only be stored on offline volumes to minimize the risk of leaking.

GPG keys which are used on a daily basis. Usually it is a signing key, encryption key and authentication key. All 3 keys should be stored on Hardware security device.

Master password is the only password you must remember when following this guide. This password is used for unlocking your daily password manager and to decrypt Offline Backup Volumes. This password should be long and complex to make sure it cannot be brute-forced.

Offline Backup Volume is a local storage device (e.g. Pendrive), encrypted using Master Password, which contains all secrets like GPG Master Key, which must be kept secure at all cost. Accessing it's data should only be performed from Offline Recovery Volume with no network access.

Top level X.509 certificate with RSA key-pair used in Secure Boot process. "Platform" in "Platform Key" refers to a Computing platform, which can be an actual piece of hardware you execute code on, but can also be a visualized environment or a web browser.

Owning Platform Private Key allows you to proof to the UEFI that you are the physical owner of the hardware.

This trust is established by requiring physical presence to the unit (platform) to "take ownership", so to add initial PK.

If the attacker posses the Platform Private Key and access to the hardware, it does not give them direct access to execute code on the platform, as one needs to execute code, which will add a malicious public key to the signature database.

Usually this key is populated with device manufacturer (OEM) key and it signs KEK keys, including Microsoft CA.

This guide assumes that you are platform owner, so as part of Bootstrapping process you will generate and roll your own Platform Key, which will later be stored on Offline Backup Volume, as it is not required for daily operation.

Removable device, which contains your personalized Arch Linux installer.

Resources

Here is list of online resources, which were main source of building this guide: