influxdb/services/meta/service.go

211 lines
4.3 KiB
Go
Raw Normal View History

2016-02-10 18:30:52 +00:00
package meta // import "github.com/influxdata/influxdb/services/meta"
2015-12-15 22:01:03 +00:00
import (
"crypto/tls"
"fmt"
"io/ioutil"
2015-12-15 22:01:03 +00:00
"log"
"net"
"net/http"
"os"
"strings"
"time"
"github.com/influxdata/influxdb"
)
const (
MuxHeader = 8
2015-12-15 22:01:03 +00:00
)
type Service struct {
RaftListener net.Listener
Version string
2015-12-15 22:01:03 +00:00
config *Config
2015-12-17 12:53:10 +00:00
handler *handler
2015-12-15 22:01:03 +00:00
ln net.Listener
httpAddr string
raftAddr string
2015-12-15 22:01:03 +00:00
https bool
cert string
err chan error
2015-12-18 22:31:24 +00:00
Logger *log.Logger
store *store
Node *influxdb.Node
2015-12-15 22:01:03 +00:00
}
// NewService returns a new instance of Service.
func NewService(c *Config) *Service {
2015-12-15 22:01:03 +00:00
s := &Service{
config: c,
httpAddr: c.HTTPBindAddress,
raftAddr: c.BindAddress,
2015-12-15 22:01:03 +00:00
https: c.HTTPSEnabled,
cert: c.HTTPSCertificate,
err: make(chan error),
}
if c.LoggingEnabled {
s.Logger = log.New(os.Stderr, "[meta] ", log.LstdFlags)
} else {
s.Logger = log.New(ioutil.Discard, "", 0)
}
2015-12-15 22:01:03 +00:00
return s
}
// Open starts the service
func (s *Service) Open() error {
s.Logger.Println("Starting meta service")
if s.RaftListener == nil {
panic("no raft listener set")
2015-12-19 17:10:33 +00:00
}
2015-12-15 22:01:03 +00:00
// Open listener.
if s.https {
cert, err := tls.LoadX509KeyPair(s.cert, s.cert)
if err != nil {
return err
}
listener, err := tls.Listen("tcp", s.httpAddr, &tls.Config{
Certificates: []tls.Certificate{cert},
})
if err != nil {
return err
}
s.Logger.Println("Listening on HTTPS:", listener.Addr().String())
s.ln = listener
} else {
listener, err := net.Listen("tcp", s.httpAddr)
if err != nil {
return err
}
s.Logger.Println("Listening on HTTP:", listener.Addr().String())
s.ln = listener
}
// wait for the listeners to start
timeout := time.Now().Add(raftListenerStartupTimeout)
for {
if s.ln.Addr() != nil && s.RaftListener.Addr() != nil {
break
}
if time.Now().After(timeout) {
return fmt.Errorf("unable to open without http listener running")
}
time.Sleep(10 * time.Millisecond)
}
2016-01-04 15:35:45 +00:00
var err error
if autoAssignPort(s.httpAddr) {
2016-01-04 15:35:45 +00:00
s.httpAddr, err = combineHostAndAssignedPort(s.ln, s.httpAddr)
}
if autoAssignPort(s.raftAddr) {
2016-01-04 15:35:45 +00:00
s.raftAddr, err = combineHostAndAssignedPort(s.RaftListener, s.raftAddr)
}
if err != nil {
return err
}
2015-12-15 22:01:03 +00:00
// Open the store. The addresses passed in are remotely accessible.
s.store = newStore(s.config, s.remoteAddr(s.httpAddr), s.remoteAddr(s.raftAddr))
s.store.node = s.Node
handler := newHandler(s.config, s)
handler.logger = s.Logger
handler.store = s.store
s.handler = handler
2015-12-15 22:01:03 +00:00
// Begin listening for requests in a separate goroutine.
go s.serve()
2016-01-25 04:56:06 +00:00
if err := s.store.open(s.RaftListener); err != nil {
return err
}
2015-12-15 22:01:03 +00:00
return nil
}
func (s *Service) remoteAddr(addr string) string {
hostname := s.config.RemoteHostname
if hostname == "" {
hostname = DefaultHostname
}
remote, err := DefaultHost(hostname, addr)
if err != nil {
return addr
}
return remote
}
2015-12-15 22:01:03 +00:00
// serve serves the handler from the listener.
func (s *Service) serve() {
// The listener was closed so exit
// See https://github.com/golang/go/issues/4373
2015-12-17 12:53:10 +00:00
err := http.Serve(s.ln, s.handler)
2015-12-15 22:01:03 +00:00
if err != nil && !strings.Contains(err.Error(), "closed") {
s.err <- fmt.Errorf("listener failed: addr=%s, err=%s", s.ln.Addr(), err)
2015-12-15 22:01:03 +00:00
}
}
// Close closes the underlying listener.
func (s *Service) Close() error {
2016-01-07 16:10:36 +00:00
if err := s.handler.Close(); err != nil {
return err
}
if err := s.store.close(); err != nil {
return err
}
2015-12-15 22:01:03 +00:00
if s.ln != nil {
if err := s.ln.Close(); err != nil {
return err
}
2015-12-15 22:01:03 +00:00
}
2016-01-07 16:10:36 +00:00
return nil
2015-12-15 22:01:03 +00:00
}
// HTTPAddr returns the bind address for the HTTP API
func (s *Service) HTTPAddr() string {
2015-12-15 22:01:03 +00:00
return s.httpAddr
}
// RaftAddr returns the bind address for the Raft TCP listener
func (s *Service) RaftAddr() string {
return s.raftAddr
}
2015-12-15 22:01:03 +00:00
// Err returns a channel for fatal errors that occur on the listener.
func (s *Service) Err() <-chan error { return s.err }
// SetLogger sets the internal logger to the logger passed in.
func (s *Service) SetLogger(l *log.Logger) {
s.Logger = l
}
func autoAssignPort(addr string) bool {
2016-01-04 15:35:45 +00:00
_, p, _ := net.SplitHostPort(addr)
return p == "0"
}
2016-01-04 15:35:45 +00:00
func combineHostAndAssignedPort(ln net.Listener, autoAddr string) (string, error) {
host, _, err := net.SplitHostPort(autoAddr)
if err != nil {
return "", err
}
_, port, err := net.SplitHostPort(ln.Addr().String())
if err != nil {
return "", err
}
return net.JoinHostPort(host, port), nil
2015-12-15 22:01:03 +00:00
}