2020-04-16 19:42:38 +03:00
|
|
|
package di
|
|
|
|
|
|
|
|
import (
|
2020-04-19 02:31:09 +03:00
|
|
|
"net/http"
|
2024-02-07 17:34:57 +01:00
|
|
|
"slices"
|
2020-04-19 02:31:09 +03:00
|
|
|
"strings"
|
|
|
|
|
2023-12-13 17:29:12 +01:00
|
|
|
"github.com/defval/di"
|
|
|
|
"github.com/etherlabsio/healthcheck/v2"
|
2020-04-16 19:42:38 +03:00
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/spf13/viper"
|
|
|
|
|
2024-02-01 08:12:34 +01:00
|
|
|
. "ely.by/chrly/internal/http"
|
2024-03-05 13:07:54 +01:00
|
|
|
"ely.by/chrly/internal/security"
|
2020-04-16 19:42:38 +03:00
|
|
|
)
|
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
const ModuleSkinsystem = "skinsystem"
|
|
|
|
const ModuleProfiles = "profiles"
|
|
|
|
const ModuleSigner = "signer"
|
|
|
|
|
2024-02-07 17:34:57 +01:00
|
|
|
var handlersDiOptions = di.Options(
|
2020-04-19 02:31:09 +03:00
|
|
|
di.Provide(newHandlerFactory, di.As(new(http.Handler))),
|
2024-03-05 13:07:54 +01:00
|
|
|
di.Provide(newSkinsystemHandler, di.WithName(ModuleSkinsystem)),
|
|
|
|
di.Provide(newProfilesApiHandler, di.WithName(ModuleProfiles)),
|
|
|
|
di.Provide(newSignerApiHandler, di.WithName(ModuleSigner)),
|
2020-04-16 19:42:38 +03:00
|
|
|
)
|
|
|
|
|
2020-04-19 02:31:09 +03:00
|
|
|
func newHandlerFactory(
|
|
|
|
container *di.Container,
|
|
|
|
config *viper.Viper,
|
|
|
|
) (*mux.Router, error) {
|
|
|
|
enabledModules := config.GetStringSlice("modules")
|
|
|
|
|
|
|
|
// gorilla.mux has no native way to combine multiple routers.
|
|
|
|
// The hack used later in the code works for prefixes in addresses, but leads to misbehavior
|
|
|
|
// if you set an empty prefix. Since the main application should be mounted at the root prefix,
|
|
|
|
// we use it as the base router
|
|
|
|
var router *mux.Router
|
2024-03-05 13:07:54 +01:00
|
|
|
if slices.Contains(enabledModules, ModuleSkinsystem) {
|
|
|
|
if err := container.Resolve(&router, di.Name(ModuleSkinsystem)); err != nil {
|
2020-04-19 02:31:09 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
router = mux.NewRouter()
|
|
|
|
}
|
|
|
|
|
|
|
|
router.StrictSlash(true)
|
2024-02-07 18:33:06 +01:00
|
|
|
router.NotFoundHandler = http.HandlerFunc(NotFoundHandler)
|
2020-04-19 02:31:09 +03:00
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
if slices.Contains(enabledModules, ModuleProfiles) {
|
|
|
|
var profilesApiRouter *mux.Router
|
|
|
|
if err := container.Resolve(&profilesApiRouter, di.Name(ModuleProfiles)); err != nil {
|
2020-04-19 02:31:09 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var authenticator Authenticator
|
|
|
|
if err := container.Resolve(&authenticator); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
profilesApiRouter.Use(NewAuthenticationMiddleware(authenticator, security.ProfilesScope))
|
2020-04-19 02:31:09 +03:00
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
mount(router, "/api/profiles", profilesApiRouter)
|
|
|
|
}
|
|
|
|
|
|
|
|
if slices.Contains(enabledModules, ModuleSigner) {
|
|
|
|
var signerApiRouter *mux.Router
|
|
|
|
if err := container.Resolve(&signerApiRouter, di.Name(ModuleSigner)); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var authenticator Authenticator
|
|
|
|
if err := container.Resolve(&authenticator); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
authMiddleware := NewAuthenticationMiddleware(authenticator, security.SignScope)
|
|
|
|
conditionalAuth := NewConditionalMiddleware(func(req *http.Request) bool {
|
|
|
|
return req.Method != "GET"
|
|
|
|
}, authMiddleware)
|
|
|
|
signerApiRouter.Use(conditionalAuth)
|
|
|
|
|
|
|
|
mount(router, "/api/signer", signerApiRouter)
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Resolve health checkers last, because all the services required by the application
|
|
|
|
// must first be initialized and each of them can publish its own checkers
|
2020-04-20 13:23:02 +03:00
|
|
|
var healthCheckers []*namedHealthChecker
|
2023-12-13 17:29:12 +01:00
|
|
|
if has, _ := container.Has(&healthCheckers); has {
|
2020-04-19 02:31:09 +03:00
|
|
|
if err := container.Resolve(&healthCheckers); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
checkersOptions := make([]healthcheck.Option, len(healthCheckers))
|
|
|
|
for i, checker := range healthCheckers {
|
2020-04-20 13:23:02 +03:00
|
|
|
checkersOptions[i] = healthcheck.WithChecker(checker.Name, checker.Checker)
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
2020-04-20 22:18:27 +03:00
|
|
|
router.Handle("/healthcheck", healthcheck.Handler(checkersOptions...)).Methods("GET")
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return router, nil
|
|
|
|
}
|
|
|
|
|
2020-04-16 19:42:38 +03:00
|
|
|
func newSkinsystemHandler(
|
|
|
|
config *viper.Viper,
|
2024-01-30 09:05:04 +01:00
|
|
|
profilesProvider ProfilesProvider,
|
2024-03-05 13:07:54 +01:00
|
|
|
texturesSigner SignerService,
|
2020-04-16 19:42:38 +03:00
|
|
|
) *mux.Router {
|
2020-04-20 15:40:20 +03:00
|
|
|
config.SetDefault("textures.extra_param_name", "chrly")
|
|
|
|
config.SetDefault("textures.extra_param_value", "how do you tame a horse in Minecraft?")
|
|
|
|
|
2020-04-19 02:31:09 +03:00
|
|
|
return (&Skinsystem{
|
2024-01-30 09:05:04 +01:00
|
|
|
ProfilesProvider: profilesProvider,
|
2024-03-05 13:07:54 +01:00
|
|
|
SignerService: texturesSigner,
|
2020-04-16 19:42:38 +03:00
|
|
|
TexturesExtraParamName: config.GetString("textures.extra_param_name"),
|
|
|
|
TexturesExtraParamValue: config.GetString("textures.extra_param_value"),
|
2020-04-19 02:31:09 +03:00
|
|
|
}).Handler()
|
|
|
|
}
|
2020-04-16 19:42:38 +03:00
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
func newProfilesApiHandler(profilesManager ProfilesManager) *mux.Router {
|
|
|
|
return (&ProfilesApi{
|
2024-01-30 09:05:04 +01:00
|
|
|
ProfilesManager: profilesManager,
|
2020-04-19 02:31:09 +03:00
|
|
|
}).Handler()
|
2020-04-16 19:42:38 +03:00
|
|
|
}
|
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
func newSignerApiHandler(signer Signer) *mux.Router {
|
|
|
|
return (&SignerApi{
|
|
|
|
Signer: signer,
|
|
|
|
}).Handler()
|
|
|
|
}
|
|
|
|
|
2020-04-19 02:31:09 +03:00
|
|
|
func mount(router *mux.Router, path string, handler http.Handler) {
|
|
|
|
router.PathPrefix(path).Handler(
|
|
|
|
http.StripPrefix(
|
|
|
|
strings.TrimSuffix(path, "/"),
|
|
|
|
handler,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
type namedHealthChecker struct {
|
|
|
|
Name string
|
|
|
|
Checker healthcheck.Checker
|
|
|
|
}
|