mirror of
https://github.com/snowykami/neo-blog.git
synced 2025-09-26 02:56:22 +00:00
- Implemented Sidebar component with collapsible functionality. - Added SidebarProvider for managing open state and keyboard shortcuts. - Created SidebarTrigger, SidebarRail, and various sidebar elements (Header, Footer, Content, etc.). - Integrated mobile responsiveness using Sheet component. - Added utility hooks for mobile detection. feat: create table component for structured data display - Developed Table component with subcomponents: TableHeader, TableBody, TableFooter, TableRow, TableCell, and TableCaption. - Enhanced styling for better readability and usability. feat: implement tabs component for navigation - Created Tabs component with TabsList, TabsTrigger, and TabsContent for tabbed navigation. - Ensured accessibility and responsive design. feat: add toggle group component for grouped toggle buttons - Developed ToggleGroup and ToggleGroupItem components for managing toggle states. - Integrated context for consistent styling and behavior. feat: create toggle component for binary state representation - Implemented Toggle component with variant and size options. - Enhanced user interaction with visual feedback. feat: add tooltip component for contextual information - Developed Tooltip, TooltipTrigger, and TooltipContent for displaying additional information on hover. - Integrated animations for a smoother user experience. feat: implement mobile detection hook - Created useIsMobile hook to determine if the user is on a mobile device. - Utilized matchMedia for responsive design adjustments.
129 lines
4.1 KiB
Go
129 lines
4.1 KiB
Go
package middleware
|
||
|
||
import (
|
||
"context"
|
||
"strings"
|
||
"time"
|
||
|
||
"github.com/cloudwego/hertz/pkg/app"
|
||
"github.com/sirupsen/logrus"
|
||
"github.com/snowykami/neo-blog/internal/ctxutils"
|
||
"github.com/snowykami/neo-blog/internal/repo"
|
||
"github.com/snowykami/neo-blog/pkg/constant"
|
||
"github.com/snowykami/neo-blog/pkg/resps"
|
||
"github.com/snowykami/neo-blog/pkg/utils"
|
||
)
|
||
|
||
func UseAuth(block bool) app.HandlerFunc {
|
||
return func(ctx context.Context, c *app.RequestContext) {
|
||
// For cookie
|
||
tokenFromCookie := string(c.Cookie("token"))
|
||
tokenFromHeader := strings.TrimPrefix(string(c.GetHeader("Authorization")), "Bearer ")
|
||
refreshToken := string(c.Cookie("refresh_token"))
|
||
|
||
// 尝试用普通 tokenFromCookie 认证
|
||
if tokenFromCookie != "" {
|
||
tokenClaims, err := utils.Jwt.ParseJsonWebTokenWithoutState(tokenFromCookie)
|
||
if err == nil && tokenClaims != nil {
|
||
ctx = context.WithValue(ctx, constant.ContextKeyUserID, tokenClaims.UserID)
|
||
c.Next(ctx)
|
||
logrus.Debugf("UseAuth: tokenFromCookie authenticated successfully, userID: %d", tokenClaims.UserID)
|
||
return
|
||
}
|
||
logrus.Debugf("UseAuth: tokenFromCookie authentication failed, error: %v", err)
|
||
}
|
||
// tokenFromCookie 认证失败,尝试用 Bearer tokenFromHeader 认证
|
||
if tokenFromHeader != "" {
|
||
tokenClaims, err := utils.Jwt.ParseJsonWebTokenWithoutState(tokenFromHeader)
|
||
if err == nil && tokenClaims != nil {
|
||
ctx = context.WithValue(ctx, constant.ContextKeyUserID, tokenClaims.UserID)
|
||
c.Next(ctx)
|
||
logrus.Debugf("UseAuth: tokenFromHeader authenticated successfully, userID: %d", tokenClaims.UserID)
|
||
return
|
||
}
|
||
logrus.Debugf("UseAuth: tokenFromHeader authentication failed, error: %v", err)
|
||
}
|
||
// tokenFromCookie 失效 使用 refresh tokenFromCookie 重新签发和鉴权
|
||
refreshTokenClaims, err := utils.Jwt.ParseJsonWebTokenWithoutState(refreshToken)
|
||
if err == nil && refreshTokenClaims != nil {
|
||
ok, err := isStatefulJwtValid(refreshTokenClaims)
|
||
if err == nil && ok {
|
||
ctx = context.WithValue(ctx, constant.ContextKeyUserID, refreshTokenClaims.UserID)
|
||
// 生成新 tokenFromCookie
|
||
newTokenClaims := utils.Jwt.NewClaims(
|
||
refreshTokenClaims.UserID,
|
||
refreshTokenClaims.SessionKey,
|
||
refreshTokenClaims.Stateful,
|
||
time.Duration(utils.Env.GetAsInt(constant.EnvKeyRefreshTokenDuration, 30)*int(time.Hour)),
|
||
)
|
||
newToken, err := newTokenClaims.ToString()
|
||
if err == nil {
|
||
ctxutils.SetTokenCookie(c, newToken)
|
||
} else {
|
||
resps.InternalServerError(c, resps.ErrInternalServerError)
|
||
}
|
||
c.Next(ctx)
|
||
logrus.Debugf("UseAuth: refreshToken authenticated successfully, userID: %d", refreshTokenClaims.UserID)
|
||
return
|
||
}
|
||
}
|
||
|
||
// 所有认证方式都失败
|
||
if block {
|
||
logrus.Debug("UseAuth: all authentication methods failed, blocking request")
|
||
resps.Unauthorized(c, resps.ErrUnauthorized)
|
||
c.Abort()
|
||
} else {
|
||
logrus.Debug("UseAuth: all authentication methods failed, blocking request")
|
||
c.Next(ctx)
|
||
}
|
||
}
|
||
}
|
||
|
||
// UseRole 检查用户角色是否符合要求,必须在 UseAuth 之后使用
|
||
// requiredRole 可以是 "admin", "editor", "user" 等
|
||
// admin包含editor, editor包含user
|
||
func UseRole(requiredRole string) app.HandlerFunc {
|
||
return func(ctx context.Context, c *app.RequestContext) {
|
||
currentUserID := ctx.Value(constant.ContextKeyUserID)
|
||
if currentUserID == nil {
|
||
resps.Unauthorized(c, resps.ErrUnauthorized)
|
||
c.Abort()
|
||
return
|
||
}
|
||
userID := currentUserID.(uint)
|
||
user, err := repo.User.GetUserByID(userID)
|
||
if err != nil {
|
||
resps.InternalServerError(c, resps.ErrInternalServerError)
|
||
c.Abort()
|
||
return
|
||
}
|
||
if user == nil {
|
||
resps.Unauthorized(c, resps.ErrUnauthorized)
|
||
c.Abort()
|
||
return
|
||
}
|
||
|
||
roleHierarchy := map[string]int{
|
||
constant.RoleUser: 1,
|
||
constant.RoleEditor: 2,
|
||
constant.RoleAdmin: 3,
|
||
}
|
||
userRoleLevel := roleHierarchy[user.Role]
|
||
requiredRoleLevel := roleHierarchy[requiredRole]
|
||
if userRoleLevel < requiredRoleLevel {
|
||
resps.Forbidden(c, resps.ErrForbidden)
|
||
c.Abort()
|
||
return
|
||
}
|
||
c.Next(ctx)
|
||
}
|
||
}
|
||
|
||
func isStatefulJwtValid(claims *utils.Claims) (bool, error) {
|
||
if !claims.Stateful {
|
||
return true, nil
|
||
}
|
||
return repo.Session.IsSessionValid(claims.SessionKey)
|
||
}
|