This document describes how to set up a YubiKey.
- Generate gpg keys
- Backup gpg keys
- SSH public key authentication
- GitHub signing commits
- Two-factor authentication (2FA) (for Google, Amazon, Twitter, etc. accounts)
- FIDO U2F (Universal 2nd Factor) (so called 2FA using
Security Key
) - Time-based one-time password (TOTP) (2FA using authenticator apps like Google Authenticator, Authy, etc.)
- FIDO U2F (Universal 2nd Factor) (so called 2FA using
- Passkey authentication (WIP)
- Nostr private key (WIP)
- https://support.yubico.com/hc/en-us/articles/360013790259-Using-Your-YubiKey-with-OpenPGP
- https://github.com/drduh/YubiKey-Guide
- https://musigma.blog/2021/05/09/gpg-ssh-ed25519.html
- https://keens.github.io/blog/2021/03/23/yubikeywotsukau_openpghen/
- https://qiita.com/shun-shobon/items/96f08aa09a30c26a55b5
- YubiKey 5 Series which supports OpenPGP.
- I bought a YubiKey 5 NFC.
- USB drive or SD card for key backup.
Note: In my opinion, you don't need to buy 2 YubiKeys if you back up your keys carefully. If you lose a YubiKey, you can restore your keys from the backup. Also, it might be better to revoke the old keys and generate new ones from scratch, as lost keys could be used for malicious purposes.
- For macOS,
brew install gnupg
brew install ykman
- This commands will reset the YubiKey OpenPGP data (not FIDO2, OTP, PIV, etc.). See details here.
- ARE YOU SURE YOU WANT TO RESET?
ykman openpgp reset
rm -rf ~/.gnupg # if you already have gpg keys, backup them somewhere
- Default PINs are here. We will change them later.
PIN: 123456
Reset code: NOT SET
Admin PIN: 12345678
- Choose the key algorithm, RSA or ECC. I chose ECC.
- If you use RSA, choose 4096 bits. See details here.
- If you use ECC, I recommend to use Curve 25519. See details here. I chose Curve 25519.
- There are 4 types of keys: S, C, E, and A. S is for signing, C is for certification, E is for encryption, and A is for authentication.
gpg
default procedure will generate a master key for S and C, and a subkey for E.- We will create a subkey for A later.
Note: Email address must be set to the same email address as your GitHub account. Otherwise, you cannot sign commits with GPG. In my case,
[email protected]
.
gpg --expert --full-gen-key
Please select what kind of key you want:
(1) RSA and RSA
(2) DSA and Elgamal
(3) DSA (sign only)
(4) RSA (sign only)
(7) DSA (set your own capabilities)
(8) RSA (set your own capabilities)
(9) ECC (sign and encrypt) *default*
(10) ECC (sign only)
(11) ECC (set your own capabilities)
(13) Existing key
(14) Existing key from card
Your selection? 9
Please select which elliptic curve you want:
(1) Curve 25519 *default*
(2) Curve 448
(3) NIST P-256
(4) NIST P-384
(5) NIST P-521
(6) Brainpool P-256
(7) Brainpool P-384
(8) Brainpool P-512
(9) secp256k1
Your selection? 1
Please specify how long the key should be valid.
0 = key does not expire
<n> = key expires in n days
<n>w = key expires in n weeks
<n>m = key expires in n months
<n>y = key expires in n years
Key is valid for? (0) 0
Key does not expire at all
Is this correct? (y/N) y
GnuPG needs to construct a user ID to identify your key.
Real name: Susumu OTA
Email address: [email protected]
Comment:
You selected this USER-ID:
"Susumu OTA <[email protected]>"
Change (N)ame, (C)omment, (E)mail or (O)kay/(Q)uit? o
# input passphrase
pub ed25519 2023-03-17 [SC]
KEY_FINGERPRINT_HERE
uid Susumu OTA <[email protected]>
sub cv25519 2023-03-17 [E]
- Now, we have a master key for S and C with Curve 25519 and a subkey for E with Curve 25519.
- Save the key ID. e.g.
3AA5C34371567BD2
.
gpg --list-secret-keys --keyid-format=long
export GPG_KEY_ID="3AA5C34371567BD2" # replace with your key ID
- Right now, we have the keys for S, C and E.
- We need to add a subkey for A.
gpg --expert --edit-key $GPG_KEY_ID
# add a subkey for A
gpg> addkey
Please select what kind of key you want:
(3) DSA (sign only)
(4) RSA (sign only)
(5) Elgamal (encrypt only)
(6) RSA (encrypt only)
(7) DSA (set your own capabilities)
(8) RSA (set your own capabilities)
(10) ECC (sign only)
(11) ECC (set your own capabilities)
(12) ECC (encrypt only)
(13) Existing key
(14) Existing key from card
Your selection? 11
Possible actions for this ECC key: Sign Authenticate
Current allowed actions: Sign
(S) Toggle the sign capability
(A) Toggle the authenticate capability
(Q) Finished
Your selection? a
Possible actions for this ECC key: Sign Authenticate
Current allowed actions: Sign Authenticate
(S) Toggle the sign capability
(A) Toggle the authenticate capability
(Q) Finished
Your selection? s
Possible actions for this ECC key: Sign Authenticate
Current allowed actions: Authenticate
(S) Toggle the sign capability
(A) Toggle the authenticate capability
(Q) Finished
Your selection? q
Please select which elliptic curve you want:
(1) Curve 25519 *default*
(2) Curve 448
(3) NIST P-256
(4) NIST P-384
(5) NIST P-521
(6) Brainpool P-256
(7) Brainpool P-384
(8) Brainpool P-512
(9) secp256k1
Your selection? 1
Please specify how long the key should be valid.
0 = key does not expire
<n> = key expires in n days
<n>w = key expires in n weeks
<n>m = key expires in n months
<n>y = key expires in n years
Key is valid for? (0) 0
Key does not expire at all
Is this correct? (y/N) y
Really create? (y/N) y
# input passphrase
gpg> save
- Confirm the keys.
gpg -k
-----------------------------
pub ed25519 2023-03-17 [SC]
KEY_FINGERPRINT_HERE
uid [ultimate] Susumu OTA <[email protected]>
sub cv25519 2023-03-17 [E]
sub ed25519 2023-03-17 [A]
- Verify the keys.
brew install hopenpgp-tools
rehash
gpg --export $GPG_KEY_ID | hokey lint
- Set the expiration time for the subkeys to 1 year. Renew them every year.
gpg --expert --edit-key $GPG_KEY_ID
gpg> key 1 # select E key
gpg> expire
Changing expiration time for a subkey.
Please specify how long the key should be valid.
0 = key does not expire
<n> = key expires in n days
<n>w = key expires in n weeks
<n>m = key expires in n months
<n>y = key expires in n years
Key is valid for? (0) 1y
Key expires at Sun Mar 17 00:41:24 2024 JST
Is this correct? (y/N) y
# input passphrase
gpg> key 1 # unselect E key
gpg> key 2 # select A key
gpg> expire
Changing expiration time for a subkey.
Please specify how long the key should be valid.
0 = key does not expire
<n> = key expires in n days
<n>w = key expires in n weeks
<n>m = key expires in n months
<n>y = key expires in n years
Key is valid for? (0) 1y
Key expires at Sun Mar 17 00:41:45 2024 JST
Is this correct? (y/N) y
# input passphrase
gpg> save
- Confirm the keys.
gpg -k
-----------------------------
pub ed25519 2023-03-17 [SC]
KEY_FINGERPRINT_HERE
uid [ultimate] Susumu OTA <[email protected]>
sub cv25519 2023-03-17 [E] [expires: 2024-03-16]
sub ed25519 2023-03-17 [A] [expires: 2024-03-16]
- Follow these instructions to export your secret keys.
- Prepare a USB drive or a SD card or something to backup the keys.
- Create a tmpfs for temporary workspace on local machine.
mkdir -p workspace
sudo mount_tmpfs workspace # for macOS
# sudo mount -t tmpfs tmpfs workspace # for Linux
cd workspace
- Export the keys.
gpg --armor --export-secret-keys $GPG_KEY_ID > gpg_secret_keys.asc
gpg --armor --export-secret-subkeys $GPG_KEY_ID > gpg_secret_subkeys.asc
gpg --armor --export $GPG_KEY_ID > gpg_public_keys.asc
- Confirm the exported keys whether they can be imported.
gpg --import gpg_secret_keys.asc
gpg --import gpg_secret_subkeys.asc
gpg --import gpg_public_keys.asc
gpg -k
- Create the revocation certificate.
gpg --output gpg_revoke.asc --gen-revoke $GPG_KEY_ID
Create a revocation certificate for this key? (y/N) y
Please select the reason for the revocation:
0 = No reason specified
1 = Key has been compromised
2 = Key is superseded
3 = Key is no longer used
Q = Cancel
(Probably you want to select 1 here)
Your decision? 1
Enter an optional description; end it with an empty line:
>
Reason for revocation: Key has been compromised
(No description given)
Is this okay? (y/N) y
- Copy the keys to the USB drive or the SD card.
mkdir -p /Volumes/GPGBAK1/20230317-01
mkdir -p /Volumes/GPGBAK1/20230317-02
cp -p gpg_* /Volumes/GPGBAK1/20230317-01
cp -p gpg_* /Volumes/GPGBAK1/20230317-02
mkdir -p /Volumes/GPGBAK2/20230317-01
mkdir -p /Volumes/GPGBAK2/20230317-02
cp -p gpg_* /Volumes/GPGBAK2/20230317-01
cp -p gpg_* /Volumes/GPGBAK2/20230317-02
gpg --card-edit
gpg/card> admin
- Enables YubiKey to store the hash of PIN.
gpg/card> kdf-setup
gpg/card> list # KDF setting ......: on
- Change PIN.
gpg/card> passwd
1 - change PIN
2 - unblock PIN
3 - change Admin PIN
4 - set the Reset Code
Q - quit
# select 1, 3, 4 then q
gpg/card> quit # needs to quit to save PIN
gpg --card-edit
gpg/card> admin
gpg/card> name
gpg/card> salutation
gpg/card> lang
gpg/card> login
# gpg/card> url # setup this later
gpg/card> list
gpg/card> quit
Note: Transferring keys to YubiKey using keytocard is a destructive, one-way operation only. Make sure you have a backup of your keys before proceeding.
gpg --edit-key $GPG_KEY_ID
gpg> keytocard
Really move the primary key? (y/N) y
Please select where to store the key:
(1) Signature key
(3) Authentication key
Your selection? 1
# input passphrase and PIN
gpg> key 1 # select E key
gpg> keytocard
Please select where to store the key:
(2) Encryption key
Your selection? 2
# input passphrase and PIN
gpg> key 1 # unselect E key
gpg> key 2 # select A key
gpg> keytocard
Please select where to store the key:
(3) Authentication key
Your selection? 3
gpg> save
- Confirm the keys.
gpg --card-edit
# confirm these lines
Key attributes ...: ed25519 cv25519 ed25519
Signature key ....: xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx
created ....: 2023-03-17 15:30:34
Encryption key....: xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx
created ....: 2023-03-17 15:30:34
Authentication key: xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx
created ....: 2023-03-17 15:35:09
gpg/card> quit
- Plug the YubiKey. Test sign, encrypt, verify and decrypt.
echo test > test.txt
gpg -sea --default-recipient-self test.txt # encrypt and sign
gpg -d test.txt.asc # verify and decrypt
- Unplug the YubiKey. It should fail to decrypt.
gpg -d test.txt.asc
- Plug the YubiKey again. It should succeed to decrypt.
gpg -d test.txt.asc
- If you confirmed the keys are stored in the YubiKey, you can delete the temporary workspace on local machine.
cd ..
sudo umount workspace
- Follow the instructions in the following link.
- Create
gpg-agent.conf
.
brew install pinentry-mac # for macOS
rehash
which pinentry-mac # /usr/local/bin/pinentry-mac
cd ~/.gnupg
wget https://raw.githubusercontent.com/drduh/config/master/gpg-agent.conf
# edit gpg-agent.conf, comment out default pinentry-program and enable pinentry-mac
# pinentry-program /usr/local/bin/pinentry-mac
~/.gnupg/gpg-agent.conf
# https://github.com/drduh/config/blob/master/gpg-agent.conf
# https://www.gnupg.org/documentation/manuals/gnupg/Agent-Options.html
enable-ssh-support
ttyname $GPG_TTY
default-cache-ttl 60
max-cache-ttl 120
#pinentry-program /usr/bin/pinentry-curses
#pinentry-program /usr/bin/pinentry-tty
#pinentry-program /usr/bin/pinentry-gtk-2
#pinentry-program /usr/bin/pinentry-x11
#pinentry-program /usr/bin/pinentry-qt
#pinentry-program /usr/local/bin/pinentry-curses
pinentry-program /usr/local/bin/pinentry-mac
- Add the following to
~/.zshrc
.
export GPG_TTY="$(tty)"
export SSH_AUTH_SOCK=$(gpgconf --list-dirs agent-ssh-socket)
gpgconf --launch gpg-agent
- Save the public key.
ssh-add -L | grep "cardno" > ~/.ssh/id_ed25519_yubikey.pub
- Add the public key to the server's
~/.ssh/authroized_keys
.
ssh -p 10022 [email protected]
emacs ~/.ssh/authorized_keys
# add the content of id_ed25519_yubikey.pub
exit
- Add the following to
~/.ssh/config
.
cat << EOF >> ~/.ssh/config
Host pi3.local
IdentitiesOnly yes
IdentityFile ~/.ssh/id_ed25519_yubikey.pub
EOF
- Delete
[pi3.local]
lines in~/.ssh/know_hosts
.
emacs ~/.ssh/known_hosts
# delete all of the [pi3.local] lines
- Try to login again.
ssh -p 10022 [email protected] -vvv
-
Follow these instructions.
-
Backup the existing
~/.gitconfig
.
cp -p ~/.gitconfig ~/.gitconfig.bak
- Show public key and paste it to GitHub.
gpg --armor --export $GPG_KEY_ID
- Unset
gpg.format
if it is set.
git config --global --unset gpg.format
cat ~/.gitconfig
- Copy the long form of the GPG key ID. e.g.
3AA5C34371567BD2
gpg --list-secret-keys --keyid-format=long # copy your key ID
export GPG_KEY_ID="3AA5C34371567BD2" # replace with your key ID
git config --global user.signingkey $GPG_KEY_ID
git config --global commit.gpgsign true # add this to omit -S option
cat ~/.gitconfig
diff -u ~/.gitconfig.bak ~/.gitconfig
- Commit and push something.
git add README.md
git commit -m "test" # or git commit -S -m "test" if you didn't set commit.gpgsign true
git push origin main
-
See the commit log on GitHub. You will see green
Verified
badge and click it to confirmGPG key ID
is same as$GPG_KEY_ID
. -
You can also try to unplug and plug the YubiKey to test
pinentry
.- You will see
pinentry
dialog whengit commit
andgit push
.
- You will see
- FIDO U2F (2FA using
Security Key
on Google, Twitter etc.) does not store any credential information on YubiKey. Just use a device-specific key which was generated on-chip at the time of manufacturing. - You don't need to manage or back up anything on your YubiKey. Simply store safely the recovery code given by the service provider.
- You should encrypt the recovery code using
gpg
and copy it to the safe place (e.g. USB drive or SD card).
gpg -sea --default-recipient-self recovery.txt # encrypt and sign
gpg -d recovery.txt.asc # verify and decrypt
- TOTP (2FA using authenticator apps like
Google Authenticator
) stores the secret key, so that you need to back up it. - You can download an app Yubico Authenticator, and it can store 32 secret keys on the YubiKey.
- When you see QR code on the service provider's website, you can scan it by Yubico Authenticator app.
- Also, you can scan QR code by Google Authenticator, Authy or any other TOTP app for backup purpose.
- At the same time, there MUST be an option to show the SECRET KEY instead of QR code. e.g. you will find a link like
Click here if you can't scan the QR code
. - I recommend to save the secret key manually.
- You can input the secret key manually on Yubico Authenticator app or
ykman
command (see below). - The secret key is different from recovery code or backup code which are service provider specific.
- If you save the secret key, you can change TOTP apps or mobile devices anytime.
- If you save the secret key, you can completely restore the 2FA apps without recovery code or backup code.
- You should encrypt the secret key, recovery code and backup code using
gpg
and save them to the safe place (e.g. USB drive or SD card).
gpg -sea --default-recipient-self secret_key.txt # encrypt and sign
gpg -d secret_key.txt.asc # verify and decrypt
- You can add the secret key manually by
ykman oath accounts add
. - Also, you can see the account list by
ykman oath accounts list
.
ykman oath accounts list
# Amazon:[email protected]
# Google:[email protected]
# Twitter:foo
# Twitter:bar
- And generate codes by
ykman oath accounts code
.
ykman oath accounts code
# Amazon:[email protected] 111111
# Google:[email protected] 222222
# Twitter:foo 333333
# Twitter:bar 444444
That's all.
MIT
S. Ota