A complicated aspect of security is reliability and guaranteeing the consistency of all security controls. Instead of relying on a central authentication authority such as LDAP or Kerberos, we can take advantage of SSH or, more specifically OpenSSH, to provide both.
In addition to authenticating SSH client access with PIV and PKCS#11, it is possible to increment the security of the remote SSH authentication. Facebook and Yahoo have switched to SSH User Certificates to avoid lockdown if the central authentication system goes down. It also helps maintaining the
authorized_keys file, as it does not scale well (it requires a 1:1 match).
A SSH User Certificate Authority can sign and thus securely authenticate each client connecting to a server.
The signed certificate also designates the principals (login identities) that can be used with that certificate. For each user, the principals can be described on a file:
❯ mkdir /etc/ssh/auth_principals ❯ echo -e 'access-root' > /etc/ssh/auth_principals/root ❯ echo -e 'access-databases' > /etc/ssh/auth_principals/foobar
In this example, any signed certificate with the
access-root principal would be allowed to SSH into that host with the
root username, and any signed certificate with the
access-databases principal would be able to login with the
Now let's create the SSH User Certificate Authority.
Using an air gapped computer, generate the user certificate authority:
❯ ssh-keygen -C "SSH User Certificate Authority" -f sshuser.root.ca
Distribute the public key (
/etc/ssh/on every host. Make sure the file is
/etc/ssh/sshd_configto include the new CA and principals file:
TrustedUserCAKeys /etc/ssh/sshuser.root.ca.pub AuthorizedPrincipalsFile /etc/ssh/auth_principals/%u
Have the user/client extract the public key from their Yubikey so it can be signed by the new CA on the air gapped computer:
ssh-keygen -D /usr/local/opt/opensc/lib/pkcs11/opensc-pkcs11.so -e
Sign the user certificate on the air gapped computer, with particular attention to the login name (
<user>), the principals which this certificate will be able to claim (
<principals>, separated by commas), the certificate expiration time (
+52w) and the serial number (
<serial>, an integer which should be tracked):
❯ ssh-keygen -s sshuser.root.ca -I <user> -n <principals> -V +52w -z <serial> <user>.pub Signed user key foobar-cert.pub: id "foobar" serial 1928121 for access-root valid from 2016-12-10T00:10:00 to 2017-12-09T00:10:10
Confirm the user certificate looks good:
❯ ssh-keygen -Lf <user>-cert.pub user-cert.pub: Type: firstname.lastname@example.org user certificate Public key: RSA-CERT SHA256:NWmw3siRlxn3bsIhzaFrCsh66KKIWapFuZsNiDXhRLw Signing CA: RSA SHA256:HLD1Eb4XiCoyXew23skyisJt+3P02MOsrHHbK/DmlgY Key ID: "foobar" Serial: 1928121 Valid: from 2016-12-10T00:10:00 to 2017-12-09T00:10:10 Principals: access-root Critical Options: (none) Extensions: permit-X11-forwarding permit-agent-forwarding permit-port-forwarding permit-pty permit-user-rc
<user>-cert.pubto the client's
~/.sshdirectory and name it
id_rsa-cert.pub. The name is quite specific, as there seems to be a limitation on
opensc-pkcs11to detect a certificate other than