when we get a tls connection with an unrecognized sni hostname/ip, cause an alert "unrecognized name" rather than "internal error"

more helpful error for users trying to debug whats going on.

problem pointed out by arnt, thanks!
This commit is contained in:
Mechiel Lukkien 2024-04-08 14:22:52 +02:00
parent ecf6163409
commit 89a9a8bc97
No known key found for this signature in database

View file

@ -161,6 +161,11 @@ func Load(name, acmeDir, contactEmail, directoryURL string, eabKeyID string, eab
loggingGetCertificate := func(hello *tls.ClientHelloInfo) (*tls.Certificate, error) { loggingGetCertificate := func(hello *tls.ClientHelloInfo) (*tls.Certificate, error) {
log := mlog.New("autotls", nil).WithContext(hello.Context()) log := mlog.New("autotls", nil).WithContext(hello.Context())
// We handle missing invalid hostnames/ip's by returning a nil certificate and nil
// error, which crypto/tls turns into a TLS alert "unrecognized name", which can be
// interpreted by clients as a hint that they are using the wrong hostname, or a
// certificate is missing.
// Handle missing SNI to prevent logging an error below. // Handle missing SNI to prevent logging an error below.
// At startup, during config initialization, we already adjust the tls config to // At startup, during config initialization, we already adjust the tls config to
// inject the listener hostname if there isn't one in the TLS client hello. This is // inject the listener hostname if there isn't one in the TLS client hello. This is
@ -168,17 +173,16 @@ func Load(name, acmeDir, contactEmail, directoryURL string, eabKeyID string, eab
// verification of the certificate. // verification of the certificate.
if hello.ServerName == "" { if hello.ServerName == "" {
log.Debug("tls request without sni servername, rejecting", slog.Any("localaddr", hello.Conn.LocalAddr()), slog.Any("supportedprotos", hello.SupportedProtos)) log.Debug("tls request without sni servername, rejecting", slog.Any("localaddr", hello.Conn.LocalAddr()), slog.Any("supportedprotos", hello.SupportedProtos))
return nil, fmt.Errorf("sni server name required") return nil, nil
} }
cert, err := m.GetCertificate(hello) cert, err := m.GetCertificate(hello)
if err != nil { if err != nil && errors.Is(err, errHostNotAllowed) {
if errors.Is(err, errHostNotAllowed) {
log.Debugx("requesting certificate", err, slog.String("host", hello.ServerName)) log.Debugx("requesting certificate", err, slog.String("host", hello.ServerName))
} else { return nil, nil
} else if err != nil {
log.Errorx("requesting certificate", err, slog.String("host", hello.ServerName)) log.Errorx("requesting certificate", err, slog.String("host", hello.ServerName))
} }
}
return cert, err return cert, err
} }