Clash.Meta/component/tls/config.go

111 lines
2.7 KiB
Go
Raw Normal View History

2022-07-10 20:44:24 +08:00
package tls
import (
"bytes"
"crypto/sha256"
"crypto/tls"
"crypto/x509"
"encoding/hex"
"fmt"
"strings"
"sync"
"time"
)
var globalFingerprints [][32]byte
var mutex sync.Mutex
2022-07-10 20:44:24 +08:00
2022-07-11 12:37:27 +08:00
func verifyPeerCertificateAndFingerprints(fingerprints [][32]byte, insecureSkipVerify bool) func(rawCerts [][]byte, verifiedChains [][]*x509.Certificate) error {
return func(rawCerts [][]byte, verifiedChains [][]*x509.Certificate) error {
2022-07-11 12:37:27 +08:00
if insecureSkipVerify {
return nil
}
2022-07-10 20:44:24 +08:00
var preErr error
for i := range rawCerts {
rawCert := rawCerts[i]
cert, err := x509.ParseCertificate(rawCert)
if err == nil {
opts := x509.VerifyOptions{
CurrentTime: time.Now(),
2022-07-10 20:44:24 +08:00
}
if _, err := cert.Verify(opts); err == nil {
return nil
} else {
fingerprint := sha256.Sum256(cert.Raw)
for _, fp := range fingerprints {
if bytes.Equal(fingerprint[:], fp[:]) {
return nil
}
}
preErr = err
}
2022-07-10 20:44:24 +08:00
}
}
return preErr
}
2022-07-10 20:44:24 +08:00
}
func AddCertFingerprint(fingerprint string) error {
fpByte, err2 := convertFingerprint(fingerprint)
if err2 != nil {
return err2
}
mutex.Lock()
globalFingerprints = append(globalFingerprints, *fpByte)
mutex.Unlock()
return nil
}
func convertFingerprint(fingerprint string) (*[32]byte, error) {
2022-07-10 20:44:24 +08:00
fp := strings.Replace(fingerprint, ":", "", -1)
fpByte, err := hex.DecodeString(fp)
if err != nil {
return nil, err
2022-07-10 20:44:24 +08:00
}
if len(fpByte) != 32 {
return nil, fmt.Errorf("fingerprint string length error,need sha25 fingerprint")
2022-07-10 20:44:24 +08:00
}
return (*[32]byte)(fpByte), nil
2022-07-10 20:44:24 +08:00
}
func GetDefaultTLSConfig() *tls.Config {
2022-07-11 12:37:27 +08:00
return MixinTLSConfig(nil)
}
// GetTLSConfigWithSpecifiedFingerprint specified fingerprint
func GetTLSConfigWithSpecifiedFingerprint(tlsConfig *tls.Config, fingerprint string) (*tls.Config, error) {
if fingerprintBytes, err := convertFingerprint(fingerprint); err != nil {
return nil, err
} else {
if tlsConfig == nil {
return &tls.Config{
InsecureSkipVerify: true,
2022-07-11 12:37:27 +08:00
VerifyPeerCertificate: verifyPeerCertificateAndFingerprints([][32]byte{*fingerprintBytes}, false),
}, nil
} else {
2022-07-11 12:37:27 +08:00
tlsConfig.VerifyPeerCertificate = verifyPeerCertificateAndFingerprints([][32]byte{*fingerprintBytes}, tlsConfig.InsecureSkipVerify)
tlsConfig.InsecureSkipVerify = true
return tlsConfig, nil
}
}
2022-07-10 20:44:24 +08:00
}
func MixinTLSConfig(tlsConfig *tls.Config) *tls.Config {
if tlsConfig == nil {
2022-07-11 12:37:27 +08:00
return &tls.Config{
InsecureSkipVerify: true,
VerifyPeerCertificate: verifyPeerCertificateAndFingerprints(globalFingerprints, false),
}
2022-07-10 20:44:24 +08:00
}
2022-07-11 12:37:27 +08:00
tlsConfig.VerifyPeerCertificate = verifyPeerCertificateAndFingerprints(globalFingerprints, tlsConfig.InsecureSkipVerify)
2022-07-10 20:44:24 +08:00
tlsConfig.InsecureSkipVerify = true
return tlsConfig
}