implement changes from review

This commit is contained in:
Raal Goff 2021-11-04 14:05:07 +08:00
parent 26cb52a573
commit 222b52db13
9 changed files with 224 additions and 46 deletions

View file

@ -46,7 +46,8 @@ type Authority interface {
GetRoots() (federation []*x509.Certificate, err error) GetRoots() (federation []*x509.Certificate, err error)
GetFederation() ([]*x509.Certificate, error) GetFederation() ([]*x509.Certificate, error)
Version() authority.Version Version() authority.Version
GenerateCertificateRevocationList(force bool) ([]byte, error) GenerateCertificateRevocationList() error
GetCertificateRevocationList() ([]byte, error)
} }
// TimeDuration is an alias of provisioner.TimeDuration // TimeDuration is an alias of provisioner.TimeDuration

View file

@ -2,23 +2,53 @@ package api
import ( import (
"encoding/pem" "encoding/pem"
"fmt"
"github.com/pkg/errors"
"net/http" "net/http"
) )
// CRL is an HTTP handler that returns the current CRL in PEM format // CRL is an HTTP handler that returns the current CRL in DER or PEM format
func (h *caHandler) CRL(w http.ResponseWriter, r *http.Request) { func (h *caHandler) CRL(w http.ResponseWriter, r *http.Request) {
crlBytes, err := h.Authority.GenerateCertificateRevocationList(false) crlBytes, err := h.Authority.GetCertificateRevocationList()
_, formatAsPEM := r.URL.Query()["pem"]
if err != nil { if err != nil {
w.WriteHeader(500) w.WriteHeader(500)
_, err = fmt.Fprintf(w, "%v\n", err)
if err != nil {
panic(errors.Wrap(err, "error writing http response"))
}
return return
} }
pemBytes := pem.EncodeToMemory(&pem.Block{ if crlBytes == nil {
Type: "X509 CRL", w.WriteHeader(404)
Bytes: crlBytes, _, err = fmt.Fprintln(w, "No CRL available")
}) if err != nil {
panic(errors.Wrap(err, "error writing http response"))
}
return
}
if formatAsPEM {
pemBytes := pem.EncodeToMemory(&pem.Block{
Type: "X509 CRL",
Bytes: crlBytes,
})
w.Header().Add("Content-Type", "application/x-pem-file")
w.Header().Add("Content-Disposition", "attachment; filename=\"crl.pem\"")
_, err = w.Write(pemBytes)
} else {
w.Header().Add("Content-Type", "application/pkix-crl")
w.Header().Add("Content-Disposition", "attachment; filename=\"crl.der\"")
_, err = w.Write(crlBytes)
}
w.WriteHeader(200) w.WriteHeader(200)
_, err = w.Write(pemBytes)
if err != nil {
panic(errors.Wrap(err, "error writing http response"))
}
} }

View file

@ -64,6 +64,9 @@ type Authority struct {
sshCAUserFederatedCerts []ssh.PublicKey sshCAUserFederatedCerts []ssh.PublicKey
sshCAHostFederatedCerts []ssh.PublicKey sshCAHostFederatedCerts []ssh.PublicKey
// CRL vars
crlChannel chan int
// Do not re-initialize // Do not re-initialize
initOnce bool initOnce bool
startTime time.Time startTime time.Time
@ -550,6 +553,16 @@ func (a *Authority) init() error {
// not be repeated. // not be repeated.
a.initOnce = true a.initOnce = true
// Start the CRL generator
if a.config.CRL != nil {
if a.config.CRL.Generate && a.config.CRL.CacheDuration.Duration > time.Duration(0) {
err := a.startCRLGenerator()
if err != nil {
return err
}
}
}
return nil return nil
} }
@ -615,3 +628,60 @@ func (a *Authority) requiresSCEPService() bool {
func (a *Authority) GetSCEPService() *scep.Service { func (a *Authority) GetSCEPService() *scep.Service {
return a.scepService return a.scepService
} }
func (a *Authority) startCRLGenerator() error {
if a.config.CRL.CacheDuration.Duration > time.Duration(0) {
// Check that there is a valid CRL in the DB right now. If it doesnt exist
// or is expired, generated one now
crlDB, ok := a.db.(db.CertificateRevocationListDB)
if !ok {
return errors.Errorf("CRL Generation requested, but database does not support CRL generation")
}
crlInfo, err := crlDB.GetCRL()
if err != nil {
return errors.Wrap(err, "could not retrieve CRL from database")
}
if crlInfo == nil {
log.Println("No CRL exists in the DB, generating one now")
err = a.GenerateCertificateRevocationList()
if err != nil {
return errors.Wrap(err, "could not generate a CRL")
}
}
if crlInfo.ExpiresAt.Before(time.Now().UTC()) {
log.Printf("Existing CRL has expired (at %v), generating a new one", crlInfo.ExpiresAt)
err = a.GenerateCertificateRevocationList()
if err != nil {
return errors.Wrap(err, "could not generate a CRL")
}
}
log.Printf("CRL will be auto-generated every %v", a.config.CRL.CacheDuration)
tickerDuration := a.config.CRL.CacheDuration.Duration - time.Minute // generate the new CRL 1 minute before it expires
if tickerDuration <= 0 {
log.Printf("WARNING: Addition of jitter to CRL generation time %v creates a negative duration (%v). Using 1 minute cacheDuration", a.config.CRL.CacheDuration, tickerDuration)
tickerDuration = time.Minute
}
crlTicker := time.NewTicker(tickerDuration)
go func() {
for {
select {
case <-crlTicker.C:
log.Println("Regenerating CRL")
err := a.GenerateCertificateRevocationList()
if err != nil {
// TODO: log or panic here?
panic(errors.Wrap(err, "authority.crlGenerator encountered an error"))
}
}
}
}()
}
return nil
}

View file

@ -64,6 +64,7 @@ type Config struct {
TLS *TLSOptions `json:"tls,omitempty"` TLS *TLSOptions `json:"tls,omitempty"`
Password string `json:"password,omitempty"` Password string `json:"password,omitempty"`
Templates *templates.Templates `json:"templates,omitempty"` Templates *templates.Templates `json:"templates,omitempty"`
CRL *CRLConfig `json:"crl,omitempty"`
} }
// ASN1DN contains ASN1.DN attributes that are used in Subject and Issuer // ASN1DN contains ASN1.DN attributes that are used in Subject and Issuer
@ -95,6 +96,12 @@ type AuthConfig struct {
EnableAdmin bool `json:"enableAdmin,omitempty"` EnableAdmin bool `json:"enableAdmin,omitempty"`
} }
// CRLConfig represents config options for CRL generation
type CRLConfig struct {
Generate bool `json:"generate,omitempty"`
CacheDuration *provisioner.Duration `json:"cacheDuration,omitempty"`
}
// init initializes the required fields in the AuthConfig if they are not // init initializes the required fields in the AuthConfig if they are not
// provided. // provided.
func (c *AuthConfig) init() { func (c *AuthConfig) init() {

View file

@ -364,6 +364,16 @@ func (a *Authority) Revoke(ctx context.Context, revokeOpts *RevokeOptions) error
p provisioner.Interface p provisioner.Interface
err error err error
) )
// Attempt to get the certificate expiry using the serial number.
cert, err := a.db.GetCertificate(revokeOpts.Serial)
// Revocation of a certificate not in the database may be requested, so fill in the expiry only
// if we can
if err == nil {
rci.ExpiresAt = cert.NotAfter
}
// If not mTLS then get the TokenID of the token. // If not mTLS then get the TokenID of the token.
if !revokeOpts.MTLS { if !revokeOpts.MTLS {
token, err := jose.ParseSigned(revokeOpts.OTT) token, err := jose.ParseSigned(revokeOpts.OTT)
@ -430,7 +440,10 @@ func (a *Authority) Revoke(ctx context.Context, revokeOpts *RevokeOptions) error
err = a.revoke(revokedCert, rci) err = a.revoke(revokedCert, rci)
// Generate a new CRL so CRL requesters will always get an up-to-date CRL whenever they request it // Generate a new CRL so CRL requesters will always get an up-to-date CRL whenever they request it
_, _ = a.GenerateCertificateRevocationList(true) err = a.GenerateCertificateRevocationList()
if err != nil {
return errs.Wrap(http.StatusInternalServerError, err, "authority.Revoke", opts...)
}
} }
switch err { switch err {
case nil: case nil:
@ -463,36 +476,64 @@ func (a *Authority) revokeSSH(crt *ssh.Certificate, rci *db.RevokedCertificateIn
return a.db.Revoke(rci) return a.db.Revoke(rci)
} }
// GenerateCertificateRevocationList returns a DER representation of a signed CRL. // GetCertificateRevocationList will return the currently generated CRL from the DB, or a not implemented
// It will look for a valid generated CRL in the database, check if it has expired, and generate // error if the underlying AuthDB does not support CRLs
// a new CRL on demand if it has expired (or a CRL does not already exist). func (a *Authority) GetCertificateRevocationList() ([]byte, error) {
// if a.config.CRL == nil {
// force set to true will force regeneration of the CRL regardless of whether it has actually expired return nil, errs.Wrap(http.StatusInternalServerError, errors.Errorf("Certificate Revocation Lists are not enabled"), "authority.GetCertificateRevocationList")
func (a *Authority) GenerateCertificateRevocationList(force bool) ([]byte, error) {
// check for an existing CRL in the database, and return that if its valid
crlInfo, err := a.db.GetCRL()
if err != nil {
return nil, err
} }
if !force && crlInfo != nil && crlInfo.ExpiresAt.After(time.Now().UTC()) { crlDB, ok := a.db.(db.CertificateRevocationListDB)
return crlInfo.DER, nil if !ok {
return nil, errs.Wrap(http.StatusInternalServerError, errors.Errorf("Database does not support Certificate Revocation Lists"), "authority.GetCertificateRevocationList")
}
crlInfo, err := crlDB.GetCRL()
if err != nil {
return nil, errs.Wrap(http.StatusInternalServerError, err, "authority.GetCertificateRevocationList")
}
if crlInfo == nil {
return nil, nil
}
return crlInfo.DER, nil
}
// GenerateCertificateRevocationList generates a DER representation of a signed CRL and stores it in the
// database. Returns nil if CRL generation has been disabled in the config
func (a *Authority) GenerateCertificateRevocationList() error {
if a.config.CRL == nil {
// CRL is disabled
return nil
}
crlDB, ok := a.db.(db.CertificateRevocationListDB)
if !ok {
return errors.Errorf("Database does not support CRL generation")
} }
// some CAS may not implement the CRLGenerator interface, so check before we proceed // some CAS may not implement the CRLGenerator interface, so check before we proceed
caCRLGenerator, ok := a.x509CAService.(casapi.CertificateAuthorityCRLGenerator) caCRLGenerator, ok := a.x509CAService.(casapi.CertificateAuthorityCRLGenerator)
if !ok { if !ok {
return nil, errors.Errorf("CRL Generator not implemented") return errors.Errorf("CA does not support CRL Generation")
} }
revokedList, err := a.db.GetRevokedCertificates() crlInfo, err := crlDB.GetCRL()
if err != nil {
return errors.Wrap(err, "could not retrieve CRL from database")
}
revokedList, err := crlDB.GetRevokedCertificates()
if err != nil {
return errors.Wrap(err, "could not retrieve revoked certificates list from database")
}
// Number is a monotonically increasing integer (essentially the CRL version number) that we need to // Number is a monotonically increasing integer (essentially the CRL version number) that we need to
// keep track of and increase every time we generate a new CRL // keep track of and increase every time we generate a new CRL
var n int64 = 0 var n int64
var bn big.Int var bn big.Int
if crlInfo != nil { if crlInfo != nil {
@ -515,37 +556,36 @@ func (a *Authority) GenerateCertificateRevocationList(force bool) ([]byte, error
} }
// Create a RevocationList representation ready for the CAS to sign // Create a RevocationList representation ready for the CAS to sign
// TODO: use a config value for the NextUpdate time duration
// TODO: allow SignatureAlgorithm to be specified? // TODO: allow SignatureAlgorithm to be specified?
revocationList := x509.RevocationList{ revocationList := x509.RevocationList{
SignatureAlgorithm: 0, SignatureAlgorithm: 0,
RevokedCertificates: revokedCertificates, RevokedCertificates: revokedCertificates,
Number: &bn, Number: &bn,
ThisUpdate: time.Now().UTC(), ThisUpdate: time.Now().UTC(),
NextUpdate: time.Now().UTC().Add(time.Minute * 10), NextUpdate: time.Now().UTC().Add(a.config.CRL.CacheDuration.Duration),
ExtraExtensions: nil, ExtraExtensions: nil,
} }
certificateRevocationList, err := caCRLGenerator.CreateCertificateRevocationList(&revocationList) certificateRevocationList, err := caCRLGenerator.CreateCRL(&casapi.CreateCRLRequest{RevocationList: &revocationList})
if err != nil { if err != nil {
return nil, err return errors.Wrap(err, "could not create CRL")
} }
// Create a new db.CertificateRevocationListInfo, which stores the new Number we just generated, the // Create a new db.CertificateRevocationListInfo, which stores the new Number we just generated, the
// expiry time, and the DER-encoded CRL - then store it in the DB // expiry time, and the DER-encoded CRL
newCRLInfo := db.CertificateRevocationListInfo{ newCRLInfo := db.CertificateRevocationListInfo{
Number: n, Number: n,
ExpiresAt: revocationList.NextUpdate, ExpiresAt: revocationList.NextUpdate,
DER: certificateRevocationList, DER: certificateRevocationList.CRL,
} }
err = a.db.StoreCRL(&newCRLInfo) // Store the CRL in the database ready for retrieval by api endpoints
err = crlDB.StoreCRL(&newCRLInfo)
if err != nil { if err != nil {
return nil, err return errors.Wrap(err, "could not store CRL in database")
} }
// Finally, return our CRL in DER return nil
return certificateRevocationList, nil
} }
// GetTLSCertificate creates a new leaf certificate to be used by the CA HTTPS server. // GetTLSCertificate creates a new leaf certificate to be used by the CA HTTPS server.

View file

@ -144,3 +144,13 @@ type CreateCertificateAuthorityResponse struct {
PrivateKey crypto.PrivateKey PrivateKey crypto.PrivateKey
Signer crypto.Signer Signer crypto.Signer
} }
// CreateCRLRequest is the request to create a Certificate Revocation List.
type CreateCRLRequest struct {
RevocationList *x509.RevocationList
}
// CreateCRLResponse is the response to a Certificate Revocation List request.
type CreateCRLResponse struct {
CRL []byte //the CRL in DER format
}

View file

@ -17,7 +17,7 @@ type CertificateAuthorityService interface {
// CertificateAuthorityCRLGenerator is an optional interface implemented by CertificateAuthorityService // CertificateAuthorityCRLGenerator is an optional interface implemented by CertificateAuthorityService
// that has a method to create a CRL // that has a method to create a CRL
type CertificateAuthorityCRLGenerator interface { type CertificateAuthorityCRLGenerator interface {
CreateCertificateRevocationList(crl *x509.RevocationList) ([]byte, error) CreateCRL(req *CreateCRLRequest) (*CreateCRLResponse, error)
} }
// CertificateAuthorityGetter is an interface implemented by a // CertificateAuthorityGetter is an interface implemented by a

View file

@ -113,15 +113,15 @@ func (c *SoftCAS) RevokeCertificate(req *apiv1.RevokeCertificateRequest) (*apiv1
}, nil }, nil
} }
// CreateCertificateRevocationList will create a new CRL based on the RevocationList passed to it // CreateCRL will create a new CRL based on the RevocationList passed to it
func (c *SoftCAS) CreateCertificateRevocationList(crl *x509.RevocationList) ([]byte, error) { func (c *SoftCAS) CreateCRL(req *apiv1.CreateCRLRequest) (*apiv1.CreateCRLResponse, error) {
revocationList, err := x509.CreateRevocationList(rand.Reader, crl, c.CertificateChain[0], c.Signer) revocationListBytes, err := x509.CreateRevocationList(rand.Reader, req.RevocationList, c.CertificateChain[0], c.Signer)
if err != nil { if err != nil {
return nil, err return nil, err
} }
return revocationList, nil return &apiv1.CreateCRLResponse{CRL: revocationListBytes}, nil
} }
// CreateCertificateAuthority creates a root or an intermediate certificate. // CreateCertificateAuthority creates a root or an intermediate certificate.

View file

@ -51,9 +51,6 @@ type AuthDB interface {
IsSSHRevoked(sn string) (bool, error) IsSSHRevoked(sn string) (bool, error)
Revoke(rci *RevokedCertificateInfo) error Revoke(rci *RevokedCertificateInfo) error
RevokeSSH(rci *RevokedCertificateInfo) error RevokeSSH(rci *RevokedCertificateInfo) error
GetRevokedCertificates() (*[]RevokedCertificateInfo, error)
GetCRL() (*CertificateRevocationListInfo, error)
StoreCRL(*CertificateRevocationListInfo) error
GetCertificate(serialNumber string) (*x509.Certificate, error) GetCertificate(serialNumber string) (*x509.Certificate, error)
StoreCertificate(crt *x509.Certificate) error StoreCertificate(crt *x509.Certificate) error
UseToken(id, tok string) (bool, error) UseToken(id, tok string) (bool, error)
@ -63,6 +60,13 @@ type AuthDB interface {
Shutdown() error Shutdown() error
} }
// CertificateRevocationListDB is an interface to indicate whether the DB supports CRL generation
type CertificateRevocationListDB interface {
GetRevokedCertificates() (*[]RevokedCertificateInfo, error)
GetCRL() (*CertificateRevocationListInfo, error)
StoreCRL(*CertificateRevocationListInfo) error
}
// DB is a wrapper over the nosql.DB interface. // DB is a wrapper over the nosql.DB interface.
type DB struct { type DB struct {
nosql.DB nosql.DB
@ -109,6 +113,7 @@ type RevokedCertificateInfo struct {
ReasonCode int ReasonCode int
Reason string Reason string
RevokedAt time.Time RevokedAt time.Time
ExpiresAt time.Time
TokenID string TokenID string
MTLS bool MTLS bool
} }
@ -215,7 +220,22 @@ func (db *DB) GetRevokedCertificates() (*[]RevokedCertificateInfo, error) {
if err := json.Unmarshal(e.Value, &data); err != nil { if err := json.Unmarshal(e.Value, &data); err != nil {
return nil, err return nil, err
} }
revokedCerts = append(revokedCerts, data)
if !data.ExpiresAt.IsZero() && data.ExpiresAt.After(time.Now().UTC()) {
revokedCerts = append(revokedCerts, data)
} else if data.ExpiresAt.IsZero() {
cert, err := db.GetCertificate(data.Serial)
if err != nil {
revokedCerts = append(revokedCerts, data) // a revoked certificate may not be in the database,
// so its expiry date is undiscoverable and will need
// to be added to the crl always
continue
}
if cert.NotAfter.After(time.Now().UTC()) {
revokedCerts = append(revokedCerts, data)
}
}
} }
return &revokedCerts, nil return &revokedCerts, nil