Spring Boot实战之Filter实现使用JWT进行接口认证

Spring Boot实战之Filter实现使用JWT进行接口认证

jwt(json web token)

用户发送按照约定,向服务端发送 Header、Payload 和 Signature,并包含认证信息(密码),验证通过后服务端返回一个token,之后用户使用该token作为登录凭证,适合于移动端和api

jwt使用流程

本文示例接上面几篇文章中的代码进行编写,请阅读本文的同时可以参考前面几篇文章

1、添加依赖库jjwt,本文中构造jwt及解析jwt都使用了jjwt库


[html] view plain copy

  

io.jsonwebtoken  

jjwt  

0.6.0  


2、添加登录获取token时,所需要的认证信息类LoginPara.java

[java] view plain copy

package com.xiaofangtech.sunt.jwt;  


public class LoginPara {  

private String clientId;  

private String userName;  

private String password;  

private String captchaCode;  

private String captchaValue;  


public String getClientId() {  

return clientId;  

    }  

public void setClientId(String clientId) {  

this.clientId = clientId;  

    }  

public String getUserName() {  

return userName;  

    }  

public void setUserName(String userName) {  

this.userName = userName;  

    }  

public String getPassword() {  

return password;  

    }  

public void setPassword(String password) {  

this.password = password;  

    }  

public String getCaptchaCode() {  

return captchaCode;  

    }  

public void setCaptchaCode(String captchaCode) {  

this.captchaCode = captchaCode;  

    }  

public String getCaptchaValue() {  

return captchaValue;  

    }  

public void setCaptchaValue(String captchaValue) {  

this.captchaValue = captchaValue;  

    }  

}  

3、添加构造jwt及解析jwt的帮助类JwtHelper.java

[java] view plain copy

package com.xiaofangtech.sunt.jwt;  


import java.security.Key;  

import java.util.Date;  


import javax.crypto.spec.SecretKeySpec;  

import javax.xml.bind.DatatypeConverter;  


import io.jsonwebtoken.Claims;  

import io.jsonwebtoken.JwtBuilder;  

import io.jsonwebtoken.Jwts;  

import io.jsonwebtoken.SignatureAlgorithm;  


public class JwtHelper {  

public static Claims parseJWT(String jsonWebToken, String base64Security){  

try  

        {  

            Claims claims = Jwts.parser()  

                       .setSigningKey(DatatypeConverter.parseBase64Binary(base64Security))  

                       .parseClaimsJws(jsonWebToken).getBody();  

return claims;  

        }  

catch(Exception ex)  

        {  

return null;  

        }  

    }  


public static String createJWT(String name, String userId, String role,   

String audience, String issuer,long TTLMillis, String base64Security)   

    {  

        SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;  


long nowMillis = System.currentTimeMillis();  

Date now =new Date(nowMillis);  


//生成签名密钥  

byte[] apiKeySecretBytes = DatatypeConverter.parseBase64Binary(base64Security);  

Key signingKey =new SecretKeySpec(apiKeySecretBytes, signatureAlgorithm.getJcaName());  


//添加构成JWT的参数  

JwtBuilder builder = Jwts.builder().setHeaderParam("typ", "JWT")  

.claim("role", role)  

.claim("unique_name", name)  

.claim("userid", userId)  

                                        .setIssuer(issuer)  

                                        .setAudience(audience)  

                                        .signWith(signatureAlgorithm, signingKey);  

//添加Token过期时间  

if (TTLMillis >= 0) {  

long expMillis = nowMillis + TTLMillis;  

Date exp =new Date(expMillis);  

            builder.setExpiration(exp).setNotBefore(now);  

        }  


//生成JWT  

return builder.compact();  

    }   

}  

4、添加token返回结果类AccessToken.java

[java] view plain copy

package com.xiaofangtech.sunt.jwt;  


public class AccessToken {  

private String access_token;  

private String token_type;  

private long expires_in;  

public String getAccess_token() {  

return access_token;  

    }  

public void setAccess_token(String access_token) {  

this.access_token = access_token;  

    }  

public String getToken_type() {  

return token_type;  

    }  

public void setToken_type(String token_type) {  

this.token_type = token_type;  

    }  

public long getExpires_in() {  

return expires_in;  

    }  

public void setExpires_in(long expires_in) {  

this.expires_in = expires_in;  

    }  

}  

5、添加获取token的接口,通过传入用户认证信息(用户名、密码)进行认证获取

[java] view plain copy

package com.xiaofangtech.sunt.jwt;  


import org.springframework.beans.factory.annotation.Autowired;  

import org.springframework.web.bind.annotation.RequestBody;  

import org.springframework.web.bind.annotation.RequestMapping;  

import org.springframework.web.bind.annotation.RestController;  


import com.xiaofangtech.sunt.bean.UserInfo;  

import com.xiaofangtech.sunt.repository.UserInfoRepository;  

import com.xiaofangtech.sunt.utils.MyUtils;  

import com.xiaofangtech.sunt.utils.ResultMsg;  

import com.xiaofangtech.sunt.utils.ResultStatusCode;  


@RestController  

public class JsonWebToken {  

@Autowired  

private UserInfoRepository userRepositoy;  


@Autowired  

private Audience audienceEntity;  


@RequestMapping("oauth/token")  

public Object getAccessToken(@RequestBody LoginPara loginPara)  

    {  

        ResultMsg resultMsg;  

try  

        {  

if(loginPara.getClientId() == null   

|| (loginPara.getClientId().compareTo(audienceEntity.getClientId()) !=0))  

            {  

resultMsg =new ResultMsg(ResultStatusCode.INVALID_CLIENTID.getErrcode(),   

ResultStatusCode.INVALID_CLIENTID.getErrmsg(),null);  

return resultMsg;  

            }  


//验证码校验在后面章节添加  



//验证用户名密码  

            UserInfo user = userRepositoy.findUserInfoByName(loginPara.getUserName());  

if (user == null)  

            {  

resultMsg =new ResultMsg(ResultStatusCode.INVALID_PASSWORD.getErrcode(),  

ResultStatusCode.INVALID_PASSWORD.getErrmsg(),null);  

return resultMsg;  

            }  

else  

            {  

                String md5Password = MyUtils.getMD5(loginPara.getPassword()+user.getSalt());  


if (md5Password.compareTo(user.getPassword()) != 0)  

                {  

resultMsg =new ResultMsg(ResultStatusCode.INVALID_PASSWORD.getErrcode(),  

ResultStatusCode.INVALID_PASSWORD.getErrmsg(),null);  

return resultMsg;  

                }  

            }  


//拼装accessToken  

            String accessToken = JwtHelper.createJWT(loginPara.getUserName(), String.valueOf(user.getName()),  

                    user.getRole(), audienceEntity.getClientId(), audienceEntity.getName(),  

audienceEntity.getExpiresSecond() *1000, audienceEntity.getBase64Secret());  


//返回accessToken  

AccessToken accessTokenEntity =new AccessToken();  

            accessTokenEntity.setAccess_token(accessToken);  

            accessTokenEntity.setExpires_in(audienceEntity.getExpiresSecond());  

accessTokenEntity.setToken_type("bearer");  

resultMsg =new ResultMsg(ResultStatusCode.OK.getErrcode(),   

                    ResultStatusCode.OK.getErrmsg(), accessTokenEntity);  

return resultMsg;  


        }  

catch(Exception ex)  

        {  

resultMsg =new ResultMsg(ResultStatusCode.SYSTEM_ERR.getErrcode(),   

ResultStatusCode.SYSTEM_ERR.getErrmsg(),null);  

return resultMsg;  

        }  

    }  

}  

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容