This document covers the procedure for configuring a YubiKey as a GPG smartcard for SSH authentication. This is a decent solution for two-factor authentication: in this case a hardware token (the Yubikey), and the passphrase required to unlock it for each session.
While this document is written for the YubiKey NEO, these instructions work for Yubikey4 as well (either standard or Nano sizes). Unless you plan to use the NFC functionality of the Yubikey NEO, it is recommended that you get Yubikey 4, which supports 4096-bit PGP keys. The NEO is limited to 2048-bit keys.
Examples below are using a Fedora 23 x86_64 and Ubuntu 15.04 x86_64 fresh install. There are other tutorials for other operating systems and keys available online. See the CREDITS section below for alternate tutorials, examples, etc.
To configure authentication using the previously generated GnuPG key, the GNOME-Shell needs some adjustements. With help from several resources, configure the system to allow gpg-agent
to take over SSH authentication.
Certain software must be installed, including utilities for the YubiKey libyubikey-devel
(libyubikey-dev
on Ubuntu), gnupg2
(which is probably already installed), gnupg2-smime
(gpgsm
on Ubuntu)i, pcsc-lite-ccid
, and pcsc-lite
(pcscd
and libpcsclite1
on Ubuntu).
Fedora:
$ sudo dnf install ykpers-devel libyubikey-devel libusb-devel \ autoconf gnupg gnupg2-smime pcsc-lite pcsc-lite-ccid
Ubuntu:
$ sudo apt-get install gnupg-agent gnupg2 pinentry-gtk2 scdaemon \ libccid pcscd libpcsclite1 gpgsm yubikey-personalization \ libyubikey-dev libykpers-1-dev
Optional: Install the Yubikey NEO Manager GUI. If running Ubuntu, you can install the yubikey neo manager and other yubikey software from the Yubico PPA.
This will enable the smartcard portion of your yubi key neo:
$ ykpersonalize -m82
If you have a dev key, Reboot your yubikey (remove and reinsert) so that ykneomgr works.
Configure GPG to use its agent (only for smartcard):
$ echo "use-agent" >> ~/.gnupg/gpg.conf
Enable ssh-agent drop in replacement support for gpg-agent:
$ echo "enable-ssh-support" >> ~/.gnupg/gpg-agent.conf
Allow admin actions on your YubiKey (if your gnupg version is < 2.0.11):
$ echo "allow-admin" >> ~/.gnupg/scdaemon.conf
Open Startup Applications
Uncheck "GPG Password Agent" and "SSH Key Agent"
Edit /usr/share/upstart/sessions/gpg-agent.conf
so that the pre-start script contains the following:
eval "$(gpg-agent --daemon --enable-ssh-support --sh)" >/dev/null initctl set-env --global GPG_AGENT_INFO=$GPG_AGENT_INFO initctl set-env --global SSH_AUTH_SOCK=$SSH_AUTH_SOCK initctl set-env --global SSH_AGENT_PID=$SSH_AGENT_PID
Add the following lines to the post-stop script section:
initctl unset-env --global SSH_AUTH_SOCK initctl unset-env --global SSH_AGENT_PID
Comment out the use-ssh-agent
line in /etc/X11/XSession.options
file.
Disable the other system gpg-agent:
mv /etc/X11/Xsession.d/90gpg-agent ~/bak/90gpg-agent
Note: We could have used the Xsession gpg-agent and trashed the upstart one, but there is an open bug report for 90gpg-agent. Also, the upstart script has the capability of exporting the environment variables globally with initctl set-env --global.
If running gnome, you have to prevent gnome-keyring from autostarting its ssh-agent implementation, because we will use GPG's instead:
- sed -e '$aX-GNOME-Autostart-enabled=false' -e '$aHidden=true'
- /etc/xdg/autostart/gnome-keyring-ssh.desktop > $HOME/.config/autostart/gnome-keyring-ssh.desktop
On GNOME 3.16 or earlier, do the same for the broken gpg-agent implementation, which does not support smartcards:
- sed -e '$aX-GNOME-Autostart-enabled=false' -e '$aHidden=true'
- /etc/xdg/autostart/gnome-keyring-gpg.desktop > $HOME/.config/autostart/gnome-keyring-gpg.desktop
Next, place the following in ~/.bashrc
to ensure gpg-agent starts with --enable-ssh-support
if [ ! -f /run/user/$(id -u)/gpg-agent.env ]; then killall gpg-agent; eval $(gpg-agent --daemon --enable-ssh-support > /run/user/$(id -u)/gpg-agent.env); fi . /run/user/$(id -u)/gpg-agent.env
Now go to next step (Reload GNOME-Shell) :)
Otherwise, there is another option:
A rather tricky part of this configuration is to have a simple wrapper script, called gpg-agent-wrapper. This script is used with thanks from Diego E. Pettenò:
wget -O ~/.gnupg/gpg-agent-wrapper https://github.com/lfit/ssh-gpg-smartcard-config/raw/master/gpg-agent-wrapper && chmod +x ~/.gnupg/gpg-agent-wrapper
NOTE: The above code has been altered to allow the .gpg-agent-info
to run after SSH_AUTH_SOCK. Please see the CREDITS section below for details.
The above gpg-agent-wrapper script is invoked using X and bash (or favorite shell). Please create the following files as below.
The X session:
$ cat /etc/X11/xinit/xinitrc.d/01-xsession [ -f ${HOME}/.xsession ] && source ${HOME}/.xsession $ ls -l /etc/X11/xinit/xinitrc.d/01-xsession -rwxr-xr-x. 1 root root 53 Nov 23 10:54 /etc/X11/xinit/xinitrc.d/01-xsession $ cat ~/.xsession source ${HOME}/.gnupg/gpg-agent-wrapper
The shell rc file:
$ cat ~/.bashrc # .bashrc # Source global definitions if [ -f /etc/bashrc ]; then . /etc/bashrc fi ..snip.. # ssh authentication component source ${HOME}/.gnupg/gpg-agent-wrapper ..snip..
Rebooting the machine works the best.
Included with the gemalto token and GnuPG Smartcard version 2 should be a document describing the default PIN values. There is a regular PIN, which is used to unlock the token for Signing, Encryption or Authentication. Additionally, there is an admin PIN, which is used to reset the PIN and/or the Reset Code for the key itself.
default pins are 123456 and 12345678 respectivly
$ gpg2 --card-edit ..snip.. gpg/card> admin Admin commands are allowed gpg/card> passwd gpg: OpenPGP card no. D27600012401020000050000158A0000 detected 1 - change PIN 2 - unblock PIN 3 - change Admin PIN 4 - set the Reset Code Q - quit Your selection? 3
Enter the Current Admin PIN
Then enter the New Admin PIN twice
1 - change PIN 2 - unblock PIN 3 - change Admin PIN 4 - set the Reset Code Q - quit Your selection? 1
Enter the Current PIN
Then enter the New PIN twice
NOTE: If the Admin PIN has not been entered, it may be required before changes are applied.
There are several ways to generate an SSH Key using GnuPG. A common way is to link the new authentication key to an already existing key:
$ gpg2 --edit-key 8A8F1D53 gpg (GnuPG) 1.4.12; Copyright (C) 2012 Free Software Foundation, Inc. This is free software: you are free to change and redistribute it. There is NO WARRANTY, to the extent permitted by law. Secret key is available. pub 3072R/8A8F1D53 created: 2012-10-06 expires: never usage: SC trust: ultimate validity: ultimate sub 3072R/2F15E06B created: 2012-11-23 expires: 2022-11-21 usage: S sub 3072R/EB8B4EBD created: 2012-11-24 expires: 2022-11-22 usage: E sub 3072R/6BB325E9 created: 2012-11-24 expires: 2022-11-22 usage: A [ultimate] (1). Clint Savage <[email protected]> [ultimate] (2) Clint Savage <[email protected]> [ultimate] (3) Clint Savage <[email protected]> gpg>
Once in the edit-key
dialog, create a key on the card:
gpg> addcardkey Signature key ....: 91BC 60CC B9EC 8E73 923A FC6D 58CD 88A6 2F15 E06B Encryption key....: 0CC3 DC3E 0D17 6111 A62B F656 63C6 4DA9 EB8B 4EBD Authentication key: 9EBF A9FE 8AE1 0FEB 1699 CE9A 779F 43D5 EC6F CC13 Please select the type of key to generate: (1) Signature key (2) Encryption key (3) Authentication key Your selection? 3 IT WILL PROMPT YOU TO ENTER THE ADMIN PIN, AND THEN THE REGULAR PIN. Don't fat finger this part! gpg: WARNING: such a key has already been stored on the card! Replace existing key? (y/N) y What keysize do you want for the Authentication key? (3072) Key is protected. You need a passphrase to unlock the secret key for user: "Clint Savage <[email protected]>" 3072-bit RSA key, ID 8A8F1D53, created 2012-10-06 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) 10y Key expires at Mon 21 Nov 2022 05:29:00 PM MST Is this correct? (y/N) y Really create? (y/N) y gpg: Note that the key does not use the suggested creation date pub 3072R/8A8F1D53 created: 2012-10-06 expires: never usage: SC trust: ultimate validity: ultimate sub 3072R/2F15E06B created: 2012-11-23 expires: 2022-11-21 usage: S sub 3072R/EB8B4EBD created: 2012-11-24 expires: 2022-11-22 usage: E sub 3072R/6BB325E9 created: 2012-11-24 expires: 2022-11-22 usage: A [ultimate] (1). Clint Savage <[email protected]> [ultimate] (2) Clint Savage <[email protected]> [ultimate] (3) Clint Savage <[email protected]>
Upon completion of the key, be sure to save the record to the card and gpg key:
gpg> save $
Assuming everything above is configured correctly, a simple test is performed with the SmartCard inserted:
$ ssh-add -L ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABgQDL/XmU......BL0luE= cardno:00050000158A
If you find that pinentry continues to ask for your PIN and you were expecting your PIN to be cached, you may need to toggle the "forcesig" flag on your YubiKey:
$ gpg2 --card-edit gpg/card> admin gpg/card> list ... Signature PIN ....: forced ... gpg/card> forcesig gpg/card> list ... Signature PIN ....: not forced
From https://dilfridge.blogspot.com.au/2013/05/openpgp-smartcards-and-gentoo-part-2.html:
"forcesig" toggles a flag inside the card that has been introduced because of German legislative requirements for some smartcard applications. Normally, once you have inserted the card into the reader, you enter the PIN once for unlocking e.g. the encryption or the signature key, and then the key remains open for the moment. If the signature PIN is "forced", you will have to reenter the PIN again each time you want to make a signature.
Thanks to askb for pointing out this flag to help solve a mutt + gpg + pin cache issue.
The github repository contains all the files to make the changes above. Please feel free to read through them.
A special thanks to the following people and/or links.