热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

phpjwtredis实现,token自动续期实现:jwt+redis

传统的会话认证使用的是sessioncookie,前后端分离的项目,更多使用的是token。本文主要实现的token,基于eggjsred

传统的会话认证使用的是session+COOKIE,前后端分离的项目,更多使用的是token。本文主要实现的token,基于eggjs+redis+jwt+COOKIE

token会话流程图

先来了解最终实现的token会话流程图

3edaa366ac2d5e29197391d4ac0f5868.png

对几个关键流程,做以下记录

服务端使用JWT生成token

在使用JWT前,需要提前准备公钥/私钥,怎么生成密钥,可参考:https://www.imqianduan.com/server/RSA-public-private.html

nodejs生成token,使用jsonwebtoken(https://www.npmjs.com/package/jsonwebtoken)

代码

/**

* 生成token

* @param {*} data 数据(eg:{uuid: 'xxx-xx-xx-xxx',role:'admin'})

* @param {*} expired token失效时间

*/

createToken(data,expired){

// 时间:秒

constcreated=Math.floor(Date.now()/1000);

// RSA私钥

constcert=fs.readFileSync(

path.resolve(__dirname,'./rsa_private_key.pem'),

);

// token密钥

// exp:自纪元以来的秒数

consttoken=jwt.sign(

{

data,

exp:created+expired,

},

cert,

{algorithm:'RS256'},

);

returntoken;

},

服务端写入redis

写入redis比较简单,需要注意:

1、redis的过期时间跟token过期时间保持一致

2、下面的redisKey可以在token能找的到

app.redis.set(redisKey,token);

app.redis.expire(redisKey,Expired);

服务端写入COOKIE

注意的两点:

1、设置COOKIE的httpOnly为true

2、maxAge跟token和redis的过期时间一致

3、使用eggjs,可以使用encrypt加密

ctx.COOKIEs.set(config.user.tokenName,tokenData[config.user.tokenName],{

httpOnly:true,

maxAge:tokenExpired*1000,

encrypt:true,

domain:'imqianduan.com',

});

ctx.COOKIEs.set('x-cid',1,{

httpOnly:false,

signed:false,

maxAge:tokenExpired*1000,

domain:'imqianduan.com',

});

前端记录登录状态

前面服务端已经把token写入了COOKIE中,并设置了httpOnly为true,所以JS已经不能读取这个token了,这也是为什么token不能localstorage存储的原因之一

上面一步中,我们还在COOKIE中写了一个x-cid的COOKIE,我们可以通过它来判断是否已经登录

前端发送业务请求

在前端使用最多的框架react/vue中,axios和fetch是最常见的发送http请求的方法之一,但axios和fetch默认都是不携带COOKIE的,要携带COOKIE,都是要进行一定的设置

fetch

设置credentials为include

credentials:'include'

axios

设置withCredentials为true

axios.defaults.withCredentials=true;

服务端中间件验证token

如果token验证通过,会返回加密的data数据 ,如{uuid:’xx-xx-xx-xx’,role:’admin’},根据这个数据 ,可以知道这个用户是谁(以前是服务端在session中记录,会消息服务器内存,现在是加密存到了客户端)

/**

* 验证token

* @param {*} token 登录token

*/

verifyToken(token){

// 公钥

constcert=fs.readFileSync(

path.resolve(__dirname,'./rsa_public_key.pem'),

);

letres='';

try{

constresult=jwt.verify(token,cert,{algorithms:['RS256']})||{};

const{exp}=result,

current=Math.floor(Date.now()/1000);

if(current<&#61;exp){

res&#61;result.data||{};

}

}catch(e){

console.log(e);

}

returnres;

}

**根据token中取到的uuid&#xff0c;查询redis

中间件部分关键代码

constresult&#61;verifyToken(token);

consttokenKey&#61;config.user.redisPrefix.login&#43;result.uuid;

constredisToken&#61;await app.redis.get(tokenKey);

// 服务器端可以找到token

if(redisToken){

// 验证是否为最新的token

if(token&#61;&#61;&#61;redisToken){

consttokenExpired&#61;await app.redis.ttl(tokenKey);

// 如果token超过一半时间&#xff0c;重新续命

if(tokenExpired

// 重新生成token

consttoken&#61;createToken(

{uuid:result.uuid},

config.user.tokenExpired,

);

// 更新redis

app.redis.set(tokenKey,token);

app.redis.expire(tokenKey,config.user.tokenExpired);

// 重置COOKIE值

ctx.COOKIEs.set(config.user.tokenName,token,{

httpOnly:true,

maxAge:config.user.tokenExpired*1000,

encrypt:true,

domain:&#39;imqianduan.com&#39;,

});

}

// 存储用户信息到ctx,便于controller获取

ctx.userInfo&#61;{

uuid:result.uuid,

};

awaitnext();

}else{

// 登录失败移除COOKIE

removeCOOKIE();

ctx.status&#61;422;

// 如果不是最新token&#xff0c;则代表用户在另一个机器上进行操作&#xff0c;需要用户重新登录保存最新token

ctx.body&#61;{

status:1001,

msg:

&#39;您的账号已在其他机器保持登录&#xff0c;如果继续将清除其他机器的登录状态&#39;,

};

}

}

结束

到这里OVER了&#xff0c;剩下的就是请求业务接口的问题了

关于CORS跨域问题

要实现token机制&#xff0c;跨域问题是不得不解决的

eggjs可以使用egg-cors(https://www.npmjs.com/package/egg-cors)

// config/plugin.js

cors:{

enable:true,

package:&#39;egg-cors&#39;,

},

// config/config.default.js

config.cors&#61;{

origin:&#39;http://a.imqianduan.com&#39;,

allowMethods:&#39;GET,POST,OPTIONS&#39;,

// 要使用COOKIE,credentials一定要设置为true

credentials:true,

// maxAge:6000, //seconds

// 如果要使用自定义token,可以在下面设置

allowHeaders:[&#39;x-csrf-token&#39;],

};

nginx

直接上代码&#xff0c;自己去调整吧

server{

listen80;

server_name a.imqianduan.com;

location/{

# add_header &#39;Access-Control-Allow-Origin&#39; "http://a.imqianduan.com";

# add_header &#39;Access-Control-Allow-Methods&#39; &#39;GET, POST, PUT, DELETE, OPTIONS&#39;;

# add_header &#39;Access-Control-Allow-Headers&#39; &#39;DNT,X-CustomHeader,Keep-Alive,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type&#39;;

# add_header &#39;Access-Control-Allow-Credentials&#39; &#39;true&#39;;

# add_header &#39;Access-Control-Expose-Headers&#39; &#39;x-auth-token&#39;;

if($request_method&#61;&#39;OPTIONS&#39;){

add_header&#39;Access-Control-Allow-Origin&#39;"http://a.imqianduan.com";

add_header&#39;Access-Control-Allow-Methods&#39;&#39;GET, POST, PUT, DELETE, OPTIONS&#39;;

add_header&#39;Access-Control-Allow-Headers&#39;&#39;DNT,X-CustomHeader,Keep-Alive,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,x-csrf-token&#39;;

add_header&#39;Access-Control-Allow-Credentials&#39;&#39;true&#39;;

#add_header &#39;Access-Control-Expose-Headers&#39; &#39;x-auth-token&#39;;

return200;

}

proxy_pass http://127.0.0.1:7001;

proxy_set_headerHost$host;

proxy_redirect off;

proxy_set_header X-Real-IP $remote_addr;

proxy_set_header X-Forwarded-For$proxy_add_x_forwarded_for;

proxy_set_headerAccess-Control-Allow-Credentials&#39;true&#39;;

proxy_connect_timeout60;

proxy_read_timeout60;

proxy_send_timeout60;

}

}



推荐阅读
author-avatar
Mr_cool
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有