• OAuth 第三方登录授权码(authorization code)方式的小例子


     假如上面的网站A,可以通过GitHub账号登录;

    下面以OAuth其中一种方式,授权码(authorization code)方式为例。

    一、第三方登录的原理

    所谓第三方登录,实质就是 OAuth 授权

    数据的所有者告诉系统,同意授权第三方应用进入系统,获取数据。系统从而产生一个短期的进入令牌(token),用来代替密码,供第三方应用使用。

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

    这里利用OAuth authorization code的形式。

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

    1. A 网站前端让用户带着client_id, response_type(如code授权码模式), redirect_uri, scope跳转到 GitHub
    2. GitHub 要求用户登录,然后询问用户"A 网站要求获得 xx 权限,你是否同意?", 同意后会返回授权码code同时重定向URL
    3. 用户同意,GitHub 就会重定向回 A 网站的后端API URL同时发回一个授权码,如http://localhost:8080/oauth/redirect?code=859310e7cecc9196f4af
    4. A 网站后端带着授权码code参数,向 GitHub发送请求,想获得令牌
    5. GitHub 返回令牌,这里是一个字符串.
    6. A 网站使用令牌,请求GitHub API如https://api.github.com/user,请求的时候必须在 HTTP 头信息里面带上令牌Authorization: token 361507da,就可以获得用户数据

    下面就是这个流程的代码实现。

    二、应用登记

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

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

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

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

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

    三、示例仓库

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

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

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

    然后,安装依赖。

    
    $ npm install
    

    启动服务。

    
    $ node index.js
    

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

    四、浏览器跳转 GitHub

    示例的首页很简单,就是一个链接,让用户跳转到 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}`);
    

    (完)

    来源: http://www.ruanyifeng.com/blog/2019/04/github-oauth.html

    OAuth 2.0 的四种方式: http://www.ruanyifeng.com/blog/2019/04/oauth-grant-types.html

    OAuth 2.0 的一个简单解释: http://www.ruanyifeng.com/blog/2019/04/oauth_design.html

    理解OAuth 2.0: http://www.ruanyifeng.com/blog/2014/05/oauth_2_0.html

  • 相关阅读:
    you-get 下载网络上的富媒体信息
    响应式布局
    网页头部的声明应该是用 lang="";
    htm、html、shtml网页区别
    请求头出现Provisional headers are shown
    配置nginx直接使用webpack生成的gz压缩文件,而不用nginx自己压缩
    babel-plugin-equire
    VUE中$refs的基本用法
    element-ui--按需引入
    vue ts ,vue使用typescript,三种组件传值方式
  • 原文地址:https://www.cnblogs.com/theRhyme/p/12051915.html
Copyright © 2020-2023  润新知