zoukankan      html  css  js  c++  java
  • OAuth2.0开放授权

    OAuth 2.0 是目前最流行的授权机制,用来授权第三方应用,获取用户数据。简单说,OAuth 就是一种授权机制。数据的所有者告诉系统,同意授权第三方应用进入系统,获取这些数据。系统从而产生一个短期的进入令牌(token),用来代替密码,供第三方应用使用。

    名词解释:

    (1) Third-party application:第三方应用程序,本文中又称"客户端"(client)。

    (2)HTTP service:HTTP服务提供商,本文中简称"服务提供商"。

    (3)Resource Owner:资源所有者,本文中又称"用户"(user)。

    (4)User Agent:用户代理,本文中就是指浏览器。

    (5)Authorization server:认证服务器,即服务提供商专门用来处理认证的服务器。

    (6)Resource server:资源服务器,即服务提供商存放用户生成的资源的服务器。它与认证服务器,可以是同一台服务器,也可以是不同的服务器。

    OAuth的思路:

    OAuth在"客户端"与"服务提供商"之间,设置了一个授权层(authorization layer)。"客户端"不能直接登录"服务提供商",只能登录授权层,以此将用户与客户端区分开来。"客户端"登录授权层所用的令牌(token),与用户的密码不同。用户可以在登录的时候,指定授权层令牌的权限范围和有效期。

    "客户端"登录授权层以后,"服务提供商"根据令牌的权限范围和有效期,向"客户端"开放用户储存的资料。

    运行流程:

    (A)用户打开客户端以后,客户端要求用户给予授权。

    (B)用户同意给予客户端授权。

    (C)客户端使用上一步获得的授权,向认证服务器申请令牌。

    (D)认证服务器对客户端进行认证以后,确认无误,同意发放令牌。

    (E)客户端使用令牌,向资源服务器申请获取资源。

    (F)资源服务器确认令牌无误,同意向客户端开放资源。

    快递员问题:

    我住在一个大型的居民小区。小区有门禁系统。进入的时候需要输入密码。我经常网购和外卖,每天都有快递员来送货。我必须找到一个办法,让快递员通过门禁系统,进入小区。

    我设计了一套授权机制:

    第一步,门禁系统的密码输入器下面,增加一个按钮,叫做"获取授权"。快递员需要首先按这个按钮,去申请授权。

    第二步,他按下按钮以后,屋主(也就是我)的手机就会跳出对话框:有人正在要求授权。系统还会显示该快递员的姓名、工号和所属的快递公司。

    我确认请求属实,就点击按钮,告诉门禁系统,我同意给予他进入小区的授权。

    第三步,门禁系统得到我的确认以后,向快递员显示一个进入小区的令牌(access token)。令牌就是类似密码的一串数字,只在短期内(比如七天)有效。

    第四步,快递员向门禁系统输入令牌,进入小区。

    OAuth 2.0 的标准是 RFC 6749 文件。OAuth 引入了一个授权层,用来分离两种不同的角色:客户端和资源所有者。......资源所有者同意以后,资源服务器可以向客户端颁发令牌。客户端通过令牌,去请求数据。本标准定义了获得令牌的四种授权方式(authorization grant )。

    令牌

    示例

    一、授权码

    授权码(authorization code)方式,指的是第三方应用先申请一个授权码,然后再用该码获取令牌。

    这种方式是最常用的流程,安全性也最高,它适用于那些有后端的 Web 应用。授权码通过前端传送,令牌则是储存在后端,而且所有与资源服务器的通信都在后端完成。这样的前后端分离,可以避免令牌泄漏。

    第一步,A 网站提供一个链接,用户点击后就会跳转到 B 网站,授权用户数据给 A 网站使用。下面就是 A 网站跳转 B 网站的一个示意链接。

    https://b.com/oauth/authorize?
      response_type=code&
      client_id=CLIENT_ID&
      redirect_uri=CALLBACK_URL&
      scope=read

    response_type参数表示要求返回授权码(code),client_id参数让 B 知道是谁在请求,redirect_uri参数是 B 接受或拒绝请求后的跳转网址,scope参数表示要求的授权范围(这里是只读)。

     第二步,用户跳转后,B 网站会要求用户登录,然后询问是否同意给予 A 网站授权。用户表示同意,这时 B 网站就会跳回redirect_uri参数指定的网址。跳转时,会传回一个授权码,就像下面这样。code参数就是授权码。

     https://a.com/callback?code=AUTHORIZATION_CODE

     第三步,A 网站拿到授权码以后,就可以在后端,向 B 网站请求令牌。

    https://b.com/oauth/token?
     client_id=CLIENT_ID&
     client_secret=CLIENT_SECRET&
     grant_type=authorization_code&
     code=AUTHORIZATION_CODE&
     redirect_uri=CALLBACK_URL

    client_id参数和client_secret参数用来让 B 确认 A 的身份(client_secret参数是保密的,因此只能在后端发请求),grant_type参数的值是AUTHORIZATION_CODE,表示采用的授权方式是授权码,code参数是上一步拿到的授权码,redirect_uri参数是令牌颁发后的回调网址。

    第四步,B 网站收到请求以后,就会颁发令牌。具体做法是向redirect_uri指定的网址,发送一段 JSON 数据。

        {    
          "access_token":"ACCESS_TOKEN",
          "token_type":"bearer",
          "expires_in":2592000,
          "refresh_token":"REFRESH_TOKEN",
          "scope":"read",
          "uid":100101,
          "info":{...}
        }

    access_token字段就是令牌,A 网站在后端拿到了

     

     二、隐藏式

    有些 Web 应用是纯前端应用,没有后端。这时就不能用上面的方式了,必须将令牌储存在前端。RFC 6749 就规定了第二种方式,允许直接向前端颁发令牌。这种方式没有授权码这个中间步骤,所以称为(授权码)"隐藏式"(implicit)。

    第一步,A 网站提供一个链接,要求用户跳转到 B 网站,授权用户数据给 A 网站使用。

    https://b.com/oauth/authorize?
      response_type=token&
      client_id=CLIENT_ID&
      redirect_uri=CALLBACK_URL&
      scope=read

    response_type参数为token,表示要求直接返回令牌。

    第二步,用户跳转到 B 网站,登录后同意给予 A 网站授权。这时,B 网站就会跳回redirect_uri参数指定的跳转网址,并且把令牌作为 URL 参数,传给 A 网站。

    https://a.com/callback#token=ACCESS_TOKEN

    token参数就是令牌,A 网站因此直接在前端拿到令牌。

    注意:令牌的位置是 URL 锚点(fragment),而不是查询字符串(querystring),这是因为 OAuth 2.0 允许跳转网址是 HTTP 协议,因此存在"中间人攻击"的风险,而浏览器跳转时,锚点不会发到服务器,就减少了泄漏令牌的风险。这种方式把令牌直接传给前端,是很不安全的。因此,只能用于一些安全要求不高的场景,并且令牌的有效期必须非常短,通常就是会话期间(session)有效,浏览器关掉,令牌就失效了。

    三、密码式

    如果你高度信任某个应用,RFC 6749 也允许用户把用户名和密码,直接告诉该应用。该应用就使用你的密码,申请令牌,这种方式称为"密码式"(password)。

    第一步,A 网站要求用户提供 B 网站的用户名和密码。拿到以后,A 就直接向 B 请求令牌。

    https://oauth.b.com/token?
      grant_type=password&
      username=USERNAME&
      password=PASSWORD&
      client_id=CLIENT_ID

    grant_type参数是授权方式,这里的password表示"密码式",usernamepassword是 B 的用户名和密码。

    第二步,B 网站验证身份通过后,直接给出令牌。

    注意:这时不需要跳转,而是把令牌放在 JSON 数据里面,作为 HTTP 回应,A 因此拿到令牌。

    四、凭证式

    适用于没有前端的命令行应用,即在命令行下请求令牌。

    第一步,A 应用在命令行向 B 发出请求。

    https://oauth.b.com/token?
      grant_type=client_credentials&
      client_id=CLIENT_ID&
      client_secret=CLIENT_SECRET

    grant_type参数等于client_credentials表示采用凭证式,client_idclient_secret用来让 B 确认 A 的身份。

    第二步,B 网站验证通过以后,直接返回令牌。

    注意:这种方式给出的令牌,是针对第三方应用的,而不是针对用户的,即有可能多个用户共享同一个令牌。

    令牌

    令牌的使用

    A 网站拿到令牌以后,就可以向 B 网站的 API 请求数据了。此时,每个发到 API 的请求,都必须带有令牌。具体做法是在请求的头信息,加上一个Authorization字段,令牌就放在这个字段里面。

    curl -H "Authorization: Bearer ACCESS_TOKEN"  "https://api.b.com"

    ACCESS_TOKEN就是拿到的令牌。

    更新令牌

    令牌的有效期到了,如果让用户重新走一遍上面的流程,再申请一个新的令牌,很可能体验不好,而且也没有必要。OAuth 2.0 允许用户自动更新令牌。

    具体方法是,B 网站颁发令牌的时候,一次性颁发两个令牌,一个用于获取数据,另一个用于获取新的令牌(refresh token 字段)。令牌到期前,用户使用 refresh token 发一个请求,去更新令牌。

    https://b.com/oauth/token?
      grant_type=refresh_token&
      client_id=CLIENT_ID&
      client_secret=CLIENT_SECRET&
      refresh_token=REFRESH_TOKEN

    grant_type参数为refresh_token表示要求更新令牌,client_id参数和client_secret参数用于确认身份,refresh_token参数就是用于更新令牌的令牌。

    B 网站验证通过以后,就会颁发新的令牌。

    示例:

    一、第三方登录的原理

    所谓第三方登录,实质就是 OAuth 授权。用户想要登录 A 网站,A 网站让用户提供第三方网站的数据,证明自己的身份。获取第三方网站的身份数据,就需要 OAuth 授权。

    举例来说,A 网站允许 GitHub 登录,背后就是下面的流程。

    1         A 网站让用户跳转到 GitHub。
    2         GitHub 要求用户登录,然后询问"A 网站要求获得 xx 权限,你是否同意?"
    3         用户同意,GitHub 就会重定向回 A 网站,同时发回一个授权码。
    4         A 网站使用授权码,向 GitHub 请求令牌。
    5         GitHub 返回令牌.
    6         A 网站使用令牌,向 GitHub 请求用户数据。

    二、应用登记

    一个应用要求 OAuth 授权,必须先到对方网站登记,让对方知道是谁在请求。

    所以,你要先去 GitHub 登记一下。当然,我已经登记过了,你使用我的登记信息也可以,但为了完整走一遍流程,还是建议大家自己登记。这是免费的。

    访问这个网址,填写登记表。

    应用的名称随便填,主页 URL 填写http://localhost:8080,跳转网址填写 http://localhost:8080/oauth/redirect

    提交表单以后,GitHub 应该会返回客户端 ID(client ID)和客户端密钥(client secret),这就是应用的身份识别码。

    三、示例仓库

    我写了一个代码仓库,请将它克隆到本地。

    git clone git@github.com:ruanyf/node-oauth-demo.git
    cd node-oauth-demo

    两个配置项要改一下,写入上一步的身份识别码。

      index.js:改掉变量clientID and clientSecret
      public/index.html:改掉变量client_id

    安装依赖。

    npm install

    启动服务。

    node index.js

    浏览器访问http://localhost:8080,就可以看到这个示例了。

    四、浏览器跳转 GitHub

     跳转的 URL 如下。

    https://github.com/login/oauth/authorize?
      client_id=7e015d8ce32370079895&
      redirect_uri=http://localhost:8080/oauth/redirect

    这个 URL 指向 GitHub 的 OAuth 授权网址,带有两个参数:client_id告诉 GitHub 谁在请求,redirect_uri是稍后跳转回来的网址。

    用户点击到了 GitHub,GitHub 会要求用户登录,确保是本人在操作。

    五、授权码

    登录后,GitHub 询问用户,该应用正在请求数据,你是否同意授权。

     用户同意授权, GitHub 就会跳转到redirect_uri指定的跳转网址,并且带上授权码,跳转回来的 URL 就是下面的样子。

    http://localhost:8080/oauth/redirect?code=859310e7cecc9196f4af

    后端收到这个请求以后,就拿到了授权码(code参数)。

    六、后端实现

    示例的后端采用 Koa 框架编写,具体语法请看教程

    这里的关键是针对/oauth/redirect的请求,编写一个路由,完成 OAuth 认证。

    const oauth = async ctx => {
      // ...
    };
    
    app.use(route.get('/oauth/redirect', oauth));

    上面代码中,oauth函数就是路由的处理函数。下面的代码都写在这个函数里面。路由函数的第一件事,是从 URL 取出授权码。

    const requestToken = ctx.request.query.code;

    七、令牌

    后端使用这个授权码,向 GitHub 请求令牌。

    const tokenResponse = await axios({
      method: 'post',
      url: 'https://github.com/login/oauth/access_token?' +
        `client_id=${clientID}&` +
        `client_secret=${clientSecret}&` +
        `code=${requestToken}`,
      headers: {
        accept: 'application/json'
      }
    });

    上面代码中,GitHub 的令牌接口https://github.com/login/oauth/access_token需要提供三个参数。

       client_id:客户端的 ID
       client_secret:客户端的密钥
       code:授权码

    作为回应,GitHub 会返回一段 JSON 数据,里面包含了令牌accessToken

    const accessToken = tokenResponse.data.access_token;

    八、API 数据

    有了令牌以后,就可以向 API 请求数据了。

    const result = await axios({
      method: 'get',
      url: `https://api.github.com/user`,
      headers: {
        accept: 'application/json',
        Authorization: `token ${accessToken}`
      }
    });

    上面代码中,GitHub API 的地址是https://api.github.com/user,请求的时候必须在 HTTP 头信息里面带上令牌Authorization: token 361507da

    然后,就可以拿到用户数据,得到用户的身份。

    const name = result.data.name;
    ctx.response.redirect(`/welcome.html?name=${name}`);

     豆瓣示例:

    作者:zhangshuai
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须在文章页面给出原文连接,否则保留追究法律责任的权利。
  • 相关阅读:
    [C#]MagicLibrary.dll控件的使用(一)(下拉菜单)
    [SQL server]查询用户表及表结构
    [ASP.net]ASP.net的RUL重写
    [C#]简单XP菜单的实现(一)
    [Javascript]IFRAME运用(1)
    [ASP.net]未解的疑惑!
    [乱七八糟]Google搜索使用详细
    [Javascript]IFRAME运用(2)
    [随文杂记]残局
    [CSS]RevealTrans 滤镜
  • 原文地址:https://www.cnblogs.com/zhangshaui/p/15368675.html
Copyright © 2011-2022 走看看