Laravel 如何实现自定义资源路由

Laravel 如何实现自定义资源路由

最近在开发过程中,发现总有一些路由需要重复定义,比如切换状态,导出,回收站啊之类的。如果使用 Laravel 自带的资源路由方法,还不足以满足重复劳动得过程。所以是否有方法可以自定义项目得资源路由呢?在 Laravel 中,资源路由一般有两种

服务端渲染

Route::resource('xxxxx')

resource 路由包含以下几个方法

  • index
  • store
  • create
  • show
  • edit
  • upate
  • destroy

Api 资源路由

Route::apiResource('xxxxx')

ApiResource 则是用 Resource 方法生成,仅仅保存了下面几个方法

$only  = ['index', 'show', 'store', 'update', 'destroy'];

现在项目一般都采用前后端分离,所以一般都是使用 apiResource 这个方法,这个方法如上所示只提供五个方法,即五个路由。所以如果项目中需要一些特定得每次都加得路由,必须还要自己再添加一条,很不方便。那在 Laravel 中是不是可以自己扩展属于项目独有得 resource 方法呢?答案是当然可以,而且非常简单

如何实现

一般都是在项目中这样定义资源路由的, 通过门面 Route 访问 apiResource 方法进行定义,如下

Route::apiResource('hello', HelloController::class);

实际上通过门面调用的实际是 Illuminate\Routing\RouterapiResource 方法,内容如下

public  function  apiResource($name, $controller, array  $options  = [])
{
 $only  = ['index', 'show', 'store', 'update', 'destroy'];
 if (isset($options['except'])) {
 $only  =  array_diff($only, (array) $options['except']);
    }
 // 实际调用的是 resource 方法
 return  $this->resource($name, $controller, array_merge([
 'only'  =>  $only,
], $options));
}

实际上 apiResource 是用 resource 方法包装的,所以再来看看 resource 方法做了啥

public  function  resource($name, $controller, array  $options  = [])
{
 // 这里是关键的地方
 if ($this->container  &&  $this->container->bound(ResourceRegistrar::class)) {
 $registrar  =  $this->container->make(ResourceRegistrar::class);
} else {
 $registrar  =  new  ResourceRegistrar($this);
    }
 // 这一步不用关注
 return  new  PendingResourceRegistration(
 $registrar, $name, $controller, $options
    );
}

最关键的地方就是 ResourceRegistrar,这里就叫资源路由注册器吧。

  • 首先查找容器中是否有资源路由注册器的绑定实现,如果有,直接从容器中 make 出来

  • 如果容器中没有绑定,则直接使用 new 实例化

那就是说,如果自定义自己的资源路由注册器,然后再绑定 ResourceRegistrar 的实现,就可以实现让框架使用自定义的路由注册器了。

这解决了资源路由注册器的问题,还需要一个方法来调用,从上来得知

Route::apiResource('hello', HelloController::class);

实际上就是 Illuminate\Routing\RouterapiResource 方法。那么如何在 Illuminate\Routing\Router 添加自定义方法呢?没错,首先就要想到它支不支持

Macroable?🤣 显而易见,肯定是支持的,那么依葫芦画瓢,去注册一个吧。找到 AppServiceProvider,在里面注册即可

添加自定义的资源路由方法

因为是做后台系统项目,这里就将资源路由方法定义为 adminResource


Router::macro('adminResource', function ($name, $controller, array  $options  = []) {
 // 这里添加 enable 和 export
 $only  = ['index', 'show', 'store', 'update', 'destroy', 'enable', 'export'];
 if (isset($options['except'])) {
 $only  =  array_diff($only, (array) $options['except']);
    }

 return  $this->resource($name, $controller, array_merge([
 'only'  =>  $only,
], $options));
});

实现一个资源路由注册器

当然并不是自己去实现,而是集成 Illuminate\Routing\ResourceRegistrar 即可。在它的基础上,再添加 enableexport 两个方法实现就可以,如下


namespace  Defined;
use Illuminate\Routing\ResourceRegistrar  as LaravelResourceRegistrar;
use Illuminate\Routing\Route;
class  ResourceRegistrar  extends  LaravelResourceRegistrar
{
 protected  $resourceDefaults  = ['index', 'create', 'store', 'show', 'edit', 'update', 'destroy', 'enable', 'export'];

 // 添加 enable 方法路由

 protected  function  addResourceEnable($name, $base, $controller, $options):  Route

    {

 $name  =  $this->getShallowName($name, $options);

 $uri  =  $this->getResourceUri($name).'/enable/{'.$base.'}';

 $action  =  $this->getResourceAction($name, $controller, 'enable', $options);

 return  $this->router->put($uri, $action);

    }

 // 添加 export 方法路由

 protected  function  addResourceExport($name, $base, $controller, $options):  Route
    {

 $uri  =  $this->getResourceUri($name).'/export';

 unset($options['missing']);

 $action  =  $this->getResourceAction($name, $controller, 'export', $options);

 return  $this->router->get($uri, $action);

    }
}

实现好了之后,还需要绑定到容器里, 还是找到 AppServiceProvider,在里面注册即可


// 资源路由注册器

$this->app->bind(ResourceRegistrar::class, \Defined\ResourceRegistrar::class);

完成之后测试一下看看,我本地使用 UserController 测试

Route::adminResource('users', UserController::class);

完成之后呢,使用 php artisan route:list | grep users 查看

GET|HEAD  api/users  ...................  users.index  ›  UserController@index

POST  api/users  .......................  users.store  ›  UserController@store

# put 请求 符合预期

PUT  api/users/enable/{user}  ...........users.enable  ›  UserController@enable

# Get 请求 符合预期

GET|HEAD  api/users/export  .............  users.export  ›  UserController@export

GET|HEAD  api/users/{user}  .............  users.show  ›  UserController@show

PUT|PATCH  api/users/{user}  ............users.update  ›  UserController@update

DELETE  api/users/{user}  ...............  users.destroy  ›  UserController@destroy

原文链接

本作品采用《CC 协议》,转载必须注明作者和本文链接
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
《G01 Go 实战入门》
从零开始带你一步步开发一个 Go 博客项目,让你在最短的时间内学会使用 Go 进行编码。项目结构很大程度上参考了 Laravel。
讨论数量: 3

我也考虑过这个问题,尤其是后台,自带的资源控制器根本不够看,控制器至少有以下通用需求 1、列表查询 2、新增 3、修改 4、查询 5、删除 6、批量删除

含软删除业务 7、已删除列表 8、恢复数据 9、批量恢复 10、物理删除 11、批量物理删除

批量修改需求也不少见, 默认的这个确实不够看,感谢楼主提供的方案,不知道还有没有老哥提供更简单优雅的实现

1周前 评论

我最近自己写了个 CMS 系统,后台接口路由也是用 apiResource 提供的,常规的 CRUD 外的操作手动写的,比如批量删除。

我试了作者提供的思路,还因为最近了解到 Laravel 的 Macroable,可以动态加方法,确实可行的。

这个思路 pro 是扩展 Laravel 内置的东西,不用新东西,但硬要找 con 的话:

  • Macroable 本身太 Hack,我猜测这是为什么 Laravel 文档不会介绍的原因
  • 依赖 Illuminate\Routing\ResourceRegistrar 的内部实现,假设 Laravel 以后改了 ResourceRegistrar 的实现,这个方法就不行了
  • 就算删除 adminResource 的调用,原始的 ResourceRegistrar 的也会被自己的替代
1周前 评论
JaguarJack (楼主) 1周前

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