influxdb/services/httpd/service.go

171 lines
5.2 KiB
Go
Raw Normal View History

2016-02-10 18:30:52 +00:00
package httpd // import "github.com/influxdata/influxdb/services/httpd"
2015-05-28 21:47:47 +00:00
import (
2015-07-17 23:54:06 +00:00
"crypto/tls"
2015-09-04 17:47:14 +00:00
"expvar"
2015-05-28 21:47:47 +00:00
"fmt"
"io"
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"
"time"
2015-09-04 17:47:14 +00:00
"github.com/influxdata/influxdb"
2015-09-04 17:47:14 +00:00
)
// statistics gathered by the httpd package.
const (
statRequest = "req" // Number of HTTP requests served
statCQRequest = "cqReq" // Number of CQ-execute requests served
statQueryRequest = "queryReq" // Number of query requests served
statWriteRequest = "writeReq" // Number of write requests serverd
statPingRequest = "pingReq" // Number of ping requests served
statStatusRequest = "statusReq" // Number of status requests served
statWriteRequestBytesReceived = "writeReqBytes" // Sum of all bytes in write requests
statQueryRequestBytesTransmitted = "queryRespBytes" // Sum of all bytes returned in query reponses
statPointsWrittenOK = "pointsWrittenOK" // Number of points written OK
statPointsWrittenFail = "pointsWrittenFail" // Number of points that failed to be written
statAuthFail = "authFail" // Number of authentication failures
statRequestDuration = "reqDurationNs" // Number of (wall-time) nanoseconds spent inside requests
statQueryRequestDuration = "queryReqDurationNs" // Number of (wall-time) nanoseconds spent inside query requests
statWriteRequestDuration = "writeReqDurationNs" // Number of (wall-time) nanoseconds spent inside write requests
statRequestsActive = "reqActive" // Number of currently active requests
statWriteRequestsActive = "writeReqActive" // Number of currently active write requests
statClientError = "clientError" // Number of HTTP responses due to client error
statServerError = "serverError" // Number of HTTP responses due to server error
2015-05-28 21:47:47 +00:00
)
2015-05-29 19:50:05 +00:00
// Service manages the listener and handler for an HTTP endpoint.
type Service struct {
2015-07-17 23:54:06 +00:00
ln net.Listener
addr string
https bool
cert string
key string
limit int
2015-07-17 23:54:06 +00:00
err chan error
2015-05-28 21:47:47 +00:00
2015-06-01 17:20:57 +00:00
Handler *Handler
2015-09-04 17:47:14 +00:00
Logger *log.Logger
statMap *expvar.Map
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-09-04 17:47:14 +00:00
// Configure expvar monitoring. It's OK to do this even if the service fails to open and
// should be done before any data could arrive for the service.
key := strings.Join([]string{"httpd", c.BindAddress}, ":")
tags := map[string]string{"bind": c.BindAddress}
statMap := influxdb.NewStatistics(key, "httpd", tags)
2015-06-01 17:20:57 +00:00
s := &Service{
2016-05-12 01:42:45 +00:00
addr: c.BindAddress,
https: c.HTTPSEnabled,
cert: c.HTTPSCertificate,
key: c.HTTPSPrivateKey,
limit: c.MaxConnectionLimit,
2016-05-12 01:42:45 +00:00
err: make(chan error),
Handler: NewHandler(c, statMap),
Logger: log.New(os.Stderr, "[httpd] ", log.LstdFlags),
2015-05-28 21:47:47 +00:00
}
if s.key == "" {
s.key = s.cert
}
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 {
s.Logger.Println("Starting HTTP service")
2016-05-12 01:42:45 +00:00
s.Logger.Println("Authentication enabled:", s.Handler.Config.AuthEnabled)
2015-07-22 00:53:12 +00:00
2015-05-28 21:47:47 +00:00
// Open listener.
2015-07-17 23:54:06 +00:00
if s.https {
cert, err := tls.LoadX509KeyPair(s.cert, s.key)
2015-07-17 23:54:06 +00:00
if err != nil {
2015-07-23 21:50:45 +00:00
return err
2015-07-17 23:54:06 +00:00
}
listener, err := tls.Listen("tcp", s.addr, &tls.Config{
Certificates: []tls.Certificate{cert},
})
if err != nil {
2015-07-23 21:50:45 +00:00
return err
2015-07-17 23:54:06 +00:00
}
2015-06-01 17:20:57 +00:00
s.Logger.Println("Listening on HTTPS:", listener.Addr().String())
2015-07-17 23:54:06 +00:00
s.ln = listener
} else {
2015-07-17 23:54:06 +00:00
listener, err := net.Listen("tcp", s.addr)
if err != nil {
return err
}
s.Logger.Println("Listening on HTTP:", listener.Addr().String())
2015-07-17 23:54:06 +00:00
s.ln = listener
}
2015-05-28 21:47:47 +00:00
// Enforce a connection limit if one has been given.
if s.limit > 0 {
s.ln = LimitListener(s.ln, s.limit)
}
// wait for the listeners to start
timeout := time.Now().Add(time.Second)
for {
if s.ln.Addr() != nil {
break
}
if time.Now().After(timeout) {
return fmt.Errorf("unable to open without http listener running")
}
time.Sleep(10 * time.Millisecond)
}
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
}
// SetLogOutput sets the writer to which all logs are written. It must not be
// called after Open is called.
func (s *Service) SetLogOutput(w io.Writer) {
l := log.New(w, "[httpd] ", log.LstdFlags)
2015-06-03 15:58:39 +00:00
s.Logger = l
s.Handler.Logger = l
2015-06-03 15:58:39 +00:00
}
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
}
}