3
0
corteza/pkg/api/server.go

132 lines
2.9 KiB
Go

package api
import (
"context"
"net"
"net/http"
"github.com/go-chi/chi"
"github.com/go-chi/chi/middleware"
"github.com/titpetric/factory/resputil"
"go.uber.org/zap"
"github.com/cortezaproject/corteza-server/pkg/app/options"
"github.com/cortezaproject/corteza-server/pkg/auth"
"github.com/cortezaproject/corteza-server/pkg/version"
)
type (
server struct {
log *zap.Logger
httpOpt options.HTTPServerOpt
waitForOpt options.WaitForOpt
endpoints []func(r chi.Router)
}
)
func New(log *zap.Logger, httpOpt options.HTTPServerOpt, waitForOpt options.WaitForOpt) *server {
return &server{
endpoints: make([]func(r chi.Router), 0),
log: log.Named("http"),
httpOpt: httpOpt,
waitForOpt: waitForOpt,
}
}
func (s *server) MountRoutes(mm ...func(chi.Router)) {
s.endpoints = append(s.endpoints, mm...)
}
func (s server) Serve(ctx context.Context) {
s.log.Info("Starting HTTP server with REST API", zap.String("address", s.httpOpt.Addr))
// configure resputil options
resputil.SetConfig(resputil.Options{
Trace: s.httpOpt.Tracing,
Logger: func(err error) {
// @todo: error logging
},
})
listener, err := net.Listen("tcp", s.httpOpt.Addr)
if err != nil {
s.log.Error("Can not start server", zap.Error(err))
return
}
router := chi.NewRouter()
// Base middleware, CORS, RealIP, RequestID, context-logger
router.Use(BaseMiddleware(s.log)...)
// Logging request if enabled
if s.httpOpt.LogRequest {
router.Use(LogRequest)
}
// Logging response if enabled
if s.httpOpt.LogResponse {
router.Use(LogResponse)
}
// Handle panic (sets 500 server error headers)
router.Use(handlePanic)
// Reports error to Sentry if enabled
if s.httpOpt.EnablePanicReporting {
router.Use(sentryMiddleware())
}
// Metrics tracking middleware
if s.httpOpt.EnableMetrics {
router.Use(metricsMiddleware(s.httpOpt.MetricsServiceLabel))
}
router.Group(func(r chi.Router) {
r.Use(
auth.DefaultJwtHandler.HttpVerifier(),
auth.DefaultJwtHandler.HttpAuthenticator(),
)
for _, mountRoutes := range s.endpoints {
mountRoutes(r)
}
})
if s.httpOpt.EnableMetrics {
metricsMount(router, s.httpOpt.MetricsUsername, s.httpOpt.MetricsPassword)
}
if s.httpOpt.EnableDebugRoute {
s.log.Debug("profiler: /__profiler", zap.Error(err))
router.Mount("/__profiler", middleware.Profiler())
s.log.Debug("list of routes: /__routes", zap.Error(err))
router.Get("/__routes", debugRoutes(router))
s.log.Debug("eventbus handlers: /__eventbus", zap.Error(err))
router.Get("/__eventbus", debugEventbus())
s.log.Debug("corredor service: /__corredor", zap.Error(err))
router.Get("/__corredor", debugCorredor())
}
if s.httpOpt.EnableVersionRoute {
router.Get("/version", version.HttpHandler)
}
go func() {
err = http.Serve(listener, router)
}()
<-ctx.Done()
if err == nil {
err = ctx.Err()
if err == context.Canceled {
err = nil
}
}
s.log.Info("Server stopped", zap.Error(err))
}