摘要:前言本文将会从最基本的一种权限验证说起,即,然后是基于和的权限验证,最后则是和一次性密码。由于它不依赖于,等任何外部因素,所以它是最简单的权限验证方法。当使用来进行权限验证时,有以下几点需要注意。
前言
本文将会从最基本的一种web权限验证说起,即HTTP Basic authentication,然后是基于cookies和tokens的权限验证,最后则是signatures和一次性密码。
HTTP Basic authentication当客户端发起一个请求时,它可以使用HTTP Basic authentication来提供一个用户名和密码,来进行权限验证。
由于它不依赖于cookies,sessions等任何外部因素,所以它是最简单的权限验证方法。在使用它时,客户端需要在每次请求时,都附带上Authorization请求头,用户名和密码都不会被加密,但是需要被格式化为以下结构:
用户名和密码由一个冒号连接,如username:password
这个字符串需进行Base64编码
Basic关键字需被放置在这个编码后的字符串的前面
例子:
curl --header "Authorization: Basic am9objpzZWNyZXQ=" my-website.com
在Node.js中实现它是非常简单的,以下是一个通过Express中间件来实现的例子:
import basicAuth from "basic-auth"; function unauthorized(res) { res.set("WWW-Authenticate", "Basic realm=Authorization Required"); return res.send(401); }; export default function auth(req, res, next) { const {name, pass} = basicAuth(req) || {}; if (!name || !pass) { return unauthorized(res); }; if (name === "john" && pass === "secret") { return next(); } return unauthorized(res); };
当然,你也可以在更高层上实现它,如nginx。
HTTP Basic authentication虽然十分简单,但仍有一些需要注意的地方:
用户名和密码在每次请求时都会被带上,即使请求是通过安全连接发送的,这也是潜在的可能暴露它们的地方。
如果网站使用的加密方法十分弱,或者被破解,那么用户名和密码将会马上泄露。
用户通过这种方式进行验证时,并没有登出的办法
同样,登陆超时也是没有办法做到的,你只能通过修改用户的密码来模拟。
Cookies当服务端在响应HTTP请求时,它可以在响应头里加上Set-Cookie头。然后浏览器会将这个cookie保存起来,并在以后请求同源的地址时,在Cookie请求头中附上这些cookie。
当使用cookies来进行权限验证时,有以下几点需要注意。
总是将cookies设为HttpOnly当设置cookies时,总是使用HttpOnly标识,这样以来cookies就不能通过document.cookies获取,用以减少被XSS攻击可能性。
总是使用签名(signed) cookies当使用签名cookies时,服务器则可以判断该cookie是否被客户端更改过。
不足:
需要花费额外的功夫来抵御CSRF攻击
与REST风格不匹配。因为它在一个无状态协议里注入了状态。
Tokens现今,JWT(JSON Web Token)无处不在。让我们先来看看它到底长什么样。
JWT由三部分组成:
Header,由token的类型和哈希算法组成
Payload,包含了内容主体
Signature,当你选择HMAC SHA256算法时,它由HMACSHA256( base64UrlEncode(header) + "." + base64UrlEncode(payload), secret)计算得出。
将你的Koa应用加上JWT仅需几行代码:
var koa = require("koa"); var jwt = require("koa-jwt"); var app = koa(); app.use(jwt({ secret: "very-secret" })); // Protected middleware app.use(function *(){ // content of the token will be available on this.state.user this.body = { secret: "42" }; });
例子:
curl --header "Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWV9.TJVA95OrM7E2cBab30RMHrHDcEfxjoYZgeFONFh7HgQ" my-website.com
如果你在写提供给原生移动应用或单页web应用的API,JWT是一个不错的选择。
不足:
需要额外的措施来防护XSS攻击
Signatures不论是使用cookies还是token,如果你传输的内容被他人截获,那么它们将可以很容易得伪装成真实的用户。
如果解决这个问题?当我们讨论的是API之间的通信,而不是浏览器之间的通信时,有一个办法。
当API的消费者发送一个需要权限验证的请求时,你可以对整个请求用一个私钥进行哈希。你可以使用的请求的内容有:
HTTP方法
请求路径
HTTP头
HTTP体
以及一个私钥
API的消费者和提供者都必须持有相同的私钥。在生成了signature之后,你必须将其加在query string或HTTP头中。另外,还需附上一个时间戳,用于判断过期。
当这么做时,即使你传输的内容暴露了,攻击者也无法伪装成真实用户,因为它无法自己生成signature。
不足:
不能用于浏览器/客户端中,只能用于API之间的通信中。
一次性密码一次性密码算法使用一个共享的密钥和一个当前时间戳或计数器来生成一个一次性密码:
基于时间的一次性密码算法,使用一个当前时间的时间戳
基于HMAC的一次性密码算法,使用一个计数器
这些方法被用于双重认证(two-factor authentication)中:一个用户输入了用户名和密码,然后服务器和客户端同时生成一个一次性密码。
在Node.js中,使用notp实现它是相对简单的。
不足:
如果共享密钥被窃取,那么用户的token将可以被伪造
该在何时选择何种验证方法?如果你只需支持一个web应用,那么cookies和tokens的实现都是可以的(cookies对XSRF的防护较好,而JWT则更易于防护XSS)。
如果你需要同时支持web应用和移动客户端,那么请使用基于token的验证。
如果你正在构建仅与其他API通信的API,那么就使用signatures。
最后原文链接:https://blog.risingstack.com/web-authentication-methods-explained/
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/91562.html
摘要:框架具有轻便,开源的优点,所以本译见构建用户管理微服务五使用令牌和来实现身份验证往期译见系列文章在账号分享中持续连载,敬请查看在往期译见系列的文章中,我们已经建立了业务逻辑数据访问层和前端控制器但是忽略了对身份进行验证。 重拾后端之Spring Boot(四):使用JWT和Spring Security保护REST API 重拾后端之Spring Boot(一):REST API的搭建...
showImg(https://segmentfault.com/img/bV6aHV?w=1280&h=800); 社区优秀文章 Laravel 5.5+passport 放弃 dingo 开发 API 实战,让 API 开发更省心 - 自造车轮。 API 文档神器 Swagger 介绍及在 PHP 项目中使用 - API 文档撰写方案 推荐 Laravel API 项目必须使用的 8 个...
摘要:探究系统登录验证码的实现后端掘金验证码生成类手把手教程后端博客系统第一章掘金转眼间时间就从月份到现在的十一月份了。提供了与标准不同的工作方式我的后端书架后端掘金我的后端书架月前本书架主要针对后端开发与架构。 Spring Boot干货系列总纲 | 掘金技术征文 - 掘金原本地址:Spring Boot干货系列总纲博客地址:http://tengj.top/ 前言 博主16年认识Spin...
阅读 2546·2021-11-24 10:20
阅读 2394·2021-09-10 10:51
阅读 3381·2021-09-06 15:02
阅读 3117·2019-08-30 15:55
阅读 2841·2019-08-29 18:34
阅读 3082·2019-08-29 12:14
阅读 1218·2019-08-26 13:53
阅读 2933·2019-08-26 13:43