Files
pig-farm-controller/internal/infra/utils/token/token_service.go

69 lines
1.8 KiB
Go
Raw Normal View History

2025-09-12 12:25:53 +08:00
package token
import (
2025-09-13 19:48:13 +08:00
"fmt"
2025-09-12 12:25:53 +08:00
"time"
"github.com/golang-jwt/jwt/v5"
)
// Claims 定义了 JWT 的声明结构
type Claims struct {
2025-11-10 22:23:31 +08:00
UserID uint32 `json:"user_id"`
2025-09-12 12:25:53 +08:00
jwt.RegisteredClaims
}
2025-11-05 21:40:19 +08:00
// Generator 定义了 token 操作的接口
type Generator interface {
2025-11-10 22:23:31 +08:00
GenerateToken(userID uint32) (string, error)
2025-09-12 12:25:53 +08:00
ParseToken(tokenString string) (*Claims, error)
}
2025-11-05 21:40:19 +08:00
// tokenGenerator 是 Generator 接口的实现
type tokenGenerator struct {
2025-09-12 12:25:53 +08:00
secret []byte
}
2025-11-05 21:40:19 +08:00
// NewTokenGenerator 创建并返回一个新的 Generator 实例
func NewTokenGenerator(secret []byte) Generator {
return &tokenGenerator{secret: secret}
2025-09-12 12:25:53 +08:00
}
// GenerateToken 生成一个新的 JWT token
2025-11-10 22:23:31 +08:00
func (s *tokenGenerator) GenerateToken(userID uint32) (string, error) {
2025-09-12 12:25:53 +08:00
nowTime := time.Now()
expireTime := nowTime.Add(24 * time.Hour) // Token 有效期为 24 小时
claims := Claims{
UserID: userID,
RegisteredClaims: jwt.RegisteredClaims{
ExpiresAt: jwt.NewNumericDate(expireTime),
Issuer: "pig-farm-controller",
},
}
tokenClaims := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
token, err := tokenClaims.SignedString(s.secret)
return token, err
}
// ParseToken 解析并验证 JWT token
2025-11-05 21:40:19 +08:00
func (s *tokenGenerator) ParseToken(tokenString string) (*Claims, error) {
2025-09-12 12:25:53 +08:00
token, err := jwt.ParseWithClaims(tokenString, &Claims{}, func(token *jwt.Token) (interface{}, error) {
return s.secret, nil
})
2025-09-13 19:48:13 +08:00
// 优先检查解析过程中是否发生错误
if err != nil {
return nil, err
}
// 只有当 token 对象有效时,才尝试获取 Claims 并验证
2025-09-12 12:25:53 +08:00
if claims, ok := token.Claims.(*Claims); ok && token.Valid {
return claims, nil
}
2025-09-13 19:48:13 +08:00
// 如果 token 无效(例如,过期但没有返回错误,或者 Claims 类型不匹配),则返回一个通用错误
return nil, fmt.Errorf("token is invalid")
2025-09-12 12:25:53 +08:00
}