2024-02-16 21:31:00 +00:00
|
|
|
package server
|
2024-02-10 11:59:58 +00:00
|
|
|
|
|
|
|
import (
|
2024-02-16 21:31:00 +00:00
|
|
|
"context"
|
2024-02-27 11:04:05 +00:00
|
|
|
"log/slog"
|
2024-02-10 11:59:58 +00:00
|
|
|
"net/http"
|
|
|
|
|
2024-02-27 11:04:05 +00:00
|
|
|
"code.tjo.space/mentos1386/zdravko/database"
|
2024-02-15 22:47:56 +00:00
|
|
|
"code.tjo.space/mentos1386/zdravko/internal/config"
|
2024-02-11 19:28:00 +00:00
|
|
|
"code.tjo.space/mentos1386/zdravko/internal/handlers"
|
2024-02-16 21:31:00 +00:00
|
|
|
"code.tjo.space/mentos1386/zdravko/internal/temporal"
|
2024-02-11 19:28:00 +00:00
|
|
|
"code.tjo.space/mentos1386/zdravko/web/static"
|
2024-02-20 10:24:04 +00:00
|
|
|
"code.tjo.space/mentos1386/zdravko/web/templates"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
|
|
"github.com/labstack/echo/v4/middleware"
|
2024-02-10 11:59:58 +00:00
|
|
|
)
|
|
|
|
|
2024-02-16 21:31:00 +00:00
|
|
|
type Server struct {
|
2024-02-27 11:04:05 +00:00
|
|
|
echo *echo.Echo
|
|
|
|
cfg *config.ServerConfig
|
|
|
|
logger *slog.Logger
|
2024-02-16 21:31:00 +00:00
|
|
|
}
|
|
|
|
|
2024-02-19 09:09:30 +00:00
|
|
|
func NewServer(cfg *config.ServerConfig) (*Server, error) {
|
2024-02-16 21:31:00 +00:00
|
|
|
return &Server{
|
2024-02-27 11:04:05 +00:00
|
|
|
cfg: cfg,
|
|
|
|
echo: echo.New(),
|
|
|
|
logger: slog.Default().WithGroup("server"),
|
2024-02-16 21:31:00 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) Name() string {
|
|
|
|
return "HTTP WEB and API Server"
|
|
|
|
}
|
2024-02-11 09:15:00 +00:00
|
|
|
|
2024-02-16 21:31:00 +00:00
|
|
|
func (s *Server) Start() error {
|
2024-02-21 09:41:20 +00:00
|
|
|
s.echo.Renderer = templates.NewTemplates()
|
2024-02-24 21:07:49 +00:00
|
|
|
//s.echo.Use(middleware.Logger())
|
2024-02-21 09:41:20 +00:00
|
|
|
s.echo.Use(middleware.Recover())
|
2024-02-10 11:59:58 +00:00
|
|
|
|
2024-02-27 11:04:05 +00:00
|
|
|
db, err := database.ConnectToDatabase(s.logger, s.cfg.DatabasePath)
|
2024-02-11 10:56:21 +00:00
|
|
|
if err != nil {
|
2024-02-16 21:31:00 +00:00
|
|
|
return err
|
2024-02-11 10:56:21 +00:00
|
|
|
}
|
|
|
|
|
2024-02-27 11:04:05 +00:00
|
|
|
temporalClient, err := temporal.ConnectServerToTemporal(s.logger, s.cfg)
|
2024-02-16 12:07:29 +00:00
|
|
|
if err != nil {
|
2024-02-16 21:31:00 +00:00
|
|
|
return err
|
2024-02-16 12:07:29 +00:00
|
|
|
}
|
|
|
|
|
2024-02-27 11:04:05 +00:00
|
|
|
h := handlers.NewBaseHandler(db, temporalClient, s.cfg, s.logger)
|
2024-02-11 19:28:00 +00:00
|
|
|
|
|
|
|
// Health
|
2024-02-21 09:41:20 +00:00
|
|
|
s.echo.GET("/health", func(c echo.Context) error {
|
2024-02-27 11:04:05 +00:00
|
|
|
err = db.Ping()
|
2024-02-11 19:28:00 +00:00
|
|
|
if err != nil {
|
2024-02-20 10:24:04 +00:00
|
|
|
return err
|
2024-02-11 19:28:00 +00:00
|
|
|
}
|
2024-02-20 10:24:04 +00:00
|
|
|
return c.JSON(http.StatusOK, map[string]string{"status": "ok"})
|
2024-02-11 19:28:00 +00:00
|
|
|
})
|
2024-02-11 10:56:21 +00:00
|
|
|
|
2024-02-10 11:59:58 +00:00
|
|
|
// Server static files
|
2024-02-21 09:41:20 +00:00
|
|
|
stat := s.echo.Group("/static")
|
2024-02-20 10:24:04 +00:00
|
|
|
stat.Use(middleware.StaticWithConfig(middleware.StaticConfig{
|
|
|
|
Filesystem: http.FS(static.Static),
|
|
|
|
}))
|
2024-02-10 11:59:58 +00:00
|
|
|
|
2024-02-20 10:24:04 +00:00
|
|
|
// Public
|
2024-02-21 09:41:20 +00:00
|
|
|
s.echo.GET("", h.Index)
|
2024-02-23 08:36:13 +00:00
|
|
|
s.echo.GET("/incidents", h.Incidents)
|
2024-02-11 22:48:37 +00:00
|
|
|
|
2024-02-19 14:23:17 +00:00
|
|
|
// Settings
|
2024-02-21 09:41:20 +00:00
|
|
|
settings := s.echo.Group("/settings")
|
2024-02-20 10:24:04 +00:00
|
|
|
settings.Use(h.Authenticated)
|
|
|
|
settings.GET("", h.SettingsOverviewGET)
|
2024-02-24 11:57:53 +00:00
|
|
|
settings.GET("/monitors", h.SettingsMonitorsGET)
|
|
|
|
settings.GET("/monitors/create", h.SettingsMonitorsCreateGET)
|
|
|
|
settings.POST("/monitors/create", h.SettingsMonitorsCreatePOST)
|
|
|
|
settings.GET("/monitors/:slug", h.SettingsMonitorsDescribeGET)
|
|
|
|
settings.POST("/monitors/:slug", h.SettingsMonitorsDescribePOST)
|
2024-02-24 21:07:49 +00:00
|
|
|
settings.GET("/worker-groups", h.SettingsWorkerGroupsGET)
|
|
|
|
settings.GET("/worker-groups/create", h.SettingsWorkerGroupsCreateGET)
|
|
|
|
settings.POST("/worker-groups/create", h.SettingsWorkerGroupsCreatePOST)
|
|
|
|
settings.GET("/worker-groups/:slug", h.SettingsWorkerGroupsDescribeGET)
|
2024-02-20 10:24:04 +00:00
|
|
|
settings.Match([]string{"GET", "HEAD", "POST", "PUT", "PATCH", "DELETE"}, "/temporal*", h.Temporal)
|
2024-02-11 19:28:00 +00:00
|
|
|
|
|
|
|
// OAuth2
|
2024-02-21 09:41:20 +00:00
|
|
|
oauth2 := s.echo.Group("/oauth2")
|
2024-02-20 10:24:04 +00:00
|
|
|
oauth2.GET("/login", h.OAuth2LoginGET)
|
|
|
|
oauth2.GET("/callback", h.OAuth2CallbackGET)
|
|
|
|
oauth2.GET("/logout", h.OAuth2LogoutGET, h.Authenticated)
|
2024-02-10 11:59:58 +00:00
|
|
|
|
2024-02-19 09:09:30 +00:00
|
|
|
// API
|
2024-02-21 09:41:20 +00:00
|
|
|
apiv1 := s.echo.Group("/api/v1")
|
2024-02-20 10:24:04 +00:00
|
|
|
apiv1.Use(h.Authenticated)
|
|
|
|
apiv1.GET("/workers/connect", h.ApiV1WorkersConnectGET)
|
2024-02-24 11:57:53 +00:00
|
|
|
apiv1.POST("/monitors/:slug/history", h.ApiV1MonitorsHistoryPOST)
|
2024-02-20 10:24:04 +00:00
|
|
|
|
|
|
|
// Error handler
|
2024-02-21 09:41:20 +00:00
|
|
|
s.echo.HTTPErrorHandler = func(err error, c echo.Context) {
|
2024-02-20 10:24:04 +00:00
|
|
|
code := http.StatusInternalServerError
|
|
|
|
if he, ok := err.(*echo.HTTPError); ok {
|
|
|
|
code = he.Code
|
|
|
|
}
|
2024-02-12 10:22:14 +00:00
|
|
|
|
2024-02-20 10:24:04 +00:00
|
|
|
if code == http.StatusNotFound {
|
|
|
|
_ = h.Error404(c)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
_ = c.String(code, err.Error())
|
2024-02-16 21:31:00 +00:00
|
|
|
}
|
|
|
|
|
2024-02-21 09:41:20 +00:00
|
|
|
return s.echo.Start(":" + s.cfg.Port)
|
2024-02-16 21:31:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) Stop() error {
|
|
|
|
ctx := context.Background()
|
2024-02-21 09:41:20 +00:00
|
|
|
return s.echo.Shutdown(ctx)
|
2024-02-10 11:59:58 +00:00
|
|
|
}
|