2015-12-15 22:01:03 +00:00
|
|
|
package meta
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Service struct {
|
|
|
|
config *Config
|
2015-12-17 12:53:10 +00:00
|
|
|
handler *handler
|
2015-12-15 22:01:03 +00:00
|
|
|
ln net.Listener
|
|
|
|
raftAddr string
|
|
|
|
httpAddr string
|
|
|
|
https bool
|
|
|
|
cert string
|
|
|
|
err chan error
|
2015-12-18 22:31:24 +00:00
|
|
|
Logger *log.Logger
|
|
|
|
store *store
|
2015-12-15 22:01:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewService returns a new instance of Service.
|
|
|
|
func NewService(c *Config) *Service {
|
|
|
|
s := &Service{
|
|
|
|
config: c,
|
|
|
|
raftAddr: c.RaftBindAddress,
|
|
|
|
httpAddr: c.HTTPdBindAddress,
|
|
|
|
https: c.HTTPSEnabled,
|
|
|
|
cert: c.HTTPSCertificate,
|
|
|
|
err: make(chan error),
|
|
|
|
Logger: log.New(os.Stderr, "[meta] ", log.LstdFlags),
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open starts the service
|
|
|
|
func (s *Service) Open() error {
|
|
|
|
s.Logger.Println("Starting meta service")
|
|
|
|
|
|
|
|
// Open the store
|
2015-12-17 12:53:10 +00:00
|
|
|
store := newStore(s.config)
|
2015-12-23 22:20:02 +00:00
|
|
|
// Set the peers from the config
|
|
|
|
store.peers = s.config.Peers
|
2015-12-17 12:53:10 +00:00
|
|
|
s.store = store
|
2015-12-19 17:10:33 +00:00
|
|
|
if err := s.store.open(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-12-15 22:01:03 +00:00
|
|
|
|
2015-12-17 12:53:10 +00:00
|
|
|
handler := newHandler(s.config)
|
|
|
|
handler.logger = s.Logger
|
|
|
|
handler.store = store
|
|
|
|
s.handler = handler
|
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
|
|
|
|
}
|
|
|
|
s.httpAddr = s.ln.Addr().String()
|
|
|
|
|
|
|
|
// Begin listening for requests in a separate goroutine.
|
|
|
|
go s.serve()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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.Addr(), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the underlying listener.
|
|
|
|
func (s *Service) Close() error {
|
|
|
|
if s.ln != nil {
|
|
|
|
return s.ln.Close()
|
|
|
|
}
|
|
|
|
|
2015-12-18 22:31:24 +00:00
|
|
|
if err := s.store.close(); err != nil {
|
2015-12-15 22:01:03 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// URL returns the HTTP URL.
|
|
|
|
func (s *Service) URL() string {
|
|
|
|
return s.httpAddr
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
|
|
|
// Addr returns the listener's address. Returns nil if listener is closed.
|
|
|
|
func (s *Service) Addr() net.Addr {
|
|
|
|
if s.ln != nil {
|
|
|
|
return s.ln.Addr()
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|