前后端分离如何实现用户身份认证及鉴权?

薄洪涛5年前Linux2192

假设我们不适用前后端分离,那么登录成功之后,我们会把用户的信息存储到session中,之后的每一次请求,都会带着cookies中的session_id,服务端会自行验证此用户是否登录及登录是否失效,那么问题来了,我们使用了前后端分离,那么就意味着没有了session,每一次的请求都是无状态的,作为后端我们不知道是登录用户请求的还是非登录用户请求的,所以,为了解决这个问题,我们有这么几种方式

  1. 授权的方式,可以参考OAUTH

  2. 用户登录之后,后端返给前端一个有过期时间的token,每次前端请求的时候就携带这个token

  3. 网关鉴权,就是我现在想说的

比较3种方法,前两种是比较常见的,但是都需要在项目的代码中做验证,对于高并发来说,性能还是有一些影响的,如果你使用的语言性能较差的话,每次请求都要在代码中验证token,验证redis等

第三种方法,适合高性能api,现在就详细说下咱们如何实现的

  1.使用openResty,天然支持lua脚本

  2.登录的时候,如果登录成功向redis中写入用户信息及失效时间

  3.要求前端调用接口的时候在header中加入鉴权数据,比如用户id,签名等信息

  4.使用lua去读取header内的内容,比如读取到了用户的id,然后根据自己制定的规则去验证签名,去redis中查询数据来实现鉴权

这块比较涉密,所以我拆分成几个小部分来大体说下思路

  1. lua获取header中的信息

local userId = ngx.req.get_headers()["userid"]
if userId == nil or userId == "" then
	ngx.say("{\"code\": 401, \"msg\":\"userid获取失败\" }")
	return
end

   2. lua连接redis并查询数据

local redis = require "resty.redis"
local red = redis:new()
red:set_timeout(1000)
local ok, err = red:connect('127.0.0.1', '6379')
if not ok then
 	ngx.log(ngx.ERR, "redis 连接失败")
 	ngx.say('{"code":500, "msg":"服务器错误(red err:1)"}')
	return
end
local userInfo, err = red:hmget("键值/",'字段名')

  3. 然后看下nginx的配置,通过access_by_lua_file来加载lua鉴权脚本,这样每次请求都会先去执行lua,然后在去执行我们的代码

server {
    charset utf-8;
    client_max_body_size 128M;

    listen 80; 
    root        /home/www/root/lvs;
    index       index.php;

    location / {
       access_by_lua_file /home/openresty/nginx/conf/lua_script/on_access.lua;
        try_files $uri $uri/ /index.php?$args;
    }

    location ~ \.php$ {
        include fastcgi_params;
        fastcgi_pass 127.0.0.1:9000;
        fastcgi_index index.php;
        fastcgi_param SCRIPT_FILENAME $document_root/$fastcgi_script_name;
        include fastcgi_params; 
    }

    location ~ /\.(ht|svn|git) {
        deny all;
    }
}

看下鉴权失败的情况,我传递了错误的鉴权数据

image.png

传递正确的数据

image.png

鉴权数据失效后,可以要求用户重新登录,然后再次将信息写入redis


标签: lua

相关文章

nginx转换post请求为get

nginx转换post请求为get

今天调银联收银台回调的时候,遇到一件特别坑的事情,我通过银联付款后,本来应该跳转到我自己定义的回调页面,但是每次都会返回405错误码,很是恼火;查了下日志2019/06/24 14:58:3...

时区转换的问题--解决篇

时区转换的问题--解决篇

事情是这样的,我最近做一套DNS服务器的解析的脚本,其中DNS服务器上有一条命令是这样的 [root@qip1200 ~]# date Thu Oct ...

mac下安装homebrew及开发工具

安装homebrew/bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw...

xshell做端口映射链接数据库

xshell做端口映射链接数据库

有些时候,公司的一些线上的数据库,数据太多,不适合复制到本地然后搭建本地数据库开发,直接链接数据库又不安全所以,需要通过xshell做端口转发,通过隧道来链接数据库1、首先通过xshell登陆跳板机2...

如何让命令在关闭终端后继续执行

我们在执行一些比较耗时的脚本的时候,必须要保持终端在线,如果电脑休眠了任务就直接被杀死了这时候我们可以用来执行命令,就会在后台自动执行 nohup 命令 &...

mac下使用docker来构建不同版本的php环境并安装扩展

切换到mac后,我目前php环境是php7.3 ,但是一些老项目需要7.1 甚至5.6 的环境,特别是服务器上也有这种问题,所以我决定使用docker来部署不同的项目首先安装dockerbrew&nb...

评论列表

访客
2019-12-09 10:11:43

写的挺好,楼主是做java后端吗

薄洪涛 回复:
在做,感谢来访
2019-12-11 17:58:32

发表评论    

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。