Laravel5.6 + Passport 实现 API 接口认证

很多企业做项目使用前后端分离,后端提供接口地址,前端使用接口地址拿数据,并渲染页面。那么,前端用户登录如何使用接口进行认证?网上各种教程写的不堪入目,完全看不懂,所以我根据自己的理解,写下此篇文章,希望能帮助到大家。

后端(Laravel5.6框架)

1、使用composer安装Passport,打开终端,执行命令:

composer require laravel/passport   #安装完成后,在composer.json文件中会看到文件版本信息

2、接下来,将Passport的服务提供者注册到配置文件config/app.phpproviders数组中

Laravel\Passport\PassportServiceProvider::class,

3、执行数据库迁移

php artisan migrate  #数据库中会生成接口认证所需的5张表

4、创建密码授权客户端

php artisan passport:client --password
#创建了client_id和client_secret,前端登录验证的时候必须把这两个玩意儿带着

5、获取keys

php artisan passport:keys

6、配置路由
打开服务提供者AuthServiceProvider, 在boot方法中加入如下代码:

use Laravel\Passport\Passport;
public function boot() { 
    $this->registerPolicies();  
    Passport::routes(); //接口认证的路由
}

然后将配置文件config/auth.php中授权看守器guardsapidriver选项改为passport
我这里的customer表是前端用户表,但是laravel默认的是user表,所以这里需要做如下配置:

'guards' => [
    'web' => [
        'driver' => 'session',
        'provider' => 'users',
    ],

    'api' => [
        'driver' => 'passport',
        'provider' => 'customers',
    ],
],
'providers' => [
    'users' => [
        'driver' => 'eloquent',
        'model' => App\User::class,
    ],
    'customers' => [
        'driver' => 'eloquent',
        'model' => App\Models\Shop\Customer::class,
    ],
],

7、注册中间件,在app/Http/Kernel.php文件中的$routeMiddleware数组中添加如下中间件

protected $middleware = [
    \App\Http\Middleware\CheckForMaintenanceMode::class,
    \Illuminate\Foundation\Http\Middleware\ValidatePostSize::class,
    \App\Http\Middleware\TrimStrings::class,
    \Illuminate\Foundation\Http\Middleware\ConvertEmptyStringsToNull::class,
    \App\Http\Middleware\TrustProxies::class,
    \Barryvdh\Cors\HandleCors::class,
];

**********************************************************************

protected $routeMiddleware = [
'client.credentials'=>\Laravel\Passport\Http\Middleware\CheckClientCredentials::class,
];

然后在需要认证接口路由文件routes/api.php前面加上这个中间件。

Route::prefix('cart')->middleware('client.credentials')->group(function(){
    ...
});

8、前端用户表customer模型里面做如下配置:

use Illuminate\Foundation\Auth\User as Authenticatable;
use Laravel\Passport\HasApiTokens;

class Customer extends Authenticatable
{
    use HasApiTokens;
     ....   
}

至此,后端的所有配置已完成。

接下来,打开接口测试工具(postman),输入接口地址:wechat.test/oauth/token,请求类型 POST,填上如下参数,点击 send 你会看到后台返回了前端所需的access_token

前端(vue.js)

首先去加载用户登录组件,即用户登录页面。

  1. 配置路由,在index.js文件中写入如下代码
import Login from '@/components/customer/Login'
export default new Router({
  routes: [
        ....
    {
      path: '/customer/login',
      name: 'Login',
      component: Login
    },
  ]
})

2、加载组件,在customer文件夹的Login.vue文件中写入如下代码:

<template>
  <div>
    <input type="email" v-model="customer.email" placeholder="请输入邮箱">
    <input type="password" v-model="customer.password" placeholder="请输入密码">
    <button @click="submit">登 录</button>
  </div>
</template>

<script>
  export default {
    data() {
      return {
        customer: {
          email: '',
          password: ''
        }
      }
    },
    methods: {
      submit() {
        //将数据配置好
        const data = {
          grant_type: 'password', //oauth的模式
          client_id: 1,   //上面所说的client_id
          client_secret: 'CO331cA1mqiKgGvvgiDzPxh4CUu19vSEiqxM7LHD',//同上
          username: this.customer.email,
          password: this.customer.password,
        }
        this.axios.post('/oauth/token', data)
          .then(res => {
            if (res.status == 200) { //如果成功了,就把access_token存入localStorage
              localStorage.token_type = res.data.token_type
              localStorage.access_token = res.data.access_token
              this.$router.push({name:'Index'})
            }
          })
      }
    }
  }
</script>

客户端查看localStorage,如图:

3、在http.js文件中设置拦截器,用于判断用户是否登录,若没有登录跳转到登录页面。代码如下:

//#创建http.js文件
import axios from 'axios'
import router from '@/router'

// axios 配置
axios.defaults.timeout = 5000;
axios.defaults.baseURL = 'http://wechat.test/';

// http request 拦截器
axios.interceptors.request.use(
  config => { //将所有的axios的header里加上token_type和access_token
    config.headers.Authorization = `${localStorage.token_type} ${localStorage.access_token}`;
    return config;
  },
  err => {
    return Promise.reject(err);
  });

// http response 拦截器
axios.interceptors.response.use(
  response => {
    return response;
  },
  error => {
    // 401 清除token信息并跳转到登录页面
    if (error.response.status == 401) {
      alert('您还没有登录,请先登录')
      router.replace({    //如果失败,跳转到登录页面
        name: 'Login'
      })
    }
    return Promise.reject(error.response.data)
  });

export default axios;

重新访问项目,在商品详情页面点击加入购物车,你会发觉奇迹已经出现,当你没有登录时,提示跳转到登录页面。输入账号密码,登录成功,此时就能拿到用户id。接下来,继续测试。

4、去Cart控制器中,找到购物车首页方法,获取用户的id,获取方式如下:

$customer_id = auth('api')->user()->id;
return $customer_id;

5、在postman中输入购物车首页接口地址,并传入所需参数,参数参考地址:http://laravelacademy.org/post/8909.html,如图:

拿到用户id后,把后端之前定义的customer_id全部改为通过接口方法获取。至此,Passport接口认证的全部操作已完成。

Tips:如果你在项目中跑passport命令执行报错的话,可参考这篇文章来解决问题:https://laravel-china.org/topics/16245

总结:接口认证逻辑思想

1、安装passport后,生成client_idclient_secret
2、使用usernamepasswordclient_idclient_secretgrant_type参数,调用/oauth/token接口,拿到access_token
3、需要认证的接口,加上中间件。这时候直接访问接口地址,会提示没有认证的。带上access_token后,才能拿到接口的数据。

本作品采用《CC 协议》,转载必须注明作者和本文链接
本帖由系统于 4年前 自动加精
huangdj
《L01 基础入门》
我们将带你从零开发一个项目并部署到线上,本课程教授 Web 开发中专业、实用的技能,如 Git 工作流、Laravel Mix 前端工作流等。
《L05 电商实战》
从零开发一个电商项目,功能包括电商后台、商品 & SKU 管理、购物车、订单管理、支付宝支付、微信支付、订单退款流程、优惠券等
讨论数量: 6

感谢分享。

4年前 评论

passport不是你这样用的,client_secret不应该放到前端。可以先了解下OAuth2的工作流程。

4年前 评论

个人访问令牌根据网上教程会搭建,但是个人访问令牌是不是没有刷新 token?

密码访问令牌根据 Laravel 文档的话,不会用。

4年前 评论
小李世界 (作者) 4年前

/oauth/token 是登录接口实现时用 GuzzleHttp 调用的,不应该将 client_secretclient_id 放在客户端代码里

4年前 评论

file 你好,我按照你的这个步骤安装passport 到postman测试的时候发现username 和password 是该怎么填。随便填 { "error": "invalid_credentials", "message": "The user credentials were incorrect." } 这样的错误 我是先下载laravel 在安装passport

3年前 评论
GLLALLEN (作者) 3年前

file

一直验证不通过,不知道为什么?

3年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!