Laravel+Dingo+JWT快速搭建 API系统

1.安装Dingo
你必须在你的项目中修改 composer.json 文件并且运行 composer update 命令来加载这个包(别问我 composer require dingo/api 为啥不能安装,我也不知道,尽量用composer require 命令安装,如果安装不了就用 composer update 命令)

"require": {
    "dingo/api": "2.3"
}

使用以下命令可以发布 API 的配置文件到 config 文件夹下

php artisan vendor:publish --provider="Dingo\Api\Provider\LaravelServiceProvider"

.env文件下加入

API_STANDARDS_TREE=vnd //环境
API_SUBTYPE=sdk  //子类型
API_PREFIX=api //前缀
API_VERSION=v1//版本
API_NAME="SDK"//名字
API_CONDITIONAL_REQUEST=false//带条件的请求
API_STRICT=false//严格模式
API_DEFAULT_FORMAT=json//响应格式
API_DEBUG=true//调试模式

注意:.env配置文件是不能有空格和注释的!

创建端点(API控制器的路由写在这里)通常在routes/api.php里

$api = app('Dingo\Api\Routing\Router');
$api->version('v1', function ($api) {

});

2.安装JWT

composer require tymon/jwt-auth 1.*@rc

进行一些配置

这里指的注意的是,有些文档会说要添加 Tymon\JWTAuth\Providers\LaravelServiceProvider::class ,这只在 Laravel 5.4 及以下版本是必要的,更新的 Laravel 版本无需添加。

还有一些文档说要添加 Tymon\JWTAuth\Providers\JWTAuthServiceProvider 这是很久以前的 JWT 版本的(大概 0.5.3 以前的版本)

发布配置文件

# 这条命令会在 config 下增加一个 jwt.php 的配置文件
php artisan vendor:publish --provider="Tymon\JWTAuth\Providers\LaravelServiceProvider"

生成加密密钥

# 这条命令会在 .env 文件下生成一个加密密钥,如:JWT_SECRET=foobar
php artisan jwt:secret

更新你的模型

<?php

namespace App;

use Tymon\JWTAuth\Contracts\JWTSubject;
use Illuminate\Notifications\Notifiable;
use Illuminate\Foundation\Auth\User as Authenticatable;

class User extends Authenticatable implements JWTSubject    # 这里别忘了加
{
    use Notifiable;
    
	protected $table = "";//表名
    protected $primaryKey = "";//主键
    public $timestamps = false;
    protected $fillable = [];//白名单

    protected $guarded = [];//黑名单
    // Rest omitted for brevity

    /**
     * Get the identifier that will be stored in the subject claim of the JWT.
     *
     * @return mixed
     */
    public function getJWTIdentifier()
    {
        return $this->getKey();
    }

    /**
     * Return a key value array, containing any custom claims to be added to the JWT.
     *
     * @return array
     */
    public function getJWTCustomClaims()
    {
        return [];
    }
}

注册两个 Facade
config/app.php文件里

'aliases' => [
        ...
        // 添加以下两行
        'JWTAuth' => 'Tymon\JWTAuth\Facades\JWTAuth',
        'JWTFactory' => 'Tymon\JWTAuth\Facades\JWTFactory',
],

修改 auth.php
config/auth.php

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

    'api' => [
        'driver' => 'jwt',      // 原来是 token 改成jwt
        'provider' => 'users',
    ],
],

 'providers' => [
        'users' => [
            'driver' => 'eloquent',
            'model' => App\Api\Models\PlayerApplication::class, //改成自己刚定义的模型
        ],

        // 'users' => [
        //     'driver' => 'database',
        //     'table' => 'users',
        // ],
    ],

修改api.php
config/api.php

  'auth' => [
        'jwt' => Dingo\Api\Auth\Provider\JWT::class
    ],

获取token控制器

<?php
/**
 * 用户注册即登录 并分配TOKEN 和用户登录并分配token
 */
namespace App\Api\Controllers\V1;

use App\Http\Controllers\Controller;

use App\Models\Pusers;
use Dingo\Api\Routing\Helpers;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Validator;
use Tymon\JWTAuth\Facades\JWTAuth;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;

use App\Service\GeneralMethod;

class RegisterController extends Controller
{
    //dingoAPI响应生成器
    use Helpers;
    //底层小方法引用
    use GeneralMethod;

    public function __construct() {
        $this->middleware('auth:api', ['except' => ['init','login']]);
       // $this->middleware('auth:api');
    }

    /**
     * @param Request $request
     * @return array|\Illuminate\Http\JsonResponse
     * //注册并登录
     */
    public function init(Request $request)
    {
        $User_data = $request->all();
        //校验用户名密码格式
        $validator = $this->validator($User_data);
        if($validator->fails()){
            return $this->returnErrorMessage(['status'=>'error','message'=>301]);

        }

        $init = [
            'nickname'=>$this->nickNameRand(8),
            'mobile'=>$User_data['mobile'],
            //密码是bcrypt方法加密的下面这个用的是laravel自带的bcrypt加密方法
            'password'=>HASH::make($User_data['password']),
        ];

        //插入数据库
         Pusers::Create($init);
         //验证账号密码,并生成token的方法在GeneralMethod性状中
         return $this->createToken($User_data);

    }

    /**
     * @param array $data
     * @return \Illuminate\Contracts\Validation\Validator
     * 登录
     */
    public function login(Request $request)
    {
        $data = $request->all();
        //校验用户名密码格式
        $validator = $this->loginValidator($data);
        if($validator->fails()){
            return $this->returnErrorMessage(['status'=>'error','message'=>301]);
        }
        return  $this->createToken($request->all());

    }

    //注册验证参数
    public function validator(array $data)
    {
        return Validator::make($data,[
            'mobile'=>'required|regex:/^1[345789][0-9]{9}$/|unique:p_users,mobile',//手机号格式验证,手机号唯一
            'password'=>'required|between:6,12|alpha_num'//密码6位数到12位数,由字母和数字构成
        ]);
    }

    //登录验证参数
    public function loginValidator(array $data)
    {
        return Validator::make($data,[
            'mobile'=>'required|regex:/^1[345789][0-9]{9}$/',//手机号格式验证,手机号唯一
            'password'=>'required|between:6,12|alpha_num'//密码6位数到12位数,由字母和数字构成
        ]);
    }


}

service目录里的GeneralMethod.php里的检验账号密码并生成token的方法

<?php
namespace App\Service;
use App\Models\Pusers;

/**
 * Class Service
 * @package App\Service
 * 通用底层小方法
 */
Trait GeneralMethod
{
    /**
     * @param array $data
     * API返回错误码对应信息
     */
     public function returnErrorMessage(array $data)
     {
         $data['message'] = config('apierror.'.$data['message']);
         return $data;
     }

    /**
     * 随机用户名生成
     */
     public function nickNameRand($length)
     {
         // 密码字符集,可任意添加你需要的字符
         $chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
         $str = '';
         for($i = 0; $i < $length; $i++)
         {
             // 这里提供两种字符获取方式
             // 第一种是使用 substr 截取$chars中的任意一位字符;
             // 第二种是取字符数组 $chars 的任意元素
          //   $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
            $str .= $chars[mt_rand(0, strlen($chars) - 1)];
         }
         return $str;
     }

    /**
     * @param array $data
     * @return array|\Illuminate\Http\JsonResponse
     * 检验用户名密码生成token
     */
    protected function createToken(array $data)
    {
        /**
         * 校验用户名密码并生成token(会将password字段进行bcrypt加密并检测 密码字段名必须是password,attempt方法必须传的是数组)
         */
        if (! $token = auth('api')->attempt($data)) {
            return $this->returnErrorMessage(['status'=>'error','message'=>401]);
        }
        $user = Pusers::where('mobile',$data['mobile'])->first();

        //注册并登录成功返回信息
        return $this->respondWithToken($user->nickname,$token);
    }

  /**
     * @param object $data
     * @return array|\Illuminate\Http\JsonResponse
     * 利用用户信息生成token(有时候你需要进行手机短验登录,第三方授权登录,这时候在控制器完成了授权登录的逻辑,然后把用户信息传过来生成token)
     */
    protected function createToken($data)
    {
        /**
         * 生成token(attempt方法换成login方法就好 login方法传入的是用户信息的对象,例如:User::first()查出来的对象)
         */
        if (! $token = auth('api')->login($data)) {
            return $this->returnErrorMessage(['status'=>'error','message'=>401]);
        }
      
        //注册并登录成功返回信息
        return $this->respondWithToken($data->nickname,$token);
    }



    //返回方法
    protected function respondWithToken($data,$token)
    {
        return response()->json([
            'status'=>200,
            'message'=>$data,
            'access_token' => $token,
            'token_type' => 'bearer',
            'expires_in' => auth('api')->factory()->getTTL() * 60
        ]);
    }

}

JWT token 无痛刷新实现(定义一个中间件)

执行下面的命令生成中间件

php artisan make:middleware RefreshJwtToken
<?php

  namespace App\Http\Middleware;
  
  use Closure;
  use Tymon\JWTAuth\Facades\JWTAuth;
  use Tymon\JWTAuth\Exceptions\JWTException;
  use Tymon\JWTAuth\Exceptions\TokenExpiredException;
  use Symfony\Component\HttpKernel\Exception\UnauthorizedHttpException;
  use Tymon\JWTAuth\Http\Middleware\BaseMiddleware;
  
  class RefreshJwtToken extends BaseMiddleware
  {
  
      /**
       * Handle an incoming request.
       *
       * @param  \Illuminate\Http\Request  $request
       * @param  \Closure  $next
       * @return mixed
       */
      public function handle($request, Closure $next)
      {
          //检查请求中是否带有token 如果没有token值则抛出异常
          $this->checkForToken($request);   
          try{
              //从请求中获取token并且验证token是否过期  若是不过期则请求到控制器处理业务逻辑  若是过期则进行刷新
              if ($request->user = JWTAuth::parseToken()->authenticate()) {       
                  return $next($request);
              }
              throw new UnauthorizedHttpException('jwt-auth', '未登录');
          }catch (TokenExpiredException $exception){
          // 此处捕获到了 token 过期所抛出的 TokenExpiredException 异常,我们在这里需要做的是刷新该用户的 token 并将它添加到响应头中
              try{
                  // 刷新用户的 token
                $token = $this->auth->refresh();
               // 使用一次性登录以保证此次请求的成功
                Auth::guard('api')->onceUsingId($this->auth->manager()->getPayloadFactory()->buildClaimsCollection()->toPlainArray()['sub']);
              }catch (JWTException $exception){
              // 如果捕获到此异常,即代表 refresh 也过期了,用户无法刷新令牌,需要重新登录。
                  throw new UnauthorizedHttpException('jwt-auth', $exception->getMessage());
              }
          }
          //将token值返回到请求头
          return $this->setAuthenticationHeader($next($request), $token);
      }
  }

在Kernel中注册中间件

/**
   * The application's route middleware.
   *
   * These middleware may be assigned to groups or used individually.
   *
   * @var array
   */
  protected $routeMiddleware = [
      ......
      'token.refresh' => \App\Http\Middleware\RefreshJwtToken::class,   
  ];

在路由中使用这个中间件就可以实现无痛刷新 (当然还需要前端配合跟换 token 值,后端是已经完成啦)

JWTAuth.php中方法
attempt()  尝试对用户进行身份验证并返回令牌
authenticate()  通过令牌对用户进行身份验证
user()      获取经过身份验证的用户
参考链接
DingoAPI文档
jwt完整详解
生成token的时候返回false解决
Laravel+JWT token 无痛刷新实现
Laravel+JWT token 无痛刷新实现
jwt-auth 官方 wiki
laravel 框架bcrypt加密方式的密码校验

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Lumen是一个基于Laravel框架的微型PHP框架,它可以用于构建轻量级的API服务。而Dingo是一个在Laravel框架上构建API的工具包。JWT(JSON Web Token)是一种用于进行身份验证和授权的开放标准。 在使用Lumen结合DingoJWT进行开发时,需要先安装Lumen服务提供者、JWTDingo的组件。可以使用Composer来管理这些依赖。确保你的电脑上安装了Composer。 在Lumen中,你可以使用控制器来处理请求。引用是一个示例UserController。在这个控制器中,我们注入了JWTAuth实例,并使用它来处理用户的登录请求。其中,我们首先获取请求中的参数,然后使用这些参数进行条件查询。如果登录认证成功,我们会返回一个包含JWT令牌的JSON响应。 对于跨域问题,你可以使用palanik/lumen-cors来解决。引用提供了安装和配置palanik/lumen-cors的方法。你需要通过Composer来安装该组件,并在bootstrap/app.php文件中添加cors路由中间件。 以上就是关于Lumen、DingoJWT的一些基本信息和配置方法。如果你有关于它们的更具体的问题,请告诉我。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [Lumen 配合 JWT + Dingo 开发流程](https://blog.csdn.net/qq_44149053/article/details/89444892)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *3* [lumen+dingo+jwt搭建api系统](https://blog.csdn.net/Chenlevin/article/details/111830096)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值