Use sshutil for ssh renewing and rekeying.

This commit is contained in:
Mariano Cano 2020-07-27 15:54:51 -07:00
parent b66d123572
commit d7e590908e

View file

@ -310,16 +310,6 @@ func (a *Authority) SignSSH(ctx context.Context, key ssh.PublicKey, opts provisi
// RenewSSH creates a signed SSH certificate using the old SSH certificate as a template. // RenewSSH creates a signed SSH certificate using the old SSH certificate as a template.
func (a *Authority) RenewSSH(ctx context.Context, oldCert *ssh.Certificate) (*ssh.Certificate, error) { func (a *Authority) RenewSSH(ctx context.Context, oldCert *ssh.Certificate) (*ssh.Certificate, error) {
nonce, err := randutil.ASCII(32)
if err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "renewSSH")
}
var serial uint64
if err := binary.Read(rand.Reader, binary.BigEndian, &serial); err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "renewSSH: error reading random number")
}
if oldCert.ValidAfter == 0 || oldCert.ValidBefore == 0 { if oldCert.ValidAfter == 0 || oldCert.ValidBefore == 0 {
return nil, errs.BadRequest("rewnewSSH: cannot renew certificate without validity period") return nil, errs.BadRequest("rewnewSSH: cannot renew certificate without validity period")
} }
@ -330,15 +320,15 @@ func (a *Authority) RenewSSH(ctx context.Context, oldCert *ssh.Certificate) (*ss
va := now.Add(-1 * backdate) va := now.Add(-1 * backdate)
vb := now.Add(duration - backdate) vb := now.Add(duration - backdate)
// Build base certificate with the key and some random values // Build base certificate with the old key.
// Nonce and serial will be automatically generated on signing.
cert := &ssh.Certificate{ cert := &ssh.Certificate{
Nonce: []byte(nonce),
Key: oldCert.Key, Key: oldCert.Key,
Serial: serial,
CertType: oldCert.CertType, CertType: oldCert.CertType,
KeyId: oldCert.KeyId, KeyId: oldCert.KeyId,
ValidPrincipals: oldCert.ValidPrincipals, ValidPrincipals: oldCert.ValidPrincipals,
Permissions: oldCert.Permissions, Permissions: oldCert.Permissions,
Reserved: oldCert.Reserved,
ValidAfter: uint64(va.Unix()), ValidAfter: uint64(va.Unix()),
ValidBefore: uint64(vb.Unix()), ValidBefore: uint64(vb.Unix()),
} }
@ -359,18 +349,13 @@ func (a *Authority) RenewSSH(ctx context.Context, oldCert *ssh.Certificate) (*ss
default: default:
return nil, errs.InternalServer("renewSSH: unexpected ssh certificate type: %d", cert.CertType) return nil, errs.InternalServer("renewSSH: unexpected ssh certificate type: %d", cert.CertType)
} }
cert.SignatureKey = signer.PublicKey()
// Get bytes for signing trailing the signature length. var err error
data := cert.Marshal() // Sign certificate.
data = data[:len(data)-4] cert, err = sshutil.CreateCertificate(cert, signer)
// Sign the certificate
sig, err := signer.Sign(rand.Reader, data)
if err != nil { if err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "renewSSH: error signing certificate") return nil, errs.Wrap(http.StatusInternalServerError, err, "signSSH: error signing certificate")
} }
cert.Signature = sig
if err = a.db.StoreSSHCertificate(cert); err != nil && err != db.ErrNotImplemented { if err = a.db.StoreSSHCertificate(cert); err != nil && err != db.ErrNotImplemented {
return nil, errs.Wrap(http.StatusInternalServerError, err, "renewSSH: error storing certificate in db") return nil, errs.Wrap(http.StatusInternalServerError, err, "renewSSH: error storing certificate in db")
@ -393,16 +378,6 @@ func (a *Authority) RekeySSH(ctx context.Context, oldCert *ssh.Certificate, pub
} }
} }
nonce, err := randutil.ASCII(32)
if err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "rekeySSH")
}
var serial uint64
if err := binary.Read(rand.Reader, binary.BigEndian, &serial); err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "rekeySSH; error reading random number")
}
if oldCert.ValidAfter == 0 || oldCert.ValidBefore == 0 { if oldCert.ValidAfter == 0 || oldCert.ValidBefore == 0 {
return nil, errs.BadRequest("rekeySSH; cannot rekey certificate without validity period") return nil, errs.BadRequest("rekeySSH; cannot rekey certificate without validity period")
} }
@ -413,15 +388,15 @@ func (a *Authority) RekeySSH(ctx context.Context, oldCert *ssh.Certificate, pub
va := now.Add(-1 * backdate) va := now.Add(-1 * backdate)
vb := now.Add(duration - backdate) vb := now.Add(duration - backdate)
// Build base certificate with the key and some random values // Build base certificate with the new key.
// Nonce and serial will be automatically generated on signing.
cert := &ssh.Certificate{ cert := &ssh.Certificate{
Nonce: []byte(nonce),
Key: pub, Key: pub,
Serial: serial,
CertType: oldCert.CertType, CertType: oldCert.CertType,
KeyId: oldCert.KeyId, KeyId: oldCert.KeyId,
ValidPrincipals: oldCert.ValidPrincipals, ValidPrincipals: oldCert.ValidPrincipals,
Permissions: oldCert.Permissions, Permissions: oldCert.Permissions,
Reserved: oldCert.Reserved,
ValidAfter: uint64(va.Unix()), ValidAfter: uint64(va.Unix()),
ValidBefore: uint64(vb.Unix()), ValidBefore: uint64(vb.Unix()),
} }
@ -442,18 +417,13 @@ func (a *Authority) RekeySSH(ctx context.Context, oldCert *ssh.Certificate, pub
default: default:
return nil, errs.BadRequest("rekeySSH; unexpected ssh certificate type: %d", cert.CertType) return nil, errs.BadRequest("rekeySSH; unexpected ssh certificate type: %d", cert.CertType)
} }
cert.SignatureKey = signer.PublicKey()
// Get bytes for signing trailing the signature length. var err error
data := cert.Marshal() // Sign certificate.
data = data[:len(data)-4] cert, err = sshutil.CreateCertificate(cert, signer)
// Sign the certificate.
sig, err := signer.Sign(rand.Reader, data)
if err != nil { if err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "rekeySSH; error signing certificate") return nil, errs.Wrap(http.StatusInternalServerError, err, "signSSH: error signing certificate")
} }
cert.Signature = sig
// Apply validators from provisioner. // Apply validators from provisioner.
for _, v := range validators { for _, v := range validators {