2015-05-28 21:47:47 +00:00
|
|
|
package httpd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2015-06-01 17:20:57 +00:00
|
|
|
"log"
|
2015-05-28 21:47:47 +00:00
|
|
|
"net"
|
|
|
|
"net/http"
|
2015-06-01 17:20:57 +00:00
|
|
|
"os"
|
2015-05-28 21:47:47 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2015-05-29 19:50:05 +00:00
|
|
|
// Service manages the listener and handler for an HTTP endpoint.
|
|
|
|
type Service struct {
|
2015-06-01 17:20:57 +00:00
|
|
|
ln net.Listener
|
|
|
|
addr string
|
|
|
|
err chan error
|
2015-05-28 21:47:47 +00:00
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
Handler *Handler
|
|
|
|
|
|
|
|
Logger *log.Logger
|
2015-05-28 21:47:47 +00:00
|
|
|
}
|
|
|
|
|
2015-05-29 19:50:05 +00:00
|
|
|
// NewService returns a new instance of Service.
|
2015-05-29 20:12:00 +00:00
|
|
|
func NewService(c Config) *Service {
|
2015-06-01 17:20:57 +00:00
|
|
|
s := &Service{
|
2015-05-29 15:53:33 +00:00
|
|
|
addr: c.BindAddress,
|
2015-05-28 21:47:47 +00:00
|
|
|
err: make(chan error),
|
2015-06-01 17:20:57 +00:00
|
|
|
Handler: NewHandler(
|
|
|
|
c.AuthEnabled,
|
|
|
|
c.LogEnabled,
|
2015-06-17 12:55:34 +00:00
|
|
|
c.WriteTracing,
|
2015-06-01 17:20:57 +00:00
|
|
|
),
|
|
|
|
Logger: log.New(os.Stderr, "[httpd] ", log.LstdFlags),
|
2015-05-28 21:47:47 +00:00
|
|
|
}
|
2015-06-01 17:20:57 +00:00
|
|
|
s.Handler.Logger = s.Logger
|
|
|
|
return s
|
2015-05-28 21:47:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Open starts the service
|
2015-05-29 19:50:05 +00:00
|
|
|
func (s *Service) Open() error {
|
2015-05-28 21:47:47 +00:00
|
|
|
// Open listener.
|
2015-06-01 17:20:57 +00:00
|
|
|
ln, err := net.Listen("tcp", s.addr)
|
2015-05-28 21:47:47 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-06-01 17:20:57 +00:00
|
|
|
s.ln = ln
|
|
|
|
|
|
|
|
s.Logger.Println("listening on HTTP:", ln.Addr().String())
|
2015-05-28 21:47:47 +00:00
|
|
|
|
|
|
|
// Begin listening for requests in a separate goroutine.
|
2015-05-29 15:53:33 +00:00
|
|
|
go s.serve()
|
2015-05-28 21:47:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the underlying listener.
|
2015-05-29 19:50:05 +00:00
|
|
|
func (s *Service) Close() error {
|
2015-06-01 17:20:57 +00:00
|
|
|
if s.ln != nil {
|
|
|
|
return s.ln.Close()
|
2015-05-28 21:47:47 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-06-03 15:58:39 +00:00
|
|
|
// SetLogger sets the internal logger to the logger passed in.
|
|
|
|
func (s *Service) SetLogger(l *log.Logger) {
|
|
|
|
s.Logger = l
|
|
|
|
}
|
|
|
|
|
2015-05-28 21:47:47 +00:00
|
|
|
// Err returns a channel for fatal errors that occur on the listener.
|
2015-05-29 19:50:05 +00:00
|
|
|
func (s *Service) Err() <-chan error { return s.err }
|
2015-05-28 21:47:47 +00:00
|
|
|
|
|
|
|
// Addr returns the listener's address. Returns nil if listener is closed.
|
2015-05-29 19:50:05 +00:00
|
|
|
func (s *Service) Addr() net.Addr {
|
2015-06-01 17:20:57 +00:00
|
|
|
if s.ln != nil {
|
|
|
|
return s.ln.Addr()
|
2015-05-28 21:47:47 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// serve serves the handler from the listener.
|
2015-05-29 19:50:05 +00:00
|
|
|
func (s *Service) serve() {
|
2015-05-28 21:47:47 +00:00
|
|
|
// The listener was closed so exit
|
|
|
|
// See https://github.com/golang/go/issues/4373
|
2015-06-01 17:20:57 +00:00
|
|
|
err := http.Serve(s.ln, s.Handler)
|
2015-05-28 21:47:47 +00:00
|
|
|
if err != nil && !strings.Contains(err.Error(), "closed") {
|
2015-05-29 15:53:33 +00:00
|
|
|
s.err <- fmt.Errorf("listener failed: addr=%s, err=%s", s.Addr(), err)
|
2015-05-28 21:47:47 +00:00
|
|
|
}
|
|
|
|
}
|