session-cookie 和token登录验证

  最近研究了下基于token的身份验证,并将这种机制整合在个人项目中。现在很多网站的认证方式都从传统的seesion+cookie转向token校验。对比传统的校验方式,token确实有更好的扩展性与安全性。
  
  传统的session+cookie身份验证
  
  由于HTTP是无状态的,它并不记录用户的身份。用户将账号与密码发送给服务器后,后台通过校验,但是并没有记录状态,于是下一次用户的请求仍然需要校验身份。为了解决这一问题,需要在服务端生成一条包含用户身份的记录,也就是session,再将这条记录发送给用户并存储在用户本地,即cookie。接下来用户的请求都会带上这条cookie,若客户端的cookie与服务端的session能对应上,则说明用户身份验证通过。
  
  token身份校验
  
  流程大致如下:
  
  第一次请求时,用户发送账号与密码
  
  后台校验通过,则会生成一个有时效性的token,再将此token发送给用户
  
  用户获得token后,将此token存储在本地,一般存储在localstorage或cookie
  
  之后的每次请求都会将此token添加在请求头里,所有需要校验身份的接口都会被校验token,若token解析后的数据包含用户身份信息,则身份验证通过。
  
  对比传统的校验方式,token校验有如下优势:
  
  在基于token的认证,token通过请求头传输,而不是把认证信息存储在session或者cookie中。这意味着无状态。你可以从任意一种可以发送HTTP请求的终端向服务器发送请求。
  
  可以避免CSRF攻击
  
  当在应用中进行 session的读,写或者删除操作时,会有一个文件操作发生在操作系统的temp 文件夹下,至少在第一次时。假设有多台服务器并且 session 在第一台服务上创建。当你再次发送请求并且这个请求落在另一台服务器上,session 信息并不存在并且会获得一个“未认证”的响应。我知道,你可以通过一个粘性 session 解决这个问题。然而,在基于 token 的认证中,这个问题很自然就被解决了。没有粘性 session 的问题,因为在每个发送到服务器的请求中这个请求的 token 都会被拦截。
  
  下面介绍一下利用node+jwt(jwt教程)搭建简易的token身份校验
  
  示例
  
  当用户第一次登录时,提交账号与密码至服务器,服务器校验通过,则生成对应的token,代码如下:
  
  const fs = require('fs');
  
  const path = require('path');
  
  const jwt = require('jsonwebtoken');
  
  //生成token的方法
  
  function generateToken(data){
  
  let created = Math.floor(Date.now() / 1000);
  
  let cert = fs.readFileSync(path.join(__dirname, '../config/pri.pem'));//私钥
  
  let token = jwt.sign({
  
  data,
  
  exp: created + 3600 * 24
  
  }, cert, {algorithm: 'RS256'});
  
  return token;
  
  }
  
  //登录接口
  
  router.post('/oa/login', async (ctx, next) => {
  
  let data = ctx.request.body;
  
  let {name, password} = data;
  
  let sql = 'SELECT uid FROM t_user WHERE name=? and password=? and is_delete=0', value = [name, md5(password)];
  
  await db.query(sql, value).then(res => {
  
  if (res && res.length > 0) {
  
  let val = res[0];
  
  let uid = val['uid'];
  
  let token = generateToken({uid});
  
  ctx.body = {
  
  ...Tips[0], data: {token}
  
  }
  
  } else {
  
  ctx.body www.michenggw.com/= Tips[1006];
  
  }
  
  }).catch(e =www.yigouyule2.cn> {
  
  ctx.body = www.mhylpt.com/ Tips[1002];
  
  });
  
  });
  
  用户通过校验将获取到的token存放在本地:
  
  ?
  
  1
  
  store.set('loginedtoken',token);//store为插件
  
  之后客户端请求需要验证身份的接口,都会将token放在请求头里传递给服务端:

  service.interceptors.request.use(config => {
  
  let params = config.params |www.gcyl158.com| {};
  
  let loginedtoken = store.get('loginedtoken');
  
  let time = Date.now(www.gcyL157.com);
  
  let {headers} = config;
  
  headers = {.www.feifanyule.cn/..headers,loginedtoken};
  
  params = {...params,_:time};
  
  config = {...config,params,headers};
  
  return config;
  
  }, error => {
  
  Promise.reject(error);
  
  })
  
  服务端对所有需要登录的接口均拦截token并校验合法性。
  
  ?
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  8
  
  9
  
  10
  
  11
  
  12
  
  13
  
  14
  
  15
  
  16
  
  17
  
  18
  
  19
  
  20
  
  21
  
  22
  
  23
  
  24
  
  25
  
  26
  
  27
  
  28
  
  29
  
  30
  
  31
  
  32
  
  33
  
  34
  
  35
  
  36
  
  function verifyToken(token){
  
  let cert = fs.readFileSync(path.join(__dirname, '../config/pub.pem'));//公钥
  
  try{
  
  let result = jwt.verify(token, cert, {algorithms: ['RS256']}) || {};
  
  let {exp = 0} = result,current = Math.floor(Date.now()/1000);
  
  if(current <= exp){
  
  res = result.data || {};
  
  }
  
  }catch(e){
  
  }
  
  return res;
  
  }
  
  app.use(async(ctx, next) => {
  
  let {url = ''} = ctx;
  
  if(url.indexOf('/user/') > -1){//需要校验登录态
  
  let header = ctx.request.header;
  
  let {loginedtoken} = header;
  
  if (loginedtoken) {
  
  let result = verifyToken(loginedtoken);
  
  let {uid} = result;
  
  if(uid){
  
  ctx.state = {uid};
  
  await next();
  
  }else{
  
  return ctx.body = Tips[1005];
  
  }
  
  } else {
  
  return ctx.body = Tips[1005];
  
  }
  
  }else{
  
  await next();
  
  }
  
  });
  
  本示例使用的公钥与私钥可自己生成,操作如下:
  
  打开命令行工具,输入openssl,打开openssl;
  
  生成私钥:genrsa -out rsa_private_key.pem 2048
  
  生成公钥: rsa -in rsa_private_key.pem -pubout -out rsa_public_key.pem

转载于:https://www.cnblogs.com/qwangxiao/p/10057156.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/265317.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

高效使用Bitmaps(一) 大Bitmap的加载

转自&#xff1a;http://my.oschina.net/rengwuxian/blog/182885 高效使用Bitmaps有什么好处&#xff1f; 我 们常常提到的“Android程序优化”&#xff0c;通常指的是性能和内存的优化&#xff0c;即&#xff1a;更快的响应速度&#xff0c;更低的内存占用。Android程序的性能…

android自动软键盘,Android自定义软键盘

MyKeyboardAndroid自定义键盘的使用实现步骤第一步&#xff1a;1、新建一个xml文件夹放在res目录下面&#xff0c;然后新建xml文件:money_keyboard.xml2、然后在XML文件中添加按钮布局&#xff0c;这个布局就是键盘的样子了android:horizontalGap"1dp"android:keyWid…

433M射频遥控灯、震动感应灯、WIFI避障小车

目录 433M射频遥控灯1. 433M射频灯的硬件连接2. 具体接线剖析3.课堂干货简述4. 准备工作之环境搭建KEIL_ISP_USBTOTTL驱动5. KEIL建立工程6.电子可编程开关继电器的原理7. 上电灯亮代码编写8. 烧写程序到单片机,实现灯一亮一灭9.遥控灯编程实现及效果验证震动感应灯1. 入门项目…

Bootstrap使用-1

目录 结构&#xff1a;1. 视图函数2. 模板3. 登陆4. 怎样发生的添加模板基础模板提供的block定制基础模板结构&#xff1a; $ tree -I "__pycache*|*.pyc|*.xlsx" -FCL 3 . |-- templates/ | -- h1.html -- test-boostracp.py 1. 视图函数 test-boostracp.py from …

测试 SyntaxHighlighter

#include <stdio.h>int main() {return 0; }转载于:https://www.cnblogs.com/gongnaixiao/p/3472817.html

段错误、内存溢出、内存泄漏(区别)、堆溢出、栈溢出

参考&#xff1a;内存泄漏、内存溢出、段错误、堆溢出、栈溢出 作者&#xff1a;焦木白 发布时间&#xff1a;2019-10-22 网址&#xff1a;https://blog.csdn.net/jiaomubai/article/details/102680705?spm1001.2014.3001.5501 目录段错误内存溢出内存泄漏栈溢出堆溢出段错误 …

12.2号

今日活动&#xff1a;继续改善软件时间&#xff1a;10.00-11.00地点&#xff1a;自习室 转载于:https://www.cnblogs.com/xinyue6/p/10072515.html

表单文本框提示字符点击时清空

表单文本框提示字符点击时清空&#xff08;input标签onfocus时文本框内提示信息清空&#xff09; 作用是&#xff0c;一个文本框&#xff0c;需要输入内容&#xff0c;在没有输入的时候里面有一段提示内容&#xff0c;当点击这个文本框输入的时候&#xff0c;文本框内的内容自…

C语言中指针的地址和内容

参考&#xff1a;C语言中指针的地址和内容 作者&#xff1a;wallying 发布时间&#xff1a;2018-10-18 09:47:54 网址&#xff1a;https://blog.csdn.net/qq_36631580/article/details/89375140?spm1001.2014.3001.5502 #include <stdio.h>//一般用"地址"表示…

android运行过程简书,android 启动过程(一)

1、由init.rc调用 app_main.cpp的main方法&#xff0c;启动zygote进程3、调用AndroidRuntime.cpp 的start方法传递ZygoteInit类路径5、AndroidRuntime.cpp的start方法主要启动java虚拟机跟注册虚拟机&#xff0c;调用ZygoteInit的main方法6、ZygoteInit的main方法主要 把zygote…

鼠标事件

<!DOCTYPE html><html lang"en"><head><meta charset"UTF-8"><title>mouse 事件</title><script type"text/javascript" src" ./js/jquery.min.js"></script><script type"t…

将会改变未来IT世界的十种编程语言

这里要说的都是革新&#xff0c;说这些的目的就是要保持关注最新技术。如果你是一个程序员&#xff0c;想要探寻未来技术&#xff0c;那这篇文章就是你的必读之选。我们这里列出了10种编程语言&#xff0c;10种将会改变IT世界工作方式的编程语言。这些语言已经在开始改变IT界的…

回调函数 —— 借助中间通用函数(形参里有函数指针,实现函数注册)调用不同的回调函数 (多态/分层)

回调函数传参&#xff1a;函数指针做函数参数&#xff08;回调函数&#xff09; 目录背景回调函数是实现函数分层且单向依赖的好办法&#xff0c;使用函数指针运行struct结构体回调函数代码更清晰简单理解回调函数Demo其他回调函数博文背景 这是我在实际工作中遇到的问题&…

linux下用户操作

在linux中添加ftp用&#xff0c;并设置相应的权限&#xff0c;操作步骤如下&#xff1a; 1、环境&#xff1a;ftp为vsftp。被限制用户名为test。被限制路径为/home/test。 2、建用户&#xff1a;在root用户下&#xff1a; useradd -d /home/test test //增加用户test&#…

Venkat 演讲翻译:你要清除代码中的异味

今天&#xff0c;Venkat Subramaniam 就关于清除代码异味的话题给我们做了一个非常有趣的演讲。下面就是我记录的一些他的话。 为什么我们需要有质量的代码&#xff1f; 敏捷开发方法是用来应付那些要求代码做大量改动的反馈信息的方法。如果程序没有用一种好的表达方式来表现&…

ESP8266等模块

模块讲解学习视频可参照&#xff1a;https://space.bilibili.com/323745961?spm_id_from333.788.b_765f7570696e666f.1 作者&#xff1a;叁议电子 目录ESP8266 WIFI模块介绍电脑操作ESP8266单片机上网ESP8266 WIFI模块介绍 电脑操作ESP8266 单片机上网

html5在线音乐列表播放器,HTML5列表音乐播放器SMusic

插件描述&#xff1a;一款基于HTML5、Css3的列表式音乐播放器&#xff0c;包含列表&#xff0c;音量&#xff0c;进度&#xff0c;时间&#xff0c;歌词展示以及模式等功能&#xff0c;不依赖任何库SMusic使用方法这是一款基于HTML5以及CSS3的列表式音乐播放器&#xff0c;增加…

unzipping/Users/xq/.gradle/wrapper /dists/gradle-3.3-all/55gk2rcmfc6p2dg9u9ohc3hw9/gradle-3.3-all.zi

unzipping/Users/xq/.gradle/wrapper /dists/gradle-3.3-all/55gk2rcmfc6p2dg9u9ohc3hw9/gradle-3.3-all.zip to /Users/xq/.gradle/wrapper/dists/gradle-3.3-all/55gk2rcmfc6p2dg9u9ohc3hw9 可能是下载时网络原因造成文件缺失 解决方法&#xff1a; cd /Users/xq/.gradle/w…

WIFI小车APP

对应这篇文章&#xff1a;433M射频遥控灯、震动感应灯、WIFI避障小车 目录代码概要activity_contrl.xmlContrlActivity.javaNetUtils.java详细工程代码下载代码概要 activity_contrl.xml <RelativeLayout xmlns:android"http://schemas.android.com/apk/res/android&qu…

[Web App]必胜客宅急送产品设计思路介绍[转]

O2O模式是餐饮业在移动消费趋势下主动拥抱互联网的方向&#xff0c;迎合餐饮消费者从以往经验判断为主转变为依靠移动设备、LBS、社交网络进行立体决策的过程。App成为联系消费者和餐饮业的重要纽带&#xff0c;承载着在O2O双向流程的闭环中加深消费者对品牌的认同和感情认知的…