ssh-cert-authority request --environment stage --reason "Do important maintenance work"
ssh_ca assumes that you have several environments each with different certificate authorities and that they are configured differently. For example you might have separate environments for staging and production each with different CAs. This tool supports that.
Here's a sample requester config file. The default location for this is
$HOME/.ssh_ca/requester_config.json
{ "stage": { "PublicKeyPath": "/Users/bob/.ssh/bvanzant-stage.pub", "SignerUrl": "http://ssh-ca:8080/" }, "prod": { "PublicKeyPath": "/Users/bob.ssh/bvanzant-prod.pub", "SignerUrl": "http://ssh-ca:8080/" } }
The contents should be reasonably self explanatory. Here we have a json
blob containing a stage and prod environment. The user has chosen to use
different SSH keys for production and staging, however, this is not
required. The SignerUrl
is the location of the ssh-cert-authority daemon.
Generating this configuration file can be a little cumbersome. The
ssh-cert-authority
program has a generate-config
subcommand that
tries to aid in generating this file. New users can run something like
ssh-cert-authority generate-config --url https://ssh-ca.example.com
And that will do two things. First, it goes to ssh-ca.example.com and requests a listing of the server's configured environments (e.g. "stage" and "prod"). Second it creates the configuration file inserting https://ssh-ca.example.com as the SignerUrl and $HOME/.ssh/id_rsa.pub as the PublicKeyPath.
The generated config file is printed to stdout. You can redirect or
manually copy it into the default location of
~/.ssh_ca/requester_config.json
.
Users using multiple SSH keys or keys other than id_rsa.pub will need to manually edit the configuration after it is generated.
Once configured requesting a certificate is as simple as:
ssh-cert-authority request --environment stage --reason "Do important maintenance work"
This will print out a certificate request id like so:
Cert request id: HWK6CTFJDPTXRAXD7S6NZHO3
Hand this id off to someone that can sign your cert.
If instead you got an error like
ssh-add the private half of the key you want to use.
then go do that.
Under the hood the request
command is going to use your SSH
private key to sign your certificate request. This is how the signing
daemon knows that you are the person that actually requested the cert
(you have control of the private key and your private key is actually
private, right?). This is typically as simple as ssh-add ~/.ssh/id_rsa
but if you've got lots of keys or per-environment config you'll need to
adjust and ensure you both add the right key and that your
requester_config.json
is specifying the right public key.
ssh-cert-authority sign --environment stage HWK6CTFJDPTXRAXD7S6NZHO3
A sample signer config. By default this is in
$HOME/.ssh_ca/signer_config.json
{ "stage": { "KeyFingerprint": "66:b5:be:e5:7e:09:3f:98:97:36:9b:64:ec:ea:3a:fe", "SignerUrl": "http://ssh-ca:8080/" }, "prod": { "KeyFingerprint": "66:b5:be:e5:7e:09:3f:98:97:36:9b:64:ec:ea:3a:fe", "SignerUrl": "http://ssh-ca:8080/" } }
In this case the configuration is slightly different from requesting
because we're dealing with fingerprints instead of paths. You can easily
get the fingerprint of the private key you want to sign with by doing
ssh-keygen -l -f ~/.ssh/id_rsa
or inspecting the output of ssh-add
-l
(the ssh-add -l
output is only relevant if your private key is
loaded in your agent).
In recent versions of OpenSSH the fingerprint format has changed from
MD5 (shown above) to sha256. If you fingerprint is not colon separated
like above you need to tell OpenSSH to give you an MD5 fingerprint
instead via the -E md5 option. For example: ssh-keygen -l -E md5 -f
~/.ssh/id_rsa
. When passing in md5 do not include the "MD5:" prefix on
the fingerprint.
Github issue #23 is tracking supporting sha256 (and sha384, etc).
A word of caution: Treat signing requests very seriously. This is easily the weak point in the entire system. Inspect requests intently and look for violations of your policy on shell access to machines.
The signing portion of a request begins when someone sends you a request id. You then sign it by:
$ ssh-cert-authority sign --environment stage HWK6CTFJDPTXRAXD7S6NZHO3 Certificate data: Serial: 2 Key id: [email protected] Principals: [ec2-user ubuntu] Options: map[] Permissions: map[permit-agent-forwarding: permit-port-forwarding: permit-pty:] Valid for public key: 1c:fd:36:27:db:48:3f:ad:e2:fe:55:45:67:b1:47:99 Valid from 2015-03-31 08:21:39 -0700 PDT - 2015-03-31 10:21:39 -0700 PDT Type 'yes' if you'd like to sign this cert request, 'reject' to reject it, anything else to cancel
Inspect every field and compare it to what you know about who is requesting
this certificate and why. I'll provide a brief explanation of these here
but for more information checkout the CERTIFICATES
section of
ssh-keygen(1)
- Does the key id match with who requested the cert?
- Principals specifies the list of usernames that a requester can use to login to systems as. In our example here the user is allowed to use
ec2-user
andubuntu
.- Permissions is a list of ssh permissions that this cert grants. In particular
permit-pty
will allow the user to open up a shell. Here we also seepermit-agent-forwarding
which allows the user to forward along theirssh-agent
connection (generally useful) andpermit-port-forwarding
which allows the user of this cert to forward ports along connections.
Also inspect the validity period. What is normal for your organization? In general the less time a certificate is valid for the less likely it is to be abused. sign_cert will print out the expiry time of a certificate in red if the value is more than 48 hours in the future.
If you, as a signer, are happy with the certificate request you can type
yes
and the certificate will be, effectively, +1'ed by you.
If you believe this request is a Bad Idea and should not be approved by
anyone you can reject it forcefully and authoritatively by typing
reject
. This will permanently mark the request as rejected and it can
never be signed after that.
Any other input is ignored and sign_cert exits.
In order for sign_cert to run your SSH key must be loaded in ssh-agent
(via ssh-add
). Otherwise sign
will exit with an error:
ssh-add the private half of the key you want to use.
ssh-cert-authority get --environment stage HWK6CTFJDPTXRAXD7S6NZHO3
The get command uses the requester_config.json
file described under
requesting a certificate.