Files
Primora/apps/backend/internal/app/app.go
T
2026-04-10 12:03:31 +02:00

169 lines
4.5 KiB
Go

package app
import (
"context"
"fmt"
"log/slog"
"os"
"time"
"github.com/gin-gonic/gin"
"github.com/go-playground/validator/v10"
"github.com/jackc/pgx/v5/pgxpool"
"github.com/joho/godotenv"
"github.com/redis/go-redis/v9"
"github.com/tdvorak/primora/apps/backend/internal/auth"
"github.com/tdvorak/primora/apps/backend/internal/config"
"github.com/tdvorak/primora/apps/backend/internal/database"
"github.com/tdvorak/primora/apps/backend/internal/handlers"
"github.com/tdvorak/primora/apps/backend/internal/middleware"
"github.com/tdvorak/primora/apps/backend/internal/observability"
"github.com/tdvorak/primora/apps/backend/internal/repositories"
"github.com/tdvorak/primora/apps/backend/internal/services"
"github.com/tdvorak/primora/apps/backend/internal/storage"
)
type App struct {
Config config.Config
Router *gin.Engine
Logger *slog.Logger
DB *pgxpool.Pool
Redis *redis.Client
}
func Bootstrap(ctx context.Context) (*App, error) {
_ = godotenv.Load()
logger := slog.New(slog.NewJSONHandler(os.Stdout, nil))
cfg, err := config.Load()
if err != nil {
return nil, err
}
dbPool, err := database.Connect(ctx, cfg.DatabaseURL)
if err != nil {
return nil, err
}
if err := database.RunMigrationsFromPool(dbPool, database.ResolveMigrationsDir(), logger); err != nil {
return nil, err
}
var verifier *auth.Verifier
for attempt := 1; attempt <= 20; attempt++ {
verifier, err = auth.NewVerifier(ctx, cfg.AuthInternalBaseURL+"/auth/jwks", cfg.JWTIssuer, cfg.JWTAudience)
if err == nil {
break
}
logger.Warn("auth jwks unavailable, retrying", "attempt", attempt, "error", err)
time.Sleep(2 * time.Second)
}
if verifier == nil {
return nil, err
}
var redisClient *redis.Client
if options, err := redis.ParseURL(cfg.DragonflyURL); err != nil {
logger.Warn("dragonfly configuration invalid, continuing in degraded mode", "error", err)
} else {
redisClient = redis.NewClient(options)
if err := redisClient.Ping(ctx).Err(); err != nil {
logger.Warn("dragonfly unavailable, continuing in degraded mode", "error", err)
redisClient = nil
}
}
store, err := storage.NewLocalStore(cfg.StorageRoot)
if err != nil {
return nil, err
}
repo := repositories.NewCoreRepository(dbPool)
platform := services.NewPlatformService(repo, store, services.NewMailer(cfg), os.Getenv("VITE_APP_URL"))
if cfg.Env == "production" {
gin.SetMode(gin.ReleaseMode)
}
router := gin.New()
router.Use(gin.Recovery())
router.Use(middleware.RequestID())
router.Use(middleware.Logger(logger))
metrics := observability.NewMetrics()
router.Use(middleware.Metrics(metrics))
router.Use(middleware.Compression())
// CORS configuration - update AllowedOrigins for production
corsOrigins := []string{cfg.PublicURL}
if cfg.Env == "development" {
corsOrigins = append(corsOrigins, "http://localhost", "http://localhost:3000")
}
router.Use(middleware.CORS(middleware.CORSConfig{
AllowedOrigins: corsOrigins,
}))
router.Use(middleware.AuthMiddleware{
Queries: repo,
Logger: logger,
Redis: redisClient,
Verifier: verifier,
RateLimits: middleware.RateLimitConfig{
APIKeyPerMinute: cfg.APIKeyRateLimitPerMin,
UserPerMinute: cfg.UserRateLimitPerMin,
},
}.ResolveActor())
handler := &handlers.HTTPHandler{
Platform: platform,
Validate: validator.New(),
Metrics: metrics,
Readiness: func(c *gin.Context) map[string]any {
status := map[string]any{
"status": "ok",
"checks": map[string]any{
"database": "ok",
"storage": "ok",
"dragonfly": "ok",
},
"metrics": metrics.GetStats(),
}
if err := dbPool.Ping(c.Request.Context()); err != nil {
status["status"] = "degraded"
status["checks"].(map[string]any)["database"] = err.Error()
}
if redisClient != nil {
if err := redisClient.Ping(c.Request.Context()).Err(); err != nil {
status["status"] = "degraded"
status["checks"].(map[string]any)["dragonfly"] = err.Error()
}
} else {
status["status"] = "degraded"
status["checks"].(map[string]any)["dragonfly"] = "disabled"
}
return status
},
}
handler.Register(router)
return &App{
Config: cfg,
Router: router,
Logger: logger,
DB: dbPool,
Redis: redisClient,
}, nil
}
func (a *App) Run() error {
address := ":" + a.Config.ServerPort
a.Logger.Info("primora backend starting", "address", address)
return a.Router.Run(address)
}
func (a *App) Close() error {
if a.Redis != nil {
if err := a.Redis.Close(); err != nil {
return fmt.Errorf("close redis: %w", err)
}
}
a.DB.Close()
return nil
}