micro/api/server/http/http.go

121 lines
2.3 KiB
Go
Raw Normal View History

2019-06-03 18:44:43 +01:00
// Package http provides a http server with features; acme, cors, etc
package http
import (
"crypto/tls"
"net"
"net/http"
"os"
"sync"
"github.com/gorilla/handlers"
"github.com/unistack-org/micro/v3/api/server"
"github.com/unistack-org/micro/v3/api/server/cors"
"github.com/unistack-org/micro/v3/logger"
2019-06-03 18:44:43 +01:00
)
type httpServer struct {
mux *http.ServeMux
opts server.Options
mtx sync.RWMutex
address string
exit chan chan error
}
2020-04-02 17:44:48 +01:00
func NewServer(address string, opts ...server.Option) server.Server {
var options server.Options
for _, o := range opts {
o(&options)
}
2019-06-03 18:44:43 +01:00
return &httpServer{
2020-04-02 17:44:48 +01:00
opts: options,
2019-06-03 18:44:43 +01:00
mux: http.NewServeMux(),
address: address,
exit: make(chan chan error),
}
}
func (s *httpServer) Address() string {
s.mtx.RLock()
defer s.mtx.RUnlock()
return s.address
}
func (s *httpServer) Init(opts ...server.Option) error {
for _, o := range opts {
o(&s.opts)
}
return nil
}
func (s *httpServer) Handle(path string, handler http.Handler) {
2020-04-19 00:41:03 +01:00
// TODO: move this stuff out to one place with ServeHTTP
2020-04-07 19:29:26 +01:00
// apply the wrappers, e.g. auth
for _, wrapper := range s.opts.Wrappers {
2020-04-19 00:41:03 +01:00
handler = wrapper(handler)
2020-04-07 19:29:26 +01:00
}
2020-03-04 11:40:53 +00:00
2020-04-19 00:41:03 +01:00
// wrap with cors
2020-03-04 11:40:53 +00:00
if s.opts.EnableCORS {
2020-04-19 00:41:03 +01:00
handler = cors.CombinedCORSHandler(handler)
2020-03-04 11:40:53 +00:00
}
2020-04-19 00:41:03 +01:00
// wrap with logger
handler = handlers.CombinedLoggingHandler(os.Stdout, handler)
s.mux.Handle(path, handler)
2019-06-03 18:44:43 +01:00
}
func (s *httpServer) Start() error {
var l net.Listener
var err error
2019-10-11 16:52:57 +01:00
if s.opts.EnableACME && s.opts.ACMEProvider != nil {
2019-06-03 18:44:43 +01:00
// should we check the address to make sure its using :443?
l, err = s.opts.ACMEProvider.Listen(s.opts.ACMEHosts...)
2019-06-03 18:44:43 +01:00
} else if s.opts.EnableTLS && s.opts.TLSConfig != nil {
l, err = tls.Listen("tcp", s.address, s.opts.TLSConfig)
} else {
// otherwise plain listen
l, err = net.Listen("tcp", s.address)
}
if err != nil {
return err
}
if logger.V(logger.InfoLevel, logger.DefaultLogger) {
logger.Infof("HTTP API Listening on %s", l.Addr().String())
}
2019-06-03 18:44:43 +01:00
s.mtx.Lock()
s.address = l.Addr().String()
s.mtx.Unlock()
go func() {
if err := http.Serve(l, s.mux); err != nil {
// temporary fix
logger.Error(err)
2019-06-03 18:44:43 +01:00
}
}()
go func() {
ch := <-s.exit
ch <- l.Close()
}()
return nil
}
func (s *httpServer) Stop() error {
ch := make(chan error)
s.exit <- ch
return <-ch
}
func (s *httpServer) String() string {
return "http"
}