2022-03-10 10:44:48 +01:00
|
|
|
package dtls
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2024-01-19 00:24:54 +05:30
|
|
|
func (c *handshakeConfig) getCertificate(serverName string) (*tls.Certificate, error) {
|
|
|
|
c.mu.Lock()
|
|
|
|
defer c.mu.Unlock()
|
2024-01-03 17:57:33 -04:00
|
|
|
|
2024-01-19 00:24:54 +05:30
|
|
|
if c.nameToCertificate == nil {
|
|
|
|
nameToCertificate := make(map[string]*tls.Certificate)
|
|
|
|
for i := range c.localCertificates {
|
|
|
|
cert := &c.localCertificates[i]
|
|
|
|
x509Cert := cert.Leaf
|
|
|
|
if x509Cert == nil {
|
|
|
|
var parseErr error
|
|
|
|
x509Cert, parseErr = x509.ParseCertificate(cert.Certificate[0])
|
|
|
|
if parseErr != nil {
|
|
|
|
continue
|
|
|
|
}
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
2024-01-19 00:24:54 +05:30
|
|
|
if len(x509Cert.Subject.CommonName) > 0 {
|
|
|
|
nameToCertificate[strings.ToLower(x509Cert.Subject.CommonName)] = cert
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
2024-01-19 00:24:54 +05:30
|
|
|
for _, san := range x509Cert.DNSNames {
|
|
|
|
nameToCertificate[strings.ToLower(san)] = cert
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
}
|
2024-01-19 00:24:54 +05:30
|
|
|
c.nameToCertificate = nameToCertificate
|
2022-03-10 10:44:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(c.localCertificates) == 0 {
|
|
|
|
return nil, errNoCertificates
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(c.localCertificates) == 1 {
|
|
|
|
// There's only one choice, so no point doing any work.
|
|
|
|
return &c.localCertificates[0], nil
|
|
|
|
}
|
|
|
|
|
2024-01-19 00:24:54 +05:30
|
|
|
if len(serverName) == 0 {
|
2022-03-10 10:44:48 +01:00
|
|
|
return &c.localCertificates[0], nil
|
|
|
|
}
|
|
|
|
|
2024-01-19 00:24:54 +05:30
|
|
|
name := strings.TrimRight(strings.ToLower(serverName), ".")
|
2022-03-10 10:44:48 +01:00
|
|
|
|
|
|
|
if cert, ok := c.nameToCertificate[name]; ok {
|
|
|
|
return cert, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// try replacing labels in the name with wildcards until we get a
|
|
|
|
// match.
|
|
|
|
labels := strings.Split(name, ".")
|
|
|
|
for i := range labels {
|
|
|
|
labels[i] = "*"
|
|
|
|
candidate := strings.Join(labels, ".")
|
|
|
|
if cert, ok := c.nameToCertificate[candidate]; ok {
|
|
|
|
return cert, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If nothing matches, return the first certificate.
|
|
|
|
return &c.localCertificates[0], nil
|
|
|
|
}
|