influxdb/services/httpd/service.go

88 lines
1.7 KiB
Go
Raw Normal View History

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.
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,
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
}
}