Skip to content

number5/yubikey

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

YubiKey at Datadog

Table of contents

Summary

GPG is useful for authenticating yourself over SSH and / or GPG-signing your git commits / tags. However, without hardware like the YubiKey, you would typically keep your GPG private subkeys in "plain view" on your machine, even if encrypted. That is, attackers who personally target [1, 2, 3, 4] you can compromise your machine can exfiltrate your (encrypted) private key, and your passphrase, in order to pretend to be you.

Instead, this setup lets you store your private subkeys on your YubiKey. Actually, it gives you much stronger guarantees: you cannot authenticate over SSH and / or sign GPG commits / tags without: (1) your YubiKey plugged in and operational, (2) your YubiKey PIN, and (3) touching your YubiKey. So, even if there is malware trying to get you to sign, encrypt, or authenticate something, you would almost certainly notice, because your YubiKey will flash, asking for your attention. (There is the "time of check to time of use" issue, but that is out of our scope.)

Estimated burden and prerequisites

About 2-3 hours. 15 minutes could save you 15% or more on cybersecurity insurance.

You will need macOS, Homebrew, a password manager, and a YubiKey 5.

U2F

STRONGLY recommended: configure U2F for GitHub and Google.

GPG

Please read and follow all of the instructions carefully.

$ ./gpg.sh

(Protip: set TEMPDIR=1 when preparing YubiKey for someone else to avoid polluting your default GPG homedir.)

git

STRONGLY RECOMMENDED: signing all your git commits across all repositories.

You must have first set up GPG. Then:

$ ./git.sh

SSH

NOT recommended unless you plan to use your GPG authentication subkey as your only SSH authentication key.

You must have first set up GPG. Then:

$ ./ssh.sh

Keybase

Optional: verify public key on Keybase. You can now do this using the command-line option, with only curl and gpg, and without installing any Keybase app, or uploading an encrypted copy of your private key. For example, see this profile.

If you have the Keybase application installed, you can import your YubiKey public key like this:

$ keybase pgp select

# If you already have a primary Keybase public key, use the --multi flag to import another
$ keybase pgp select --multi

See keybase pgp help select for more detail.

VMware Fusion

Optional: using YubiKey inside GNU/Linux running on VMware Fusion.

  1. Shut down your VM, find its .vmx file, edit the file to the add the following line, and then reboot it: usb.generic.allowHID = "TRUE"

  2. Connect your YubiKey to the VM once you have booted and logged in.

  3. Install libraries for smart card:

    1. Ubuntu 17.10: apt install scdaemon

    2. Fedora 27: dnf install pcsc-lite pcsc-lite-ccid

  4. Import your public key (see Step 13).

  5. Set ultimate trust for your key (see Step 20).

  6. Configure GPG (see Step 22).

  7. Test the keys (see Step 23). On Fedora, make sure to replace gpg with gpg2.

  8. Use the absolutely terrible kludge in Table 5 to make SSH work.

  9. Spawn a new shell, and test GitHub SSH (see Step 26).

  10. Test Git signing (see Step 28). On Fedora, make sure to replace gpg with gpg2: git config --global gpg.program gpg2

    # gpg-ssh hack
    gpg-connect-agent killagent /bye
    eval $(gpg-agent --daemon --enable-ssh-support --sh)
    ssh-add -l

Table 5: Add these lines to ~/.bashrc.

Docker Content Trust

Optional: using YubiKey to store the root role key for Docker Notary.

  1. Assumption: you are running all of the following under Fedora 27.

  2. Install prerequisites: dnf install golang yubico-piv-tool

  3. Set GOPATH (make sure to update PATH too), and spawn a new bash shell.

  4. Check out the Notary source code: go get github.com/theupdateframework/notary

  5. Patch source code to point to correct location of shared library on Fedora.

    1. cd ~/go/src/go get github.com/theupdateframework/notary

    2. git pull https://github.com/trishankatdatadog/notary.git trishank_kuppusamy/fedora-pkcs11

  6. Build and install the Notary client: go install -tags pkcs11 github.com/theupdateframework/notary/cmd/notary

  7. Add the lines in Table 6 to your bash profile, and spawn a new shell.

  8. Try listing keys (there should be no signing keys as yet):

    1. dockernotary key list -D

    2. If you see the line "DEBU[0000] Initialized PKCS11 library /usr/lib64/libykcs11.so.1 and started HSM session", then we are in business.

    3. Otherwise, if you see the line "DEBU[0000] No yubikey found, using alternative key storage: found library /usr/lib64/libykcs11.so.1, but initialize error pkcs11: 0x6: CKR_FUNCTION_FAILED", then you probably need to gpgconf --kill scdaemon (see this issue), and try again.

  9. Generate the root role key (can be reused across multiple Docker repositories), and export it to both YubiKey, and keep a copy on disk:

    1. Choose a strong passphrase.

    2. dockernotary key generate -D

    3. Commit passphrase to memory and / or offline storage.

    4. Try listing keys again, you should now see a copy of the same private key in two places (disk, and YubiKey).

    5. Backup private key in ~/.docker/trust/private/KEYID.key unto offline, encrypted, long-term storage.

    6. Securely delete this private key on disk.

    7. Now if you list the keys again, you should see the private key only on YubiKey.

  10. Link the yubikey library so that the prebuilt docker client can find it: sudo ln -s /usr/lib64/libykcs11.so.1 /usr/local/lib/libykcs11.so

  11. Later, when you want Docker to use the root role key on your YubiKey:

    1. When you push an image, you may have to kill scdaemon (in a separate shell) right after Docker pushes, but right before Docker uses the root role key on your YubiKey, and generates a new targets key for the repository.

    2. Use docker -D to find out exactly when to do this.

    3. This is annoying, but it works.

    # docker notary stuff
    alias dockernotary="notary -s https://notary.docker.io -d ~/.docker/trust"
    # always be using content trust
    export DOCKER_CONTENT_TRUST=1

Table 6: Add these lines to ~/.bashrc.

Reset

If you need to reset YubiKeys, you may use the following script. The script looks for every plugged YubiKey, and shows a menu to reset one specific key, or all of them. Please read and follow all of the instructions carefully. YOU WILL NOT BE ABLE TO RETRIEVE KEYS/DATA FROM THE YUBIKEY AFTER COMPLETION.

$ ./reset.sh

Troubleshooting

Blocked card

If you are blocked out of using GPG because you entered your PIN wrong too many times (3x by default), don’t panic: just follow the instructions here. Make sure you enter your Admin PIN correctly within 3x, otherwise your current keys are blocked, and you must reset your YubiKey to use new keys.

Error with git pull/fetch or when using SSH

If you try to ssh or git pull/fetch and you have the following error:

sign_and_send_pubkey: signing failed: agent refused operation

You are probably mistyping your PIN. To verify it, you can:

gpg --card-edit
gpg/card> verify
...
PIN retry counter : 3 0 3 # if it is the right PIN
PIN retry counter : 2 0 3 # if it is a wrong PIN
...

If your PIN is wrong, try 123456, which is the default PIN. If it still fails, reset your PIN:

gpg --card-edit
gpg/card> admin
gpg/card> passwd
gpg: OpenPGP card no. D2760001240102010006055532110000 detected

Your selection? 1
PIN changed.

1 - change PIN
2 - unblock PIN
3 - change Admin PIN
4 - set the Reset Code
Q - quit

Your selection? q

git rebase

If you are using the FIPS model, you can perform signing operations for 15 seconds after touching your YubiKey before having to touch it again. When running a large git rebase, you may have to touch your YubiKey multiple times. If the rebase seems to hang and the YubiKey flashes, it means you need to touch it again.

If you are still having issues when rebasing, you might consider using the --no-gpg-sign flag as a workaround.

Signing for different git repositories with different keys

The script can setup your Git installation so that all your commits and tags will be signed by default with the key contained in the YubiKey. We strongly recommend that you turn on this option. If you have done so, please stop reading here.

Otherwise, one reason for declining this option may be that you wish to sign for different repositories with different keys. There are a few ways to handle this. Perhaps the simplest is to let the script assign the YubiKey to all git repositories, and then use git config --local to override user.signingkey for different repositories.

Alternatively, let us say you use your personal key for open source projects, and the one in the YubiKey for Datadog proprietary code. One possible solution is to setup git aliases. First, make sure signing is turned on globally:

git config --global commit.gpgsign true
git config --global tag.forceSignAnnotated true

Then you can tell git to use a specific key by default, depending on which one is the one you use the most:

git config --global user.signingkey <id_of_the_key_you_want_to_use_by_default>

You can alias the commit command to override the default key and use another one to sign that specific commit:

git config --global alias.dd-commit '-c user.signingkey=<id_of_the_yubikey_key> commit'
git config --global alias.dd-tag '-c user.signingkey=<id_of_the_yubikey_key> tag'

With this setup, every time you do git commit or git tag, the default key will be used while git dd-commit and git dd-tag will use the one in the YubiKey.

TODO

  1. Instructions for revoking and / or replacing keys.

  2. Procedures for recovering from key compromise / theft / loss.

References

  1. https://ruimarinho.gitbooks.io/yubikey-handbook/content/openpgp/

  2. https://mikegerwitz.com/papers/git-horror-story

  3. http://karl.kornel.us/2017/10/welp-there-go-my-git-signatures/

  4. https://lists.linuxfoundation.org/pipermail/bitcoin-dev/2014-May/005877.html

About

YubiKey at Datadog

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Shell 100.0%