influxdb/httpd/service.go

76 lines
1.6 KiB
Go

package httpd
import (
"fmt"
"net"
"net/http"
"strings"
)
// HTTPService manages the listener and handler for an HTTP endpoint.
type HTTPService struct {
listener net.Listener
addr string
err chan error
Handler Handler
}
// NewHTTPService returns a new instance of HTTPService.
func NewHTTPService(c *Config) *HTTPService {
return &HTTPService{
addr: c.BindAddress,
err: make(chan error),
}
}
// Open starts the service
func (s *HTTPService) Open() error {
// Open listener.
listener, err := net.Listen("tcp", s.addr)
if err != nil {
return err
}
s.listener = listener
// Begin listening for requests in a separate goroutine.
go s.serve()
return nil
}
// Close closes the underlying listener.
func (s *HTTPService) Close() error {
if s.listener != nil {
return s.listener.Close()
}
return nil
}
// Err returns a channel for fatal errors that occur on the listener.
func (s *HTTPService) Err() <-chan error { return s.err }
// Addr returns the listener's address. Returns nil if listener is closed.
func (s *HTTPService) Addr() net.Addr {
if s.listener != nil {
return s.listener.Addr()
}
return nil
}
// serve serves the handler from the listener.
func (s *HTTPService) serve() {
// The listener was closed so exit
// See https://github.com/golang/go/issues/4373
err := http.Serve(s.listener, &s.Handler)
if err != nil && !strings.Contains(err.Error(), "closed") {
s.err <- fmt.Errorf("listener failed: addr=%s, err=%s", s.Addr(), err)
}
}
type Config struct {
BindAddress string `toml:"bind-address"`
Port int `toml:"port"`
LogEnabled bool `toml:"log-enabled"`
WriteTracing bool `toml:"write-tracing"`
}