2019-11-19 10:00:20 -07:00
|
|
|
package cert
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type NebulaCAPool struct {
|
|
|
|
CAs map[string]*NebulaCertificate
|
2020-08-05 19:17:47 -06:00
|
|
|
certBlocklist map[string]struct{}
|
2019-11-19 10:00:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewCAPool creates a CAPool
|
|
|
|
func NewCAPool() *NebulaCAPool {
|
|
|
|
ca := NebulaCAPool{
|
|
|
|
CAs: make(map[string]*NebulaCertificate),
|
2020-08-05 19:17:47 -06:00
|
|
|
certBlocklist: make(map[string]struct{}),
|
2019-11-19 10:00:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return &ca
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCAPoolFromBytes(caPEMs []byte) (*NebulaCAPool, error) {
|
|
|
|
pool := NewCAPool()
|
|
|
|
var err error
|
|
|
|
for {
|
|
|
|
caPEMs, err = pool.AddCACertificate(caPEMs)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if caPEMs == nil || len(caPEMs) == 0 || strings.TrimSpace(string(caPEMs)) == "" {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pool, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddCACertificate verifies a Nebula CA certificate and adds it to the pool
|
|
|
|
// Only the first pem encoded object will be consumed, any remaining bytes are returned.
|
|
|
|
// Parsed certificates will be verified and must be a CA
|
|
|
|
func (ncp *NebulaCAPool) AddCACertificate(pemBytes []byte) ([]byte, error) {
|
|
|
|
c, pemBytes, err := UnmarshalNebulaCertificateFromPEM(pemBytes)
|
|
|
|
if err != nil {
|
|
|
|
return pemBytes, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !c.Details.IsCA {
|
|
|
|
return pemBytes, fmt.Errorf("provided certificate was not a CA; %s", c.Details.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !c.CheckSignature(c.Details.PublicKey) {
|
|
|
|
return pemBytes, fmt.Errorf("provided certificate was not self signed; %s", c.Details.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.Expired(time.Now()) {
|
|
|
|
return pemBytes, fmt.Errorf("provided CA certificate is expired; %s", c.Details.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
sum, err := c.Sha256Sum()
|
|
|
|
if err != nil {
|
|
|
|
return pemBytes, fmt.Errorf("could not calculate shasum for provided CA; error: %s; %s", err, c.Details.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
ncp.CAs[sum] = c
|
|
|
|
return pemBytes, nil
|
|
|
|
}
|
|
|
|
|
2020-08-05 19:17:47 -06:00
|
|
|
// BlocklistFingerprint adds a cert fingerprint to the blocklist
|
|
|
|
func (ncp *NebulaCAPool) BlocklistFingerprint(f string) {
|
|
|
|
ncp.certBlocklist[f] = struct{}{}
|
2019-11-19 10:00:20 -07:00
|
|
|
}
|
|
|
|
|
2020-08-05 19:17:47 -06:00
|
|
|
// ResetCertBlocklist removes all previously blocklisted cert fingerprints
|
|
|
|
func (ncp *NebulaCAPool) ResetCertBlocklist() {
|
|
|
|
ncp.certBlocklist = make(map[string]struct{})
|
2019-11-19 10:00:20 -07:00
|
|
|
}
|
|
|
|
|
2020-08-05 19:17:47 -06:00
|
|
|
// IsBlocklisted returns true if the fingerprint fails to generate or has been explicitly blocklisted
|
|
|
|
func (ncp *NebulaCAPool) IsBlocklisted(c *NebulaCertificate) bool {
|
2019-11-19 10:00:20 -07:00
|
|
|
h, err := c.Sha256Sum()
|
|
|
|
if err != nil {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2020-08-05 19:17:47 -06:00
|
|
|
if _, ok := ncp.certBlocklist[h]; ok {
|
2019-11-19 10:00:20 -07:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCAForCert attempts to return the signing certificate for the provided certificate.
|
|
|
|
// No signature validation is performed
|
|
|
|
func (ncp *NebulaCAPool) GetCAForCert(c *NebulaCertificate) (*NebulaCertificate, error) {
|
|
|
|
if c.Details.Issuer == "" {
|
|
|
|
return nil, fmt.Errorf("no issuer in certificate")
|
|
|
|
}
|
|
|
|
|
|
|
|
signer, ok := ncp.CAs[c.Details.Issuer]
|
|
|
|
if ok {
|
|
|
|
return signer, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("could not find ca for the certificate")
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetFingerprints returns an array of trusted CA fingerprints
|
|
|
|
func (ncp *NebulaCAPool) GetFingerprints() []string {
|
|
|
|
fp := make([]string, len(ncp.CAs))
|
|
|
|
|
|
|
|
i := 0
|
|
|
|
for k := range ncp.CAs {
|
|
|
|
fp[i] = k
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
return fp
|
|
|
|
}
|