2018-03-19 13:53:57 -07:00
|
|
|
package connect
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto"
|
|
|
|
"crypto/ecdsa"
|
2019-07-30 17:47:39 -04:00
|
|
|
"crypto/rsa"
|
2018-04-24 16:16:37 -07:00
|
|
|
"crypto/sha1"
|
2018-03-19 13:53:57 -07:00
|
|
|
"crypto/sha256"
|
|
|
|
"crypto/x509"
|
2019-09-23 12:52:35 -05:00
|
|
|
"encoding/hex"
|
2018-03-19 13:53:57 -07:00
|
|
|
"encoding/pem"
|
|
|
|
"fmt"
|
2019-09-23 12:52:35 -05:00
|
|
|
"math/big"
|
2018-03-21 12:54:51 -10:00
|
|
|
"strings"
|
2018-03-19 13:53:57 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
// ParseCert parses the x509 certificate from a PEM-encoded value.
|
|
|
|
func ParseCert(pemValue string) (*x509.Certificate, error) {
|
2018-03-24 08:27:44 -10:00
|
|
|
// The _ result below is not an error but the remaining PEM bytes.
|
2018-03-19 13:53:57 -07:00
|
|
|
block, _ := pem.Decode([]byte(pemValue))
|
|
|
|
if block == nil {
|
|
|
|
return nil, fmt.Errorf("no PEM-encoded data found")
|
|
|
|
}
|
|
|
|
|
|
|
|
if block.Type != "CERTIFICATE" {
|
|
|
|
return nil, fmt.Errorf("first PEM-block should be CERTIFICATE type")
|
|
|
|
}
|
|
|
|
|
|
|
|
return x509.ParseCertificate(block.Bytes)
|
|
|
|
}
|
|
|
|
|
2018-04-26 20:14:37 -07:00
|
|
|
// CalculateCertFingerprint parses the x509 certificate from a PEM-encoded value
|
|
|
|
// and calculates the SHA-1 fingerprint.
|
|
|
|
func CalculateCertFingerprint(pemValue string) (string, error) {
|
2018-04-24 16:16:37 -07:00
|
|
|
// The _ result below is not an error but the remaining PEM bytes.
|
|
|
|
block, _ := pem.Decode([]byte(pemValue))
|
|
|
|
if block == nil {
|
|
|
|
return "", fmt.Errorf("no PEM-encoded data found")
|
|
|
|
}
|
|
|
|
|
2018-04-26 20:14:37 -07:00
|
|
|
if block.Type != "CERTIFICATE" {
|
|
|
|
return "", fmt.Errorf("first PEM-block should be CERTIFICATE type")
|
2018-04-24 16:16:37 -07:00
|
|
|
}
|
2018-04-26 20:14:37 -07:00
|
|
|
|
|
|
|
hash := sha1.Sum(block.Bytes)
|
|
|
|
return HexString(hash[:]), nil
|
2018-04-24 16:16:37 -07:00
|
|
|
}
|
|
|
|
|
2018-03-19 13:53:57 -07:00
|
|
|
// ParseSigner parses a crypto.Signer from a PEM-encoded key. The private key
|
|
|
|
// is expected to be the first block in the PEM value.
|
|
|
|
func ParseSigner(pemValue string) (crypto.Signer, error) {
|
2018-03-24 08:27:44 -10:00
|
|
|
// The _ result below is not an error but the remaining PEM bytes.
|
2018-03-19 13:53:57 -07:00
|
|
|
block, _ := pem.Decode([]byte(pemValue))
|
|
|
|
if block == nil {
|
|
|
|
return nil, fmt.Errorf("no PEM-encoded data found")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch block.Type {
|
|
|
|
case "EC PRIVATE KEY":
|
|
|
|
return x509.ParseECPrivateKey(block.Bytes)
|
|
|
|
|
2018-04-24 16:16:37 -07:00
|
|
|
case "RSA PRIVATE KEY":
|
|
|
|
return x509.ParsePKCS1PrivateKey(block.Bytes)
|
|
|
|
|
2018-04-20 18:46:02 -07:00
|
|
|
case "PRIVATE KEY":
|
|
|
|
signer, err := x509.ParsePKCS8PrivateKey(block.Bytes)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
pk, ok := signer.(crypto.Signer)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("private key is not a valid format")
|
|
|
|
}
|
|
|
|
|
|
|
|
return pk, nil
|
|
|
|
|
2018-03-19 13:53:57 -07:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("unknown PEM block type for signing key: %s", block.Type)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-19 14:36:17 -07:00
|
|
|
// ParseCSR parses a CSR from a PEM-encoded value. The certificate request
|
|
|
|
// must be the the first block in the PEM value.
|
|
|
|
func ParseCSR(pemValue string) (*x509.CertificateRequest, error) {
|
2018-03-24 08:27:44 -10:00
|
|
|
// The _ result below is not an error but the remaining PEM bytes.
|
2018-03-19 14:36:17 -07:00
|
|
|
block, _ := pem.Decode([]byte(pemValue))
|
|
|
|
if block == nil {
|
|
|
|
return nil, fmt.Errorf("no PEM-encoded data found")
|
|
|
|
}
|
|
|
|
|
|
|
|
if block.Type != "CERTIFICATE REQUEST" {
|
|
|
|
return nil, fmt.Errorf("first PEM-block should be CERTIFICATE REQUEST type")
|
|
|
|
}
|
|
|
|
|
|
|
|
return x509.ParseCertificateRequest(block.Bytes)
|
|
|
|
}
|
|
|
|
|
2018-03-19 13:53:57 -07:00
|
|
|
// KeyId returns a x509 KeyId from the given signing key. The key must be
|
2018-03-24 08:27:44 -10:00
|
|
|
// an *ecdsa.PublicKey currently, but may support more types in the future.
|
2018-03-19 13:53:57 -07:00
|
|
|
func KeyId(raw interface{}) ([]byte, error) {
|
2018-04-24 16:16:37 -07:00
|
|
|
switch raw.(type) {
|
|
|
|
case *ecdsa.PublicKey:
|
2019-07-30 17:47:39 -04:00
|
|
|
case *rsa.PublicKey:
|
2018-04-24 16:16:37 -07:00
|
|
|
default:
|
2018-03-19 13:53:57 -07:00
|
|
|
return nil, fmt.Errorf("invalid key type: %T", raw)
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is not standard; RFC allows any unique identifier as long as they
|
|
|
|
// match in subject/authority chains but suggests specific hashing of DER
|
2018-03-24 08:27:44 -10:00
|
|
|
// bytes of public key including DER tags.
|
2018-04-24 16:16:37 -07:00
|
|
|
bs, err := x509.MarshalPKIXPublicKey(raw)
|
2018-03-24 08:27:44 -10:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
kID := sha256.Sum256(bs)
|
2019-09-23 12:52:35 -05:00
|
|
|
return kID[:], nil
|
2018-03-19 13:53:57 -07:00
|
|
|
}
|
2018-03-21 12:54:51 -10:00
|
|
|
|
2019-09-23 12:52:35 -05:00
|
|
|
// EncodeSerialNumber encodes the given serial number as a colon-hex encoded
|
|
|
|
// string.
|
|
|
|
func EncodeSerialNumber(serial *big.Int) string {
|
|
|
|
return HexString(serial.Bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
// EncodeSigningKeyID encodes the given AuthorityKeyId or SubjectKeyId into a
|
|
|
|
// colon-hex encoded string suitable for using as a SigningKeyID value.
|
|
|
|
func EncodeSigningKeyID(keyID []byte) string { return HexString(keyID) }
|
|
|
|
|
2018-03-21 12:54:51 -10:00
|
|
|
// HexString returns a standard colon-separated hex value for the input
|
|
|
|
// byte slice. This should be used with cert serial numbers and so on.
|
|
|
|
func HexString(input []byte) string {
|
|
|
|
return strings.Replace(fmt.Sprintf("% x", input), " ", ":", -1)
|
|
|
|
}
|
2019-09-23 12:52:35 -05:00
|
|
|
|
|
|
|
// IsHexString returns true if the input is the output of HexString(). Meant
|
|
|
|
// for use in tests.
|
|
|
|
func IsHexString(input []byte) bool {
|
|
|
|
s := string(input)
|
|
|
|
if strings.Count(s, ":") < 5 { // 5 is arbitrary
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
s = strings.ReplaceAll(s, ":", "")
|
|
|
|
_, err := hex.DecodeString(s)
|
|
|
|
return err == nil
|
|
|
|
}
|