-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathcertmap.go
143 lines (132 loc) · 4.01 KB
/
certmap.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
package sniproxy
import (
"crypto"
"crypto/ecdsa"
"crypto/rsa"
"crypto/tls"
"crypto/x509"
"encoding/pem"
"errors"
"fmt"
"log"
"os"
"strings"
"time"
keystore "github.com/pavel-v-chernykh/keystore-go/v4"
)
//certMap is a map of aliases and certificates in the form ("w.a.p:ECDSA",cert)
var certMap map[string]tls.Certificate
// keyStorePass is a pointer to the key store's password byte array
var keyStorePassBytes []byte
// keyStoreFile is a pointer to the keystore file's location string
var keyStoreFile *string
//loadCertMap loads the certificate map from the keystore object
func loadCertMap(filePtr *string, password []byte,
certMap *map[string]tls.Certificate) error {
f, err := os.Open(*filePtr)
if err != nil {
err = errors.New("loadKeyStore failed with error: " + fmt.Sprintf("%v", err))
return err
}
keyStore := keystore.New(keystore.WithCaseExactAliases())
err = keyStore.Load(f, password)
if err != nil {
err = errors.New("loadKeyStore failed with error: " + fmt.Sprintf("%v", err))
return err
}
defer clearOut(&keyStore)
// Check here if atleast "default" alias RSA or ECDSA exists
if !aliasExists(&keyStore, "default:RSA") &&
!aliasExists(&keyStore, "default:ECDSA") &&
!aliasExists(&keyStore, "default:DSA") {
//throw an error back that loadCertMap failed as there is no default cert
return fmt.Errorf("No certificate exists with \"default\" alias. " +
"Please load a cert with default alias into the keystore")
}
aliases := keyStore.Aliases()
for _, alias := range aliases {
entry, getPrivateKeyEntryErr := keyStore.GetPrivateKeyEntry(alias, password)
if getPrivateKeyEntryErr != nil {
return fmt.Errorf("Failed to fetch a private key entry for alias %v", alias)
}
certChain := entry.CertificateChain
var keyPEMBlock []byte
var keyDERBlock *pem.Block
cert := tls.Certificate{}
if len(certChain) == 0 {
log.Printf("Sniproxy error - PrivateKeyEntry for alias %s does not "+
"contain a certificate chain", alias)
} else {
for i := 0; i < len(certChain); i++ {
cert.Certificate = append(cert.Certificate, certChain[i].Content)
}
keyPEMBlock = entry.PrivateKey
keyDERBlock, _ = pem.Decode(keyPEMBlock)
cert.PrivateKey, err = parsePrivateKey(keyDERBlock.Bytes)
if err != nil {
log.Printf("Sniproxy error - Privatekey load failed for for alias %s",
alias)
} else {
if strings.HasPrefix(alias, "default") {
if strings.HasSuffix(alias, ":ECDSA") {
ECDSAdefaultExists = true
*ECDSAdefault = cert
} else {
RSAdefaultExists = true
*RSAdefault = cert
}
} else {
(*certMap)[alias] = cert
}
}
zeroBytes(keyPEMBlock)
clearOut(keyDERBlock)
}
clearOut(&cert)
}
f.Close()
return nil
}
//reloadCertMap reloads the certMap once every 6 hours
func reloadCertMap(filePtr *string, password []byte,
certMap *map[string]tls.Certificate, quit <-chan struct{}, n uint) {
ticker := time.NewTicker(time.Duration(n) * time.Second)
for {
select {
case <-quit:
ticker.Stop()
return
case <-ticker.C:
KSerror := loadCertMap(filePtr, password, certMap)
if KSerror != nil {
log.Printf("Sniproxy error - Keystore reload failed with error: %v", KSerror)
}
}
}
}
func aliasExists(keyStore *keystore.KeyStore, alias string) bool {
if exists := keyStore.IsPrivateKeyEntry(alias); exists {
return true
}
return false
}
//parsePrivateKey (borrowed shamelessly from golang's x509 package) converts
// a DER encoded Private key byte slice into a PrivateKey type.
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:
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")
}