jwt的簡單使用

簡介

使用jwt對數據進行存儲加密,分爲java和golang版本。git


java版本

  • maven配置github

    <dependency>
        <groupId>com.auth0</groupId>
        <artifactId>java-jwt</artifactId>
        <version>3.4.0</version>
    </dependency>
  • 工具類 JWTUtil.javagolang

    import com.auth0.jwt.JWT;
    import com.auth0.jwt.JWTVerifier;
    import com.auth0.jwt.algorithms.Algorithm;
    import com.auth0.jwt.interfaces.DecodedJWT;
    import lombok.Getter;
    import org.springframework.boot.context.properties.ConfigurationProperties;
    import org.springframework.stereotype.Component;
    
    import java.util.Date;
    
    public class JWTUtil {
    
        private final static String ISS = "dust";
    
        private final static String SECRET = "secret";
    
        // 過時時間是3600秒,既是1個小時
        private final static long EXPIRATION = 3600;
    
    
        /**
         * 生成token
         * @param userId 用戶id
         * @return token
         */
        public static String createToken(long userId){
            Algorithm algorithm = Algorithm.HMAC384(SECRET);
            String token = JWT.create()
                      .withIssuer(ISS).withIssuedAt(new Date()).withExpiresAt(new Date(System.currentTimeMillis() + EXPIRATION * 1000))
                    .withClaim("USER_ID",userId)
                    .sign(algorithm);
    
            return token;
        }
    
        /**
         * 解析token
         * @param token token
         * @return long
         */
        public static long analyzeToken(String token){
            DecodedJWT decode = JWT.decode(token);
            return decode.getClaim("USER_ID").asLong();
        }
    
        /**
         * 校驗token的合法性
         * @param token token
         * @return  boolean
         */
        public static boolean verifyToken(String token){
            Algorithm algorithm = Algorithm.HMAC384(SECRET);
            JWTVerifier verifier = JWT.require(algorithm)
                    .withIssuer(ISS)
                    .build(); //Reusable verifier instance
            try {
                // 驗證不經過會出現異常
                verifier.verify(token);
            } catch(Exception ex){
                return false;
            }
            return true;
        }
    
    }
  • 測試 JWTTests.javaspring

    import org.junit.Test;
    public class JWTTests {
        @Test
        public void createTokenTest(){
            String token = JwtUtil.createToken(12345678);
            System.out.println(token); // eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzM4NCJ9.eyJpc3MiOiJkdXN0IiwiVVNFUl9JRCI6MTIzNDU2Nzh9.qUNgiHKB5uIWhhpf8GM6hdcaE9hTvDJG9UmmMpcCPpfncduQ713aKI7VMhCedJWP
        }
    
        @Test
        public void decodeTokenTest(){
            String token = "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzM4NCJ9.eyJpc3MiOiJkdXN0IiwiVVNFUl9JRCI6MTIzNDU2Nzh9.qUNgiHKB5uIWhhpf8GM6hdcaE9hTvDJG9UmmMpcCPpfncduQ713aKI7VMhCedJWP";
            System.out.println(JwtUtil.analyzeToken(token)); // 12345678
        }
    
        @Test
        public void verifyTokenTest(){
            String token = "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzM4NCJ9.eyJpc3MiOiJkdXN0IiwiVVNFUl9JRCI6MTIzNDU2Nzh9.qUNgiHKB5uIWhhpf8GM6hdcaE9hTvDJG9UmmMpcCPpfncduQ713aKI7VMhCedJWP";
            System.out.println(JwtUtil.verifyToken(token)); // true
        }
    }

golang版本

  • 導入包json

    go get -u github.com/dgrijalva/jwt-
  • JWTUtil.gomaven

    package testcase
    
    import (
      "errors"
      "log"
      "time"
      "github.com/dgrijalva/jwt-go"
    )
    
    type JWT struct {
      SigningKey []byte
    }
    var (
      TokenExpired error = errors.New("Token is expired")
      TokenNotValidYet error = errors.New("Token not active yet")
      TokenMalformed error = errors.New("That's not even a token")
      TokenInvalid error = errors.New("Couldn't handle this token:")
      SignKey string = "dust"
    )
    type CustomClaims struct {
      ID int64 `json:"id"`
      jwt.StandardClaims
    }
    func NewJWT() *JWT {
      return &JWT{
          []byte(GetSignKey()),
      }
    }
    func GetSignKey() string {
      return SignKey
    }
    func SetSignKey(key string) string {
      SignKey = key
      return SignKey
    }
    func (j *JWT) CreateToken(claims CustomClaims) (string, error) {
      token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
      return token.SignedString(j.SigningKey)
    }
    func (j *JWT) ParseToken(tokenString string) (*CustomClaims, error) {
      token, err := jwt.ParseWithClaims(tokenString, &CustomClaims{}, func(token *jwt.Token) (interface{}, error) {
          return j.SigningKey, nil
      })
      if err != nil {
          if ve, ok := err.(*jwt.ValidationError); ok {
              if ve.Errors&jwt.ValidationErrorMalformed != 0 {
                  return nil, TokenMalformed
              } else if ve.Errors&jwt.ValidationErrorExpired != 0 {
                  // Token is expired
                  return nil, TokenExpired
              } else if ve.Errors&jwt.ValidationErrorNotValidYet != 0 {
                  return nil, TokenNotValidYet
              } else {
                  return nil, TokenInvalid
              }
          }
      }
      if claims, ok := token.Claims.(*CustomClaims); ok && token.Valid {
          return claims, nil
      }
      return nil, TokenInvalid
    }
    
    func (j *JWT) RefreshToken(tokenString string) (string, error) {
      jwt.TimeFunc = func() time.Time {
          return time.Unix(0, 0)
      }
      token, err := jwt.ParseWithClaims(tokenString, &CustomClaims{}, func(token *jwt.Token) (interface{}, error) {
          return j.SigningKey, nil
      })
      if err != nil {
          return "", err
      }
      if claims, ok := token.Claims.(*CustomClaims); ok && token.Valid {
          jwt.TimeFunc = time.Now
          claims.StandardClaims.ExpiresAt = time.Now().Add(1 * time.Hour).Unix()
          return j.CreateToken(*claims)
      }
      return "", TokenInvalid
    }
    
    // 生成令牌
    func GenerateToken(id int64) (token string, err error) {
      j := NewJWT()
      claims := CustomClaims{
          id,
          jwt.StandardClaims{
              //NotBefore: int64(time.Now().Unix() - 1000), // 簽名生效時間
              ExpiresAt: int64(time.Now().Unix() + 1*60), // 過時時間 一小時
              Issuer:    "dust",                          //簽名的發行者
          },
      }
    
      if token, err = j.CreateToken(claims); err != nil {
          log.Printf( "j.CreateToken(claims) err:%s", err.Error())
          return
      }
    
      return token, nil
    }
  • 測試JWTUtil_test.go工具

    package testcase
    
    import (
      "testing"
      "time"
    )
    
    func TestGenerateToken(t *testing.T) {
      var (
          token string
          err   error
      )
    
      if token, err = GenerateToken(123456); err != nil {
          t.Fatalf("GenerateToken錯誤,錯誤信息:%s", err.Error())
      }
      t.Logf("token 值:%s", token)
    }
    
    func TestParseToken(t *testing.T) {
      var (
          err    error
          claims *CustomClaims
      )
      token := "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6MjM0NTY3LCJleHAiOjE1NjI4MTAyNTIsImlzcyI6ImR1c3QifQ.OxY2F0mOX8Y8XhMPHcgxmXENyHAwg_i9eCWqokIw0QE"
      jwt := NewJWT()
      if claims, err = jwt.ParseToken(token); err != nil {
          t.Fatalf("ParseToken錯誤,錯誤信息:%s", err.Error())
      }
      t.Logf("claims中ID 值:%d", claims.ID)
    }
    
    func TestRefreshToken(t *testing.T) {
      var (
          err    error
          newToken string
      )
      //token := "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6MTIzNDU2LCJleHAiOjE1NjI3NTA3NzcsImlzcyI6ImR1c3QiLCJuYmYiOjE1NjI3NDk3MTd9.W-qTms5UeuBRtF6VfksaZeZANZWfP5NdFHwYPGDMp98"
      jwt := NewJWT()
      createToken, _ := GenerateToken(234567)
      time.Sleep(62 * time.Second)
      if _, err = jwt.ParseToken(createToken); err != nil && err == TokenExpired {
          if newToken,err = jwt.RefreshToken(createToken); err == nil {
              t.Logf("newToken 值:%s", newToken)
          } else {
              t.Fatalf("RefreshToken錯誤,錯誤信息:%s", err.Error())
          }
      }
    }
相關文章
相關標籤/搜索