2023-09-12 12:58:09 +08:00

195 lines
5.4 KiB
Go

package yggdrasil
import (
"encoding/json"
"errors"
"net/http"
"github.com/julienschmidt/httprouter"
"github.com/samber/lo"
"github.com/xmdhs/authlib-skin/model/yggdrasil"
sutils "github.com/xmdhs/authlib-skin/service/utils"
yggdrasilS "github.com/xmdhs/authlib-skin/service/yggdrasil"
)
func (y *Yggdrasil) Authenticate() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
cxt := r.Context()
a, has := getAnyModel[yggdrasil.Authenticate](cxt, w, r.Body, y.validate, y.logger)
if !has {
return
}
t, err := y.yggdrasilService.Authenticate(cxt, a)
if err != nil {
if errors.Is(err, yggdrasilS.ErrPassWord) || errors.Is(err, yggdrasilS.ErrRate) {
y.logger.DebugContext(cxt, err.Error())
handleYgError(cxt, w, yggdrasil.Error{ErrorMessage: "Invalid credentials. Invalid username or password.", Error: "ForbiddenOperationException"}, 403)
return
}
y.handleYgError(cxt, w, err)
return
}
b, _ := json.Marshal(t)
w.Write(b)
}
}
func (y *Yggdrasil) Validate() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
cxt := r.Context()
a, has := getAnyModel[yggdrasil.ValidateToken](cxt, w, r.Body, y.validate, y.logger)
if !has {
return
}
err := y.yggdrasilService.ValidateToken(cxt, a)
if err != nil {
if errors.Is(err, sutils.ErrTokenInvalid) {
y.logger.DebugContext(cxt, err.Error())
handleYgError(cxt, w, yggdrasil.Error{ErrorMessage: "Invalid token.", Error: "ForbiddenOperationException"}, 403)
return
}
y.handleYgError(cxt, w, err)
return
}
w.WriteHeader(204)
}
}
func (y *Yggdrasil) Signout() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
cxt := r.Context()
a, has := getAnyModel[yggdrasil.Pass](cxt, w, r.Body, y.validate, y.logger)
if !has {
return
}
err := y.yggdrasilService.SignOut(cxt, a)
if err != nil {
if errors.Is(err, yggdrasilS.ErrPassWord) || errors.Is(err, yggdrasilS.ErrRate) {
y.logger.DebugContext(cxt, err.Error())
handleYgError(cxt, w, yggdrasil.Error{ErrorMessage: "Invalid credentials. Invalid username or password.", Error: "ForbiddenOperationException"}, 403)
return
}
y.handleYgError(cxt, w, err)
return
}
w.WriteHeader(204)
}
}
func (y *Yggdrasil) Invalidate() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
w.WriteHeader(204)
cxt := r.Context()
a, has := getAnyModel[yggdrasil.ValidateToken](cxt, w, r.Body, y.validate, y.logger)
if !has {
return
}
err := y.yggdrasilService.Invalidate(cxt, a.AccessToken)
if err != nil {
if errors.Is(err, sutils.ErrTokenInvalid) {
y.logger.DebugContext(cxt, err.Error())
return
}
y.logger.WarnContext(cxt, err.Error())
}
}
}
func (y *Yggdrasil) Refresh() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
cxt := r.Context()
a, has := getAnyModel[yggdrasil.RefreshToken](cxt, w, r.Body, y.validate, y.logger)
if !has {
return
}
t, err := y.yggdrasilService.Refresh(cxt, a)
if err != nil {
if errors.Is(err, sutils.ErrTokenInvalid) {
y.logger.DebugContext(cxt, err.Error())
handleYgError(cxt, w, yggdrasil.Error{ErrorMessage: "Invalid token.", Error: "ForbiddenOperationException"}, 403)
return
}
y.handleYgError(cxt, w, err)
return
}
b, _ := json.Marshal(t)
w.Write(b)
}
}
func (y *Yggdrasil) GetProfile() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
ctx := r.Context()
uuid := p.ByName("uuid")
unsigned := r.FormValue("unsigned")
unsignedBool := true
switch unsigned {
case "true":
case "false":
unsignedBool = false
case "":
default:
y.logger.DebugContext(ctx, "unsigned 参数类型错误")
handleYgError(ctx, w, yggdrasil.Error{ErrorMessage: "unsigned 参数类型错误"}, 400)
return
}
u, err := y.yggdrasilService.GetProfile(ctx, uuid, unsignedBool, r.Host)
if err != nil {
if errors.Is(err, yggdrasilS.ErrNotUser) {
y.logger.DebugContext(ctx, err.Error())
w.WriteHeader(204)
return
}
y.handleYgError(ctx, w, err)
return
}
b, _ := json.Marshal(u)
w.Write(b)
}
}
func (y *Yggdrasil) BatchProfile() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
ctx := r.Context()
a, has := getAnyModel[[]string](ctx, w, r.Body, nil, y.logger)
if !has {
return
}
if len(a) > 5 {
y.logger.DebugContext(ctx, "最多同时查询五个")
handleYgError(ctx, w, yggdrasil.Error{ErrorMessage: "最多同时查询五个"}, 400)
return
}
ul, err := y.yggdrasilService.BatchProfile(ctx, a)
if err != nil {
y.handleYgError(ctx, w, err)
return
}
w.Write(lo.Must1(json.Marshal(ul)))
}
}
func (y *Yggdrasil) PlayerCertificates() httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
ctx := r.Context()
token := y.getTokenbyAuthorization(ctx, w, r)
if token == "" {
return
}
c, err := y.yggdrasilService.PlayerCertificates(ctx, token)
if err != nil {
if errors.Is(err, sutils.ErrTokenInvalid) {
y.logger.DebugContext(ctx, err.Error())
handleYgError(ctx, w, yggdrasil.Error{ErrorMessage: "Invalid token.", Error: "ForbiddenOperationException"}, 403)
return
}
y.handleYgError(ctx, w, err)
return
}
w.Write(lo.Must(json.Marshal(c)))
}
}