gpt4 book ai didi

go - 有没有办法在不停机的情况下更新 net/http 服务器中的 TLS 证书?

转载 作者:IT王子 更新时间:2023-10-29 01:06:43 27 4
gpt4 key购买 nike

我有一个简单的 https 服务器,服务于这样一个简单的页面(为简洁起见,没有错误处理):

package main

import (
"crypto/tls"
"fmt"
"net/http"
)

func main() {
mux := http.NewServeMux()

mux.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
fmt.Fprintf(w, "hello!")
})

xcert, _ := tls.LoadX509KeyPair("cert1.crt", "key1.pem")

tlsConf := &tls.Config{
Certificates: []tls.Certificate{xcert},
}

srv := &http.Server{
Addr: ":https",
Handler: mux,
TLSConfig: tlsConf,
}

srv.ListenAndServeTLS("", "")
}

我想使用 Let's Encrypt通过 https 提供内容的 TLS 证书。我希望能够在不停机的情况下更新证书并更新服务器中的证书。

我尝试运行一个 goroutine 来更新 tlsConf:

go func(c *tls.Config) {
xcert, _ := tls.LoadX509KeyPair("cert2.crt", "key2.pem")

select {
case <-time.After(3 * time.Minute):
c.Certificates = []tls.Certificate{xcert}
c.BuildNameToCertificate()
fmt.Println("cert switched!")
}

}(tlsConf)

但是,这不起作用,因为服务器不会“读入”更改后的配置。无论如何要求服务器重新加载 TLSConfig

最佳答案

有:您可以使用tls.ConfigGetCertificate 成员而不是填充Certificates。首先,定义一个封装证书和重新加载功能的数据结构(在本例中接收到 SIGHUP 信号):

type keypairReloader struct {
certMu sync.RWMutex
cert *tls.Certificate
certPath string
keyPath string
}

func NewKeypairReloader(certPath, keyPath string) (*keypairReloader, error) {
result := &keypairReloader{
certPath: certPath,
keyPath: keyPath,
}
cert, err := tls.LoadX509KeyPair(certPath, keyPath)
if err != nil {
return nil, err
}
result.cert = &cert
go func() {
c := make(chan os.Signal, 1)
signal.Notify(c, syscall.SIGHUP)
for range c {
log.Printf("Received SIGHUP, reloading TLS certificate and key from %q and %q", *tlsCertPath, *tlsKeyPath)
if err := result.maybeReload(); err != nil {
log.Printf("Keeping old TLS certificate because the new one could not be loaded: %v", err)
}
}
}()
return result, nil
}

func (kpr *keypairReloader) maybeReload() error {
newCert, err := tls.LoadX509KeyPair(kpr.certPath, kpr.keyPath)
if err != nil {
return err
}
kpr.certMu.Lock()
defer kpr.certMu.Unlock()
kpr.cert = &newCert
return nil
}

func (kpr *keypairReloader) GetCertificateFunc() func(*tls.ClientHelloInfo) (*tls.Certificate, error) {
return func(clientHello *tls.ClientHelloInfo) (*tls.Certificate, error) {
kpr.certMu.RLock()
defer kpr.certMu.RUnlock()
return kpr.cert, nil
}
}

然后,在您的服务器代码中,使用:

kpr, err := NewKeypairReloader(*tlsCertPath, *tlsKeyPath)
if err != nil {
log.Fatal(err)
}
srv.TLSConfig.GetCertificate = kpr.GetCertificateFunc()

我最近implemented this pattern在 RobustIRC 中。

关于go - 有没有办法在不停机的情况下更新 net/http 服务器中的 TLS 证书?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37473201/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com