Skip to content

Commit

Permalink
Merge pull request #81 from covermymeds/support-unordered-input
Browse files Browse the repository at this point in the history
support unsorted input to address #68
  • Loading branch information
drewby08 authored May 26, 2021
2 parents d077cf4 + 8a7822a commit 23065b7
Show file tree
Hide file tree
Showing 3 changed files with 540 additions and 99 deletions.
265 changes: 167 additions & 98 deletions certutil/certutil.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,21 @@ package certutil

import (
"bytes"
"crypto/tls"
"crypto"
"crypto/ecdsa"
"crypto/ed25519"
"crypto/rsa"
"crypto/x509"
"encoding/base64"
"encoding/pem"
"errors"
"fmt"
"github.com/twmb/algoimpl/go/graph"
"golang.org/x/crypto/pkcs12"
log "github.com/sirupsen/logrus"
"strings"
)

// Takes Base64 Encoded PKCS12 as String and produces PEM Encoded PCKS8 Private Key as String
func PemPrivateKeyFromPkcs12(b64pkcs12 string) string {
p12, _ := base64.StdEncoding.DecodeString(b64pkcs12)

Expand All @@ -21,40 +26,18 @@ func PemPrivateKeyFromPkcs12(b64pkcs12 string) string {
panic(err)
}

// Append all PEM Blocks together
var pemData []byte
for _, b := range blocks {
pemData = append(pemData, pem.EncodeToMemory(b)...)
}

return PemPrivateKeyFromPem(string(pemData))
return findPrivateKeyInPemBlocks(blocks)
}

// Takes PEM Encoded data as String and produces PEM Encoded PCKS8 Private Key as String
func PemPrivateKeyFromPem(data string) string {
pemBytes := []byte(data)

// Use tls lib to construct tls certificate and key object from PEM data
// The tls.X509KeyPair function is smart enough to parse combined cert and key pem data
certAndKey, err := tls.X509KeyPair(pemBytes, pemBytes)
if err != nil {
panic(err)
}

// Get parsed private key as PKCS8 data
privBytes, err := x509.MarshalPKCS8PrivateKey(certAndKey.PrivateKey)
if err != nil {
panic(fmt.Sprintf("Unable to marshal private key: %v", err))
}

// Encode just the private key back to PEM and return it
var privPem bytes.Buffer
if err := pem.Encode(&privPem, &pem.Block{Type: "PRIVATE KEY", Bytes: privBytes}); err != nil {
panic(fmt.Sprintf("Failed to write data: %s", err))
}

return privPem.String()
// Convert string to Pem Blocks
blocks := stringToPemBlocks(data)
// Find the Private Key from these blocks
return findPrivateKeyInPemBlocks(blocks)
}

// Takes Base64 Encoded PKCS12 as String and produces PEM Encoded x509 Certificate as String
func PemCertFromPkcs12(b64pkcs12 string) string {
p12, _ := base64.StdEncoding.DecodeString(b64pkcs12)

Expand All @@ -63,40 +46,19 @@ func PemCertFromPkcs12(b64pkcs12 string) string {
if err != nil {
panic(err)
}

// Append all PEM Blocks together
var pemData []byte
for _, b := range blocks {
pemData = append(pemData, pem.EncodeToMemory(b)...)
}

return PemCertFromPem(string(pemData))
// Find the Certificate from these blocks
return findLeafCertInPemBlocks(blocks)
}

// Takes PEM Encoded data as String and produces PEM Encoded x509 Certificate as String
func PemCertFromPem(data string) string {
pemBytes := []byte(data)

// Use tls lib to construct tls certificate and key object from PEM data
// The tls.X509KeyPair function is smart enough to parse combined cert and key pem data
certAndKey, err := tls.X509KeyPair(pemBytes, pemBytes)
if err != nil {
panic(fmt.Sprintf("Error generating X509KeyPair: %v", err))
}

leaf, err := x509.ParseCertificate(certAndKey.Certificate[0])
if err != nil {
panic(err)
}

// Encode just the leaf cert as pem
var certPem bytes.Buffer
if err := pem.Encode(&certPem, &pem.Block{Type: "CERTIFICATE", Bytes: leaf.Raw}); err != nil {
panic(fmt.Sprintf("Failed to write data: %s", err))
}

return certPem.String()
// Convert string to pem blocks
blocks := stringToPemBlocks(data)
// Find the Certificate from these blocks
return findLeafCertInPemBlocks(blocks)
}

// Takes DER Encoded Byte Array and produces PEM Encoded x509 Certificate as String
func PemCertFromBytes(derBytes []byte) string {
// Encode just the leaf cert as pem
var certPem bytes.Buffer
Expand All @@ -107,6 +69,7 @@ func PemCertFromBytes(derBytes []byte) string {
return certPem.String()
}

// Takes Base64 Encoded PKCS12 as String and produces PEM Encoded x509 Certificate Chain as String
func PemChainFromPkcs12(b64pkcs12 string, justIssuers bool) string {
p12, _ := base64.StdEncoding.DecodeString(b64pkcs12)

Expand All @@ -115,41 +78,28 @@ func PemChainFromPkcs12(b64pkcs12 string, justIssuers bool) string {
if err != nil {
panic(err)
}

// Append all PEM Blocks together
var pemData []byte
for _, b := range blocks {
pemData = append(pemData, pem.EncodeToMemory(b)...)
}

return PemChainFromPem(string(pemData), justIssuers)
// Find the Certificate chain from these blocks
return findChainInPemBlocks(blocks, justIssuers)
}

// Takes PEM Encoded data as String and produces PEM Encoded x509 Certificate Chain as String
func PemChainFromPem(data string, justIssuers bool) string {
pemBytes := []byte(data)
// Get the PEM blocks from the string
blocks := stringToPemBlocks(data)

// Use tls lib to construct tls certificate and key object from PEM data
// The tls.X509KeyPair function is smart enough to parse combined cert and key pem data
certAndKey, err := tls.X509KeyPair(pemBytes, pemBytes)
if err != nil {
panic(fmt.Sprintf("Error generating X509KeyPair: %v", err))
}

return SortedChain(certAndKey.Certificate, justIssuers)
// Find the Certificate chain from these blocks
return findChainInPemBlocks(blocks, justIssuers)
}

func SortedChain(rawChain [][]byte, justIssuers bool) string {
// Sorts an array of x509.Certificate objects
func SortedChain(certs []*x509.Certificate, justIssuers bool) []x509.Certificate {
g := graph.New(graph.Directed)

// Make a graph where each node represents a certificate and the key is its subject key identifier
certGraph := make(map[string]graph.Node, 0)

// Construct each certificate in the chain into a full certificate object
for _, certBytes := range rawChain {
cert, err := x509.ParseCertificate(certBytes)
if err != nil {
panic("Unable to parse certificate chain")
}
// For each cert make a graph node
for _, cert := range certs {
certGraph[string(cert.SubjectKeyId)] = g.MakeNode()
*certGraph[string(cert.SubjectKeyId)].Value = *cert
}
Expand All @@ -162,28 +112,147 @@ func SortedChain(rawChain [][]byte, justIssuers bool) string {

// Sort the graph
sorted := g.TopologicalSort()
var sortedCerts []x509.Certificate

// If sorted only has one element that must be the leaf and we have no chain to return
if len(sorted) == 1 {
log.Print("No chain detected in input")
return ""
for i := range sorted {
cert := (*sorted[i].Value).(x509.Certificate)
sortedCerts = append(sortedCerts, cert)
}

// Construct the sorted chain PEM block
var chainPem bytes.Buffer

// If sorted len is greater than 1 we have a chain to parse
// Check if we want just the issuers or the full chain
issuers := sorted
if justIssuers {
issuers = sorted[1:]
// If we only have the leaf cert there are no issuers to return
if len(sortedCerts) <= 1 {
return nil
} else {
return sortedCerts[1:]
}
}

return sortedCerts
}

// Attempts to turn String data into array of pem.Block
func stringToPemBlocks(data string) []*pem.Block {
// Build an array of pem.Block
var blocks []*pem.Block
rest := []byte(data)
for {
var block *pem.Block
block, rest = pem.Decode(rest)
if block == nil {
break
}
blocks = append(blocks, block)
}
return blocks
}

// Attempts to find Private key in array of pem.Block and return it as PEM Encoded PKCS8 String
func findPrivateKeyInPemBlocks(blocks []*pem.Block ) string {
var keyBuffer bytes.Buffer
//Find the private key from all the blocks
for _, block := range blocks {
// Private Key?
if block.Type == "PRIVATE KEY" || strings.HasSuffix(block.Type, " PRIVATE KEY") {
key, err := parsePrivateKey(block.Bytes)
if err != nil {
panic(err)
}

// Force it to pkcs8 for consistency
privBytes, err := x509.MarshalPKCS8PrivateKey(key)
if err != nil {
panic(err)
}

// Encode the pkcs8 object as PEM
if err := pem.Encode(&keyBuffer, &pem.Block{Type: "PRIVATE KEY", Bytes: privBytes}); err != nil {
panic(fmt.Sprintf("Failed to write data: %s", err))
}
break
}
}
return keyBuffer.String()
}

// https://golang.org/src/crypto/tls/tls.go?#L370
func parsePrivateKey(der []byte) (crypto.PrivateKey, error) {
if key, err := x509.ParsePKCS1PrivateKey(der); err == nil {
return key, nil
}

if key, err := x509.ParsePKCS8PrivateKey(der); err == nil {
switch key := key.(type) {
case *rsa.PrivateKey, *ecdsa.PrivateKey, ed25519.PrivateKey:
return key, nil
default:
return nil, errors.New("tls: found unknown private key type in PKCS#8 wrapping")
}
}

if key, err := x509.ParseECPrivateKey(der); err == nil {
return key, nil
}

return nil, errors.New("tls: failed to parse private key")
}

// Attempts to find leaf certificate in array of pem.Block data and return as PEM Encoded x509 Certificate
func findLeafCertInPemBlocks(blocks []*pem.Block) string {
var certs []*x509.Certificate
//Find all the Certificate blocks
for _, block := range blocks {
// Private Key?
if block.Type == "CERTIFICATE" {
cert, err := x509.ParseCertificate(block.Bytes)

if err != nil {
panic(err)
}

certs = append(certs, cert)
}
}

// Sort the certs
sortedCerts := SortedChain(certs, false)

// PEM Encode first cert in sortedCerts
var certBuffer bytes.Buffer
if err := pem.Encode(&certBuffer, &pem.Block{Type: "CERTIFICATE", Bytes: sortedCerts[0].Raw}); err != nil {
panic(fmt.Sprintf("Failed to write data: %s", err))
}

return certBuffer.String()
}

// Attempts to find chain in array of pem.Block and return as PEM Encoded Sorted Chain of x509 Certificates
func findChainInPemBlocks(blocks []*pem.Block, justIssuers bool) string {
var certs []*x509.Certificate
//Find all the Certificate blocks
for _, block := range blocks {
// Certificate?
if block.Type == "CERTIFICATE" {
cert, err := x509.ParseCertificate(block.Bytes)

if err != nil {
panic(err)
}

certs = append(certs, cert)
}
}

// Sort the certs
sortedCerts := SortedChain(certs, justIssuers)

for i := range issuers {
if err := pem.Encode(&chainPem, &pem.Block{Type: "CERTIFICATE", Bytes: (*issuers[i].Value).(x509.Certificate).Raw}); err != nil {
// PEM Encode all the certs
var certBuffer bytes.Buffer
for i := range sortedCerts {
if err := pem.Encode(&certBuffer, &pem.Block{Type: "CERTIFICATE", Bytes: sortedCerts[i].Raw}); err != nil {
panic(fmt.Sprintf("Failed to write data: %s", err))
}
}

return chainPem.String()
return certBuffer.String()
}
8 changes: 7 additions & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ module github.com/covermymeds/azure-key-vault-agent
go 1.13

require (
cloud.google.com/go v0.63.0 // indirect
github.com/Azure/azure-sdk-for-go v37.1.0+incompatible
github.com/Azure/go-autorest/autorest v0.9.3
github.com/Azure/go-autorest/autorest/adal v0.8.1
Expand All @@ -15,14 +16,19 @@ require (
github.com/go-playground/validator/v10 v10.1.0
github.com/gobuffalo/envy v1.8.1
github.com/google/uuid v1.1.1 // indirect
github.com/googleapis/gax-go v1.0.3 // indirect
github.com/huandu/xstrings v1.3.2 // indirect
github.com/imdario/mergo v0.3.10 // indirect
github.com/jpillora/backoff v1.0.0
github.com/julienschmidt/httprouter v1.3.0 // indirect
github.com/luci/luci-go v0.0.0-20200220034857-6a27eb3e318d
github.com/mitchellh/copystructure v1.0.0 // indirect
github.com/mitchellh/go-homedir v1.1.0 // indirect
github.com/sirupsen/logrus v1.4.2
github.com/smartystreets/goconvey v1.6.4 // indirect
github.com/twmb/algoimpl v0.0.0-20170717182524-076353e90b94
golang.org/x/crypto v0.0.0-20200117160349-530e935923ad
go.chromium.org/luci v0.0.0-20200814170619-378a717791e3 // indirect
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9
google.golang.org/genproto v0.0.0-20200814021100-8c09557e8a18 // indirect
gopkg.in/yaml.v2 v2.3.0
)
Loading

0 comments on commit 23065b7

Please sign in to comment.