Conforme mencionado na yo.ian.gresposta de. Go 1.8 incluiu esta funcionalidade na biblioteca padrão.
Exemplo mínimo para Go 1.8+:
server := &http.Server{Addr: ":8080", Handler: handler}
go func() {
if err := server.ListenAndServe(); err != nil {
}
}()
stop := make(chan os.Signal, 1)
signal.Notify(stop, os.Interrupt)
<-stop
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
if err := server.Shutdown(ctx); err != nil {
}
Resposta original - Pre Go 1.8:
Com base na resposta de Uvelichitel .
Você pode criar sua própria versão do ListenAndServeque retorna um io.Closere não bloqueia.
func ListenAndServeWithClose(addr string, handler http.Handler) (io.Closer,error) {
var (
listener net.Listener
srvCloser io.Closer
err error
)
srv := &http.Server{Addr: addr, Handler: handler}
if addr == "" {
addr = ":http"
}
listener, err = net.Listen("tcp", addr)
if err != nil {
return nil, err
}
go func() {
err := srv.Serve(tcpKeepAliveListener{listener.(*net.TCPListener)})
if err != nil {
log.Println("HTTP Server Error - ", err)
}
}()
srvCloser = listener
return srvCloser, nil
}
Código completo disponível aqui .
O servidor HTTP fechará com o erro
accept tcp [::]:8080: use of closed network connection