2025-04-08 09:35:19 +08:00
|
|
|
package auth
|
2025-03-18 17:57:07 +08:00
|
|
|
|
|
|
|
|
import (
|
2025-03-31 09:09:05 +08:00
|
|
|
"context"
|
|
|
|
|
"encoding/base64"
|
|
|
|
|
"errors"
|
2025-11-17 18:38:10 +08:00
|
|
|
"fmt"
|
2025-03-31 09:09:05 +08:00
|
|
|
"log/slog"
|
2025-03-18 17:57:07 +08:00
|
|
|
"strings"
|
2025-11-17 18:38:10 +08:00
|
|
|
"time"
|
2025-03-18 17:57:07 +08:00
|
|
|
|
|
|
|
|
"github.com/gofiber/fiber/v2"
|
|
|
|
|
)
|
|
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
func Authenticate() fiber.Handler {
|
|
|
|
|
return func(ctx *fiber.Ctx) error {
|
|
|
|
|
header := ctx.Get(fiber.HeaderAuthorization)
|
|
|
|
|
authCtx, err := authHeader(ctx.Context(), header)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2025-05-13 09:28:10 +08:00
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
if authCtx == nil {
|
|
|
|
|
authCtx = &AuthCtx{}
|
|
|
|
|
}
|
2025-05-13 09:28:10 +08:00
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
SetAuthCtx(ctx, authCtx)
|
|
|
|
|
return ctx.Next()
|
|
|
|
|
}
|
2025-05-13 09:28:10 +08:00
|
|
|
}
|
|
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
func authHeader(ctx context.Context, header string) (*AuthCtx, error) {
|
|
|
|
|
if header == "" {
|
|
|
|
|
return nil, nil
|
|
|
|
|
}
|
2025-05-13 09:28:10 +08:00
|
|
|
|
2025-04-23 19:01:08 +08:00
|
|
|
var split = strings.Split(header, " ")
|
|
|
|
|
if len(split) != 2 {
|
2025-05-13 15:26:40 +08:00
|
|
|
slog.Debug("Authorization 头格式不正确")
|
2025-11-17 18:38:10 +08:00
|
|
|
return nil, ErrAuthenticateUnauthorize
|
2025-04-23 19:01:08 +08:00
|
|
|
}
|
2025-03-18 17:57:07 +08:00
|
|
|
|
2025-05-13 15:26:40 +08:00
|
|
|
var token = strings.TrimSpace(split[1])
|
2025-04-23 19:01:08 +08:00
|
|
|
if token == "" {
|
2025-05-13 15:26:40 +08:00
|
|
|
slog.Debug("提供的令牌为空")
|
2025-11-17 18:38:10 +08:00
|
|
|
return nil, ErrAuthenticateUnauthorize
|
2025-04-23 19:01:08 +08:00
|
|
|
}
|
|
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
var authCtx *AuthCtx
|
2025-04-23 19:01:08 +08:00
|
|
|
var err error
|
|
|
|
|
switch split[0] {
|
2025-04-26 17:59:34 +08:00
|
|
|
|
2025-04-23 19:01:08 +08:00
|
|
|
case "Bearer":
|
2025-11-17 18:38:10 +08:00
|
|
|
authCtx, err = authBearer(ctx, token)
|
2025-04-26 17:59:34 +08:00
|
|
|
if err != nil {
|
2025-05-06 19:05:44 +08:00
|
|
|
slog.Debug("Bearer 认证失败", "err", err)
|
2025-11-17 18:38:10 +08:00
|
|
|
return nil, ErrAuthenticateUnauthorize
|
2025-04-26 17:59:34 +08:00
|
|
|
}
|
|
|
|
|
|
2025-04-23 19:01:08 +08:00
|
|
|
case "Basic":
|
2025-11-17 18:38:10 +08:00
|
|
|
authCtx, err = authBasic(ctx, token)
|
2025-04-26 17:59:34 +08:00
|
|
|
if err != nil {
|
2025-05-06 19:05:44 +08:00
|
|
|
slog.Debug("Basic 认证失败", "err", err)
|
2025-11-17 18:38:10 +08:00
|
|
|
return nil, ErrAuthenticateUnauthorize
|
2025-04-26 17:59:34 +08:00
|
|
|
}
|
|
|
|
|
|
2025-04-23 19:01:08 +08:00
|
|
|
default:
|
2025-05-13 15:26:40 +08:00
|
|
|
slog.Debug("无效的认证方式", "method", split[0])
|
2025-11-17 18:38:10 +08:00
|
|
|
return nil, ErrAuthenticateUnauthorize
|
2025-04-23 19:01:08 +08:00
|
|
|
}
|
|
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
return authCtx, err
|
2025-03-31 09:09:05 +08:00
|
|
|
}
|
2025-03-18 17:57:07 +08:00
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
func authBearer(_ context.Context, token string) (*AuthCtx, error) {
|
|
|
|
|
session, err := FindSession(token, time.Now())
|
2025-03-31 09:09:05 +08:00
|
|
|
if err != nil {
|
2025-11-17 18:38:10 +08:00
|
|
|
slog.Debug("Bearer 认证失败", "err", err)
|
|
|
|
|
return nil, ErrAuthenticateUnauthorize
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
scopes := []string{}
|
2025-11-24 18:44:06 +08:00
|
|
|
if session.Scopes != nil {
|
|
|
|
|
scopes = strings.Split(*session.Scopes, " ")
|
2025-03-18 17:57:07 +08:00
|
|
|
}
|
2025-11-17 18:38:10 +08:00
|
|
|
return &AuthCtx{
|
|
|
|
|
User: session.User,
|
|
|
|
|
Admin: session.Admin,
|
|
|
|
|
Client: session.Client,
|
|
|
|
|
Scopes: scopes,
|
|
|
|
|
Session: session,
|
|
|
|
|
}, nil
|
2025-03-18 17:57:07 +08:00
|
|
|
}
|
2025-03-28 15:01:30 +08:00
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
func authBasic(_ context.Context, token string) (*AuthCtx, error) {
|
2025-03-28 15:01:30 +08:00
|
|
|
|
2025-03-31 09:09:05 +08:00
|
|
|
// 解析 Basic 认证信息
|
2025-04-28 11:44:54 +08:00
|
|
|
var base, err = base64.RawURLEncoding.DecodeString(token)
|
2025-03-31 09:09:05 +08:00
|
|
|
if err != nil {
|
2025-05-24 12:37:16 +08:00
|
|
|
base, err = base64.URLEncoding.DecodeString(token)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, errors.New("令牌格式错误,无法解析令牌")
|
|
|
|
|
}
|
2025-03-31 09:09:05 +08:00
|
|
|
}
|
2025-03-28 15:01:30 +08:00
|
|
|
|
2025-03-31 09:09:05 +08:00
|
|
|
var split = strings.Split(string(base), ":")
|
|
|
|
|
if len(split) != 2 {
|
2025-05-13 15:26:40 +08:00
|
|
|
return nil, errors.New("令牌格式错误,必须是 <client_id>:<client_secret> 格式")
|
2025-03-31 09:09:05 +08:00
|
|
|
}
|
2025-03-28 15:01:30 +08:00
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
client, err := authClient(split[0], split[1])
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, fmt.Errorf("客户端认证失败:%w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &AuthCtx{
|
|
|
|
|
Client: client,
|
|
|
|
|
Scopes: []string{},
|
|
|
|
|
}, nil
|
|
|
|
|
}
|