jwt_auth.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. package auth
  2. import (
  3. "encoding/json"
  4. "github.com/dgrijalva/jwt-go"
  5. "gxt-file-server/app/errors"
  6. "time"
  7. )
  8. // TokenInfo 令牌信息
  9. type TokenInfo interface {
  10. // GetAccessToken 获取访问令牌
  11. GetAccessToken() string
  12. // GetTokenType 获取令牌类型
  13. GetTokenType() string
  14. // GetExpiresAt 获取令牌到期时间戳
  15. GetExpiresAt() int64
  16. // EncodeToJSON JSON编码
  17. EncodeToJSON() ([]byte, error)
  18. }
  19. // tokenInfo 令牌信息
  20. type tokenInfo struct {
  21. AccessToken string `json:"access_token"` // 访问令牌
  22. TokenType string `json:"token_type"` // 令牌类型
  23. ExpiresAt int64 `json:"expires_at"` // 令牌到期时间
  24. }
  25. func (t *tokenInfo) GetAccessToken() string {
  26. return t.AccessToken
  27. }
  28. func (t *tokenInfo) GetTokenType() string {
  29. return t.TokenType
  30. }
  31. func (t *tokenInfo) GetExpiresAt() int64 {
  32. return t.ExpiresAt
  33. }
  34. func (t *tokenInfo) EncodeToJSON() ([]byte, error) {
  35. return json.Marshal(t)
  36. }
  37. type options struct {
  38. signingMethod jwt.SigningMethod
  39. signingKey interface{}
  40. keyfunc jwt.Keyfunc
  41. expired int
  42. tokenType string
  43. }
  44. const defaultKey = "gxt-file-server"
  45. var defaultOptions = options{
  46. tokenType: "Bearer",
  47. expired: 7200,
  48. signingMethod: jwt.SigningMethodHS512,
  49. signingKey: []byte(defaultKey),
  50. keyfunc: func(t *jwt.Token) (interface{}, error) {
  51. if _, ok := t.Method.(*jwt.SigningMethodHMAC); !ok {
  52. return nil, errors.ErrInvalidToken
  53. }
  54. return []byte(defaultKey), nil
  55. },
  56. }
  57. type Option func(*options)
  58. type JWTAuth struct {
  59. opts *options
  60. }
  61. func New(opts ...Option) *JWTAuth {
  62. o := defaultOptions
  63. for _, opt := range opts {
  64. opt(&o)
  65. }
  66. return &JWTAuth{opts: &o}
  67. }
  68. // GenerateToken 生成令牌
  69. func (a *JWTAuth) GenerateToken(userID string) (TokenInfo, error) {
  70. now := time.Now()
  71. expiresAt := now.Add(time.Duration(a.opts.expired) * time.Second).Unix()
  72. token := jwt.NewWithClaims(a.opts.signingMethod, &jwt.StandardClaims{
  73. IssuedAt: now.Unix(),
  74. ExpiresAt: expiresAt,
  75. NotBefore: now.Unix(),
  76. Subject: userID,
  77. })
  78. tokenString, err := token.SignedString(a.opts.signingKey)
  79. if err != nil {
  80. return nil, err
  81. }
  82. tokenInfo := &tokenInfo{
  83. ExpiresAt: expiresAt,
  84. TokenType: a.opts.tokenType,
  85. AccessToken: tokenString,
  86. }
  87. return tokenInfo, nil
  88. }
  89. // 解析令牌
  90. func (a *JWTAuth) parseToken(tokenString string) (*jwt.StandardClaims, error) {
  91. token, _ := jwt.ParseWithClaims(tokenString, &jwt.StandardClaims{}, a.opts.keyfunc)
  92. if !token.Valid {
  93. return nil, errors.ErrInvalidToken
  94. }
  95. return token.Claims.(*jwt.StandardClaims), nil
  96. }
  97. // ParseUserID 解析用户ID
  98. func (a *JWTAuth) ParseUserID(tokenString string) (string, error) {
  99. claims, err := a.parseToken(tokenString)
  100. if err != nil {
  101. return "", err
  102. }
  103. return claims.Subject, nil
  104. }