Commit 2fb4810c authored by Matthew Holt's avatar Matthew Holt

Fixed racy error reporting at server startup

Previously, if a listener fails to bind (for example), there was a race in caddy.go between unblocking the startup waitgroup and returning the error and putting it into errChan. Now, an error is returned directly into errChan and the closing of the startup waitgroup is defered until after that return takes place.
parent 411dd7df
...@@ -210,16 +210,10 @@ func startServers(groupings Group) error { ...@@ -210,16 +210,10 @@ func startServers(groupings Group) error {
wg.Add(1) wg.Add(1)
go func(s *server.Server, ln server.ListenerFile) { go func(s *server.Server, ln server.ListenerFile) {
defer wg.Done() defer wg.Done()
if ln != nil { if ln != nil {
err = s.Serve(ln) errChan <- s.Serve(ln)
} else { } else {
err = s.ListenAndServe() errChan <- s.ListenAndServe()
}
// "use of closed network connection" is normal if doing graceful shutdown...
if err != nil && !strings.Contains(err.Error(), "use of closed network connection") {
errChan <- err
} }
}(s, ln) }(s, ln)
...@@ -240,7 +234,10 @@ func startServers(groupings Group) error { ...@@ -240,7 +234,10 @@ func startServers(groupings Group) error {
// Return the first error, if any // Return the first error, if any
select { select {
case err := <-errChan: case err := <-errChan:
// "use of closed network connection" is normal if it was a graceful shutdown
if err != nil && !strings.Contains(err.Error(), "use of closed network connection") {
return err return err
}
default: default:
} }
......
...@@ -102,7 +102,7 @@ func New(addr string, configs []Config) (*Server, error) { ...@@ -102,7 +102,7 @@ func New(addr string, configs []Config) (*Server, error) {
func (s *Server) Serve(ln ListenerFile) error { func (s *Server) Serve(ln ListenerFile) error {
err := s.setup() err := s.setup()
if err != nil { if err != nil {
close(s.startChan) defer close(s.startChan) // MUST defer so error is properly reported, same with all cases in this file
return err return err
} }
return s.serve(ln) return s.serve(ln)
...@@ -112,7 +112,7 @@ func (s *Server) Serve(ln ListenerFile) error { ...@@ -112,7 +112,7 @@ func (s *Server) Serve(ln ListenerFile) error {
func (s *Server) ListenAndServe() error { func (s *Server) ListenAndServe() error {
err := s.setup() err := s.setup()
if err != nil { if err != nil {
close(s.startChan) defer close(s.startChan)
return err return err
} }
...@@ -130,7 +130,7 @@ func (s *Server) ListenAndServe() error { ...@@ -130,7 +130,7 @@ func (s *Server) ListenAndServe() error {
} }
} }
if !succeeded { if !succeeded {
close(s.startChan) defer close(s.startChan)
return err return err
} }
} }
...@@ -207,7 +207,7 @@ func serveTLSWithSNI(s *Server, ln net.Listener, tlsConfigs []TLSConfig) error { ...@@ -207,7 +207,7 @@ func serveTLSWithSNI(s *Server, ln net.Listener, tlsConfigs []TLSConfig) error {
config.Certificates[i], err = tls.LoadX509KeyPair(tlsConfig.Certificate, tlsConfig.Key) config.Certificates[i], err = tls.LoadX509KeyPair(tlsConfig.Certificate, tlsConfig.Key)
config.Certificates[i].OCSPStaple = tlsConfig.OCSPStaple config.Certificates[i].OCSPStaple = tlsConfig.OCSPStaple
if err != nil { if err != nil {
close(s.startChan) defer close(s.startChan)
return err return err
} }
} }
...@@ -222,7 +222,7 @@ func serveTLSWithSNI(s *Server, ln net.Listener, tlsConfigs []TLSConfig) error { ...@@ -222,7 +222,7 @@ func serveTLSWithSNI(s *Server, ln net.Listener, tlsConfigs []TLSConfig) error {
// TLS client authentication, if user enabled it // TLS client authentication, if user enabled it
err = setupClientAuth(tlsConfigs, config) err = setupClientAuth(tlsConfigs, config)
if err != nil { if err != nil {
close(s.startChan) defer close(s.startChan)
return err return err
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment