influxdb/services/admin/service.go

112 lines
2.3 KiB
Go
Raw Normal View History

2015-05-28 21:47:47 +00:00
package admin
2015-05-29 19:50:05 +00:00
import (
"crypto/tls"
2015-05-29 19:50:05 +00:00
"fmt"
"log"
2015-05-29 19:50:05 +00:00
"net"
"net/http"
"os"
2015-05-29 19:50:05 +00:00
"strings"
2015-05-28 21:47:47 +00:00
2015-05-29 19:50:05 +00:00
// Register static assets via statik.
_ "github.com/influxdata/influxdb/statik"
2015-05-29 19:50:05 +00:00
"github.com/rakyll/statik/fs"
)
// Service manages the listener for an admin endpoint.
2015-05-28 21:47:47 +00:00
type Service struct {
2015-05-29 19:50:05 +00:00
listener net.Listener
addr string
https bool
cert string
2015-05-29 19:50:05 +00:00
err chan error
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 {
return &Service{
addr: c.BindAddress,
https: c.HTTPSEnabled,
cert: c.HTTPSCertificate,
err: make(chan error),
logger: log.New(os.Stderr, "[admin] ", log.LstdFlags),
2015-05-29 19:50:05 +00:00
}
2015-05-28 21:47:47 +00:00
}
2015-05-29 19:50:05 +00:00
// Open starts the service
2015-05-28 21:47:47 +00:00
func (s *Service) Open() error {
s.logger.Printf("Starting admin service")
2015-05-29 19:50:05 +00:00
// Open listener.
if s.https {
cert, err := tls.LoadX509KeyPair(s.cert, s.cert)
if err != nil {
2015-07-23 22:00:33 +00:00
return err
}
listener, err := tls.Listen("tcp", s.addr, &tls.Config{
Certificates: []tls.Certificate{cert},
})
if err != nil {
2015-07-23 22:00:33 +00:00
return err
}
s.logger.Println("Listening on HTTPS:", listener.Addr().String())
s.listener = listener
2015-07-23 22:00:33 +00:00
} else {
listener, err := net.Listen("tcp", s.addr)
if err != nil {
return err
}
s.logger.Println("Listening on HTTP:", listener.Addr().String())
s.listener = listener
2015-05-29 19:50:05 +00:00
}
// Begin listening for requests in a separate goroutine.
go s.serve()
return nil
}
2015-05-28 21:47:47 +00:00
2015-05-29 19:50:05 +00:00
// Close closes the underlying listener.
func (s *Service) Close() error {
if s.listener != nil {
return s.listener.Close()
2015-05-28 21:47:47 +00:00
}
2015-05-29 19:50:05 +00:00
return nil
2015-05-28 21:47:47 +00:00
}
// SetLogger sets the internal logger to the logger passed in.
func (s *Service) SetLogger(l *log.Logger) {
s.logger = l
}
2015-05-29 19:50:05 +00:00
// Err returns a channel for fatal errors that occur on the listener.
func (s *Service) Err() <-chan error { return s.err }
2015-05-28 21:47:47 +00:00
2015-05-29 19:50:05 +00:00
// Addr returns the listener's address. Returns nil if listener is closed.
func (s *Service) Addr() net.Addr {
if s.listener != nil {
return s.listener.Addr()
}
return nil
2015-05-28 21:47:47 +00:00
}
2015-05-29 19:50:05 +00:00
// serve serves the handler from the listener.
func (s *Service) serve() {
// Instantiate file system from embedded admin.
statikFS, err := fs.New()
if err != nil {
panic(err)
2015-05-28 21:47:47 +00:00
}
2015-05-29 19:50:05 +00:00
// Run file system handler on listener.
err = http.Serve(s.listener, http.FileServer(statikFS))
if err != nil && !strings.Contains(err.Error(), "closed") {
2015-05-30 14:57:27 +00:00
s.err <- fmt.Errorf("listener error: addr=%s, err=%s", s.Addr(), err)
2015-05-29 19:50:05 +00:00
}
}