2024-01-26 17:11:42 -05:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2024-01-28 13:57:25 -05:00
|
|
|
"context"
|
|
|
|
|
"fmt"
|
2024-01-26 17:11:42 -05:00
|
|
|
"log"
|
2024-02-09 19:30:50 -05:00
|
|
|
"log/slog"
|
2024-01-26 17:11:42 -05:00
|
|
|
"net/http"
|
2024-02-09 19:30:50 -05:00
|
|
|
"os"
|
|
|
|
|
"time"
|
2024-01-28 13:57:25 -05:00
|
|
|
|
2024-02-02 11:15:22 -05:00
|
|
|
"forgejo.merr.is/annika/isl-api/Controllers"
|
2024-02-09 19:30:50 -05:00
|
|
|
"forgejo.merr.is/annika/isl-api/Routes"
|
2024-02-02 11:15:22 -05:00
|
|
|
"forgejo.merr.is/annika/isl-api/Services"
|
|
|
|
|
"forgejo.merr.is/annika/isl-api/sql/powerItem"
|
2024-02-07 19:15:25 -05:00
|
|
|
"github.com/go-chi/chi/v5"
|
|
|
|
|
"github.com/go-chi/cors"
|
2024-02-09 19:30:50 -05:00
|
|
|
"github.com/go-chi/httplog/v2"
|
2024-02-05 17:05:02 -05:00
|
|
|
"github.com/jackc/pgx/v4/pgxpool"
|
2024-02-09 19:30:50 -05:00
|
|
|
"github.com/lestrrat-go/jwx/v2/jwk"
|
|
|
|
|
"github.com/lestrrat-go/jwx/v2/jwt"
|
|
|
|
|
"github.com/lmittmann/tint"
|
|
|
|
|
"github.com/moosetheory/jwtauth/v5"
|
2024-01-31 14:51:42 -05:00
|
|
|
"github.com/spf13/viper"
|
2024-01-26 17:11:42 -05:00
|
|
|
)
|
|
|
|
|
|
2024-01-31 14:51:42 -05:00
|
|
|
var conf *EnvConfigs
|
2024-02-09 19:30:50 -05:00
|
|
|
var logger *slog.Logger
|
2024-01-31 14:51:42 -05:00
|
|
|
|
|
|
|
|
func init() {
|
2024-02-09 19:30:50 -05:00
|
|
|
w := os.Stderr
|
|
|
|
|
logger = slog.New(
|
|
|
|
|
tint.NewHandler(w, &tint.Options{
|
|
|
|
|
TimeFormat: time.RFC3339Nano,
|
|
|
|
|
}),
|
|
|
|
|
)
|
|
|
|
|
logger.Info("Initializing isl-api")
|
2024-01-31 14:51:42 -05:00
|
|
|
setupConfig()
|
2024-02-09 19:30:50 -05:00
|
|
|
logger = slog.New(
|
|
|
|
|
tint.NewHandler(w, &tint.Options{
|
|
|
|
|
Level: conf.GetLogLevel(),
|
|
|
|
|
TimeFormat: time.RFC3339Nano,
|
|
|
|
|
}),
|
|
|
|
|
)
|
2024-01-31 14:51:42 -05:00
|
|
|
}
|
|
|
|
|
|
2024-01-26 17:11:42 -05:00
|
|
|
func main() {
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Info("Starting isl-api")
|
2024-01-28 13:57:25 -05:00
|
|
|
deps := dependencies{}
|
|
|
|
|
deps.initializeDependencies()
|
|
|
|
|
|
2024-02-05 17:05:02 -05:00
|
|
|
fmt.Printf("Preparing to listen on `:%v`\n", conf.HttpPort)
|
2024-01-31 14:51:42 -05:00
|
|
|
err := http.ListenAndServe(fmt.Sprintf(":%v", conf.HttpPort), deps.router)
|
2024-01-26 17:11:42 -05:00
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-31 14:51:42 -05:00
|
|
|
func setupConfig() {
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Info("Loading config")
|
2024-01-31 14:51:42 -05:00
|
|
|
viper.AddConfigPath(".")
|
|
|
|
|
viper.SetConfigName(".env")
|
|
|
|
|
viper.SetConfigType("env")
|
|
|
|
|
|
|
|
|
|
viper.SetEnvPrefix("isl_api")
|
|
|
|
|
viper.AutomaticEnv()
|
|
|
|
|
|
|
|
|
|
if err := viper.ReadInConfig(); err != nil {
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Error("Error reading env file, exiting", "error", err)
|
|
|
|
|
panic(err)
|
2024-01-31 14:51:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := viper.Unmarshal(&conf); err != nil {
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Error("Unable to unmarshal configuration, exiting", "error", err)
|
|
|
|
|
panic(err)
|
2024-01-31 14:51:42 -05:00
|
|
|
}
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Info("Finished loading config")
|
2024-01-31 14:51:42 -05:00
|
|
|
}
|
|
|
|
|
|
2024-01-28 13:57:25 -05:00
|
|
|
type dependencies struct {
|
2024-02-07 19:15:25 -05:00
|
|
|
router *chi.Mux
|
2024-02-05 17:05:02 -05:00
|
|
|
postgresConnection *pgxpool.Pool
|
2024-01-28 13:57:25 -05:00
|
|
|
context context.Context
|
2024-02-09 19:30:50 -05:00
|
|
|
jwkContext context.Context
|
2024-01-28 13:57:25 -05:00
|
|
|
powerItemQuerier *powerItem.DBQuerier
|
|
|
|
|
powerItemService *Services.PowerItemService
|
|
|
|
|
powerItemController *Controllers.PowerItemController
|
2024-02-09 19:30:50 -05:00
|
|
|
jwkCache *jwk.Cache
|
2024-01-28 13:57:25 -05:00
|
|
|
}
|
2024-01-26 17:11:42 -05:00
|
|
|
|
2024-01-28 13:57:25 -05:00
|
|
|
func (d *dependencies) initializeDependencies() error {
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Info("Initializing dependencies")
|
2024-01-28 13:57:25 -05:00
|
|
|
var err error
|
2024-02-09 19:30:50 -05:00
|
|
|
d.jwkContext = context.Background()
|
|
|
|
|
d.jwkCache = jwk.NewCache(d.jwkContext)
|
|
|
|
|
d.jwkCache.Register(conf.JWKSURI, jwk.WithMinRefreshInterval(15*time.Minute))
|
|
|
|
|
_, err = d.jwkCache.Refresh(d.jwkContext, conf.JWKSURI)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logger.Error("Error setting up JWK cache", "error", err)
|
|
|
|
|
return err
|
|
|
|
|
}
|
2024-01-26 17:11:42 -05:00
|
|
|
|
2024-02-07 19:15:25 -05:00
|
|
|
d.router = chi.NewRouter()
|
2024-02-09 19:30:50 -05:00
|
|
|
|
|
|
|
|
httpLogger := httplog.NewLogger("isl-api", httplog.Options{
|
|
|
|
|
Concise: true,
|
|
|
|
|
RequestHeaders: true,
|
|
|
|
|
})
|
|
|
|
|
httpLogger.Logger = logger
|
|
|
|
|
d.router.Use(httplog.RequestLogger(httpLogger))
|
|
|
|
|
|
2024-02-07 19:15:25 -05:00
|
|
|
d.router.Use(cors.Handler(cors.Options{
|
|
|
|
|
AllowedOrigins: []string{"*"},
|
|
|
|
|
AllowedMethods: []string{"GET", "POST", "PUT", "DELETE", "OPTIONS"},
|
|
|
|
|
AllowedHeaders: []string{"Cache-Control", "Expires", "Pragma"},
|
|
|
|
|
}))
|
2024-01-28 13:57:25 -05:00
|
|
|
d.context = context.Background()
|
2024-02-05 17:05:02 -05:00
|
|
|
d.postgresConnection, err = pgxpool.Connect(d.context, conf.ConnectionString)
|
2024-01-28 13:57:25 -05:00
|
|
|
if err != nil {
|
2024-02-09 19:30:50 -05:00
|
|
|
logger.Error("Error setting up database connection", "error", err)
|
2024-01-28 13:57:25 -05:00
|
|
|
return err
|
2024-01-26 17:11:42 -05:00
|
|
|
}
|
2024-01-28 13:57:25 -05:00
|
|
|
d.powerItemQuerier = powerItem.NewQuerier(d.postgresConnection)
|
|
|
|
|
d.powerItemService = Services.NewPowerItemService(d.powerItemQuerier)
|
2024-02-07 19:15:25 -05:00
|
|
|
d.powerItemController = Controllers.NewPowerItemController(d.powerItemService)
|
2024-02-09 19:30:50 -05:00
|
|
|
|
|
|
|
|
jwkSet, err := d.jwkCache.Get(d.jwkContext, conf.JWKSURI)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logger.Error("Error getting JWKS", "error", err)
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tokenAuth := jwtauth.New("HS256", nil, nil, jwkSet, jwt.WithAcceptableSkew(30*time.Second))
|
|
|
|
|
d.router.Mount("/powerItems", Routes.SetupPowerItemRoutes(*d.powerItemController, tokenAuth))
|
|
|
|
|
logger.Info("Finished initializing dependencies")
|
2024-01-28 13:57:25 -05:00
|
|
|
return nil
|
|
|
|
|
}
|