* * * * *
[TOC]
## 简介
除了内置提供的 [用户认证](用户认证.md) 服务外,Laravel 还提供一种更简单的方式来处理用户授权动作。类似用户认证,Laravel 有 2 种主要方式来实现用户授权:gates 和策略。
可以把 gates 和策略类比于路由和控制器。Gates 提供了一个简单、基于闭包的方式来授权认证。策略则和控制器类似,在特定的模型或者资源中通过分组来实现授权认证的逻辑。我们先来看看 gates,然后再看策略。
在你的应用中,不要将 gates 和策略当作相互排斥的方式。大部分应用很可能同时包含 gates 和策略,并且能很好的工作。Gates 大部分应用在模型和资源无关的地方,比如查看管理员的面板。与此相比,策略应该用在特定的模型或者资源中。
## Gates
### 编写 Gates
Gates 是用来决定用户是否授权访问给定的动作的闭包函数,并且典型的做法是在 `App\Providers\AuthServiceProvider` 类中使用 `Gate` facade 定义。Gates 接受一个用户实例作为第一个参数,接受可选参数,比如相关的 Eloquent 模型:
~~~
/**
* 注册任意用户认证 、用户授权服务。
*
* @return void
*/
public function boot()
{
$this->registerPolicies();
Gate::define('update-post', function ($user, $post) {
return $user->id == $post->user_id;
});
}
~~~
Gates 也可以使用 `Class@method` 形式作为回调字符串,比如控制器
~~~
/**
* Register any authentication / authorization services.
*
* @return void
*/
public function boot()
{
$this->registerPolicies();
Gate::define('update-post', 'PostPolicy@update');
}
~~~
#### Resource Gates
你还可以使用 `resource` 方法一次性定义多个 `Gate` 能力
~~~
Gate::resource('posts', 'PostPolicy');
~~~
这与手动定义以下Gate定义相同:
~~~
Gate::define('posts.view', 'PostPolicy@view');
Gate::define('posts.create', 'PostPolicy@create');
Gate::define('posts.update', 'PostPolicy@update');
Gate::define('posts.delete', 'PostPolicy@delete');
~~~
默认情况下,`view`,`create`,`update`,和`delete`能力是被定义过的。你也可以通过将数组作为第三个参数传递给`resource`方法来定义其他功能。数组的键定义了该能力的名称,而该值定义了方法名称:
~~~
Gate::resource('posts', 'PostPolicy', [
'posts.photo' => 'updatePhoto',
'posts.image' => 'updateImage',
]);
~~~
### 使用 gates 授权动作
使用 gates 来授权动作时,应当使用 `allows` 方法。注意你并不需要传递当前认证通过的用户给 `allows` 方法。Laravel 会自动处理好传入的用户,然后传递给 gate 闭包函数:
~~~
if (Gate::allows('update-post', $post)) {
// 当前用户可以更新 post...
}
if (Gate::denies('update-post', $post)) {
// 当前用户不能更新 post...
}
~~~
如果需要指定一个特定的用户可以访问某个动作,可以使用 `Gate` facade 中的 `forUser` 方法:
~~~
if (Gate::forUser($user)->allows('update-post', $post)) {
// 指定用户可以更新 post...
}
if (Gate::forUser($user)->denies('update-post', $post)) {
// 指定用户不能更新 post...
}
~~~
## 创建策略
### 生成策略
策略是在特定模型或者资源中组织授权逻辑的类。例如,如果应用是一个博客,会有一个 `Post` 模型和一个相应的 `PostPolicy` 来授权用户动作,比如创建或者更新博客。
可以使用 `make:policy` [artisan 命令](Artisan命令行.md) 来生成策略。生成的策略将放置在 `app/Policies` 目录。如果在你的应用中不存在这个目录,那么 Laravel 会自动创建:
~~~
php artisan make:policy PostPolicy
~~~
`make:policy` 会生成空的策略类。如果希望生成的类包含基本的「CRUD」策略方法, 可以在使用命令时指定 `--model` 选项:
~~~
php artisan make:policy PostPolicy --model=Post
~~~
> {tip} 所有授权策略会通过 Laravel [服务容器](控制器.md) 解析,意指你可以在授权策略的构造器对任何需要的依赖使用类型提示,它们将会被自动注入。
### 注册策略
一旦该授权策略存在,需要将它进行注册。`AuthServiceProvider` 包含了一个 `policies` 属性,可将各种模型对应至管理它们的授权策略。注册一个策略将引导 Laravel 在授权动作访问给定模型时使用何种策略:
~~~
<?php
namespace App\Providers;
use App\Post;
use App\Policies\PostPolicy;
use Illuminate\Support\Facades\Gate;
use Illuminate\Foundation\Support\Providers\AuthServiceProvider as ServiceProvider;
class AuthServiceProvider extends ServiceProvider
{
/**
* 应用的策略映射。
*
* @var array
*/
protected $policies = [
Post::class => PostPolicy::class,
];
/**
* 注册任意用户认证、用户授权服务。
*
* @return void
*/
public function boot()
{
$this->registerPolicies();
//
}
}
~~~
## 编写策略
### 策略方法
一旦授权策略被生成且注册,我们就可以为每个权限的授权增加方法。例如,让我们在 `PostPolicy` 中定义一个 `update` 方法,它会判断指定的 `User` 是否可以「更新」一条 `Post`。
`update` 方法接受 `User` 和 `Post` 实例作为参数,并且应当返回 `true` 或 `false` 来指明用户是否授权更新给定的 `Post`。因此,这个例子中,我们判断用户的 id 是否和 post 中的 `user_id` 匹配:
~~~
<?php
namespace App\Policies;
use App\User;
use App\Post;
class PostPolicy
{
/**
* 判断给定博客能否被用户更新
*
* @param \App\User $user
* @param \App\Post $post
* @return bool
*/
public function update(User $user, Post $post)
{
return $user->id === $post->user_id;
}
}
~~~
你可以接着在此授权策略定义额外的方法,作为各种权限需要的授权。例如,你可以定义 `view` 或 `delete` 方法来授权 `Post` 的多种行为。可以为自定义策略方法使用自己喜欢的名字。
> {tip} 如果在 Artisan 控制台生成策略使用 `--model` 选项,会自动包含 `view`、`create`、`update` 和 `delete` 动作。
### 不包含模型方法
一些策略方法只接受当前认证通过的用户作为参数,而不用传入授权相关的模型实例。最普遍的应用场景就是授权 `create` 动作。例如,如果正在创建一篇博客,你可能希望检查一下当前用户是否授权创建博客。
当定义一个不需要传入模型实例的策略方法时,比如 `create` 方法,你需要定义这个方法只接受已授权的用户作为参数:
~~~
/**
* 判断给定用户是否可以创建博客。
*
* @param \App\User $user
* @return bool
*/
public function create(User $user)
{
//
}
~~~
### 策略过滤器
对特定用户,你可能希望通过指定的策略授权所有动作。要达到这个目的,可以在策略中定义一个 `before` 方法。`before` 方法会在策略中其他所有方法之前执行,这样提供了一种方式来授权动作而不是指定的策略方法来执行判断。这个功能最常见的场景是授权应用的管理员可以访问所有动作:
~~~
public function before($user, $ability)
{
if ($user->isSuperAdmin()) {
return true;
}
}
~~~
如果你想拒绝用户所有的授权,你应该在 `before` 方法中返回 `false`。如果返回的是 `null`,则通过其他的策略方法来决定授权与否。
## 使用策略授权动作
### 通过用户模型
Laravel 应用内置的 `User` 模型包含 2 个有用的方法来授权动作:`can` 和 `cant`。`can` 方法指定需要授权的动作和相关的模型。例如,判定一个用户是否授权更新给定的 `Post` 模型:
~~~
if ($user->can('update', $post)) {
//
}
~~~
如果给定模型的 [策略已被注册](用户认证.md),`can` 方法会自动调用核实的策略方法并且返回 boolean 值。如果没有策略注册到这个模型,`can` 方法会尝试调用和动作名相匹配的基于闭包的 Gate。
#### 不需要指定模型的动作
一些动作,比如 `create`,并不需要指定模型实例。在这种情况下,可传递一个类名给 `can` 方法。当授权动作时,这个类名将被用来判断使用哪个策略:
~~~
use App\Post;
if ($user->can('create', Post::class)) {
// 执行相关策略中的「create」方法...
}
~~~
### 通过中间件
Laravel 包含一个可以在请求到达路由或控制器之前就进行动作授权的中间件。默认,`Illuminate\Auth\Middleware\Authorize` 中间件被指定到 `App\Http\Kernel` 类中 `can` 键上。我们用一个授权用户更新博客的例子来讲解 `can` 中间件的使用:
~~~
use App\Post;
Route::put('/post/{post}', function (Post $post) {
// 当前用户可以更新博客...
})->middleware('can:update,post');
~~~
在这个例子中,我们传递给 `can` 中间件 2 个参数。第一个是需要授权的动作的名称,第二个是我们希望传递给策略方法的路由参数。这里因为使用了 [隐式模型绑定](路由.md),一个 `Post` 会被传递给策略方法。如果用户不被授权访问指定的动作,这个中间件会生成带有 `403` 状态码的 HTTP 响应。
#### 不需要指定模型的动作
同样的,一些动作,比如 `create`,并不需要指定模型实例。在这种情况下,可传递一个类名给中间件。当授权动作时,这个类名将被用来判断使用哪个策略:
~~~
Route::post('/post', function () {
// 当前用户可以创建博客...
})->middleware('can:create,App\Post');
~~~
### 通过控制器辅助函数
除了在 `User` 模型中提供辅助方法外,Laravel 也为所有继承了 `App\Http\Controllers\Controller` 基类的控制器提供了一个有用的 `authorize` 方法。和 `can` 方法类似,这个方法接收需要授权的动作和相关的模型作为参数。如果动作不被授权,`authorize` 方法会抛出 `Illuminate\Auth\Access\AuthorizationException` 异常,然后被 Laravel 默认的异常处理器转化为带有 `403` 状态码的 HTTP 响应:
~~~
<?php
namespace App\Http\Controllers;
use App\Post;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller;
class PostController extends Controller
{
/**
* 更新给定博客
*
* @param Request $request
* @param Post $post
* @return Response
*/
public function update(Request $request, Post $post)
{
$this->authorize('update', $post);
// 当前用户可以更新博客...
}
}
~~~
#### 不需要指定模型的动作
和之前讨论的一样,一些动作,比如 `create`,并不需要指定模型实例。在这种情况下,可传递一个类名给 `authorize` 方法。当授权动作时,这个类名将被用来判断使用哪个策略:
~~~
/**
* 新建博客
*
* @param Request $request
* @return Response
*/
public function create(Request $request)
{
$this->authorize('create', Post::class);
// 当前用户可以新建博客...
}
~~~
### 通过 Blade 模板
当编写 Blade 模板时,你可能希望页面的指定部分只展示给允许授权访问给定动作的用户。例如,你可能希望只展示更新表单给有权更新博客的用户。这种情况下,你可以直接使用 `@can` 和 `@cannot` 指令。
~~~
@can('update', $post)
<!-- 当前用户可以更新博客 -->
@elsecan('create', $post)
<!-- 当前用户可以新建博客 -->
@endcan
@cannot('update', $post)
<!-- 当前用户不可以更新博客 -->
@elsecannot('create', $post)
<!-- 当前用户不可以新建博客 -->
@endcannot
~~~
这些指令在编写 `@if` 和 `@unless` 时提供了方便的缩写。`@can` 和 `@cannot` 各自转化为如下声明:
~~~
@if (Auth::user()->can('update', $post))
<!-- 当前用户可以更新博客 -->
@endif
@unless (Auth::user()->can('update', $post))
<!-- 当前用户不可以更新博客 -->
@endunless
~~~
#### 不需要指定模型的动作
和大部分其他的授权方法类似,当动作不需要模型实例时,你可以传递一个类名给 `@can` 和 `@cannot` 指令:
~~~
@can('create', Post::class)
<!-- 当前用户可以新建博客 -->
@endcan
@cannot('create', Post::class)
<!-- 当前用户不可以新建博客 -->
@endcannot
~~~
- 前言
- 翻译说明
- 发行说明
- 升级说明
- 贡献导引
- 入门指南
- 安装
- 配置信息
- 文件夹结构
- 请求周期
- 开发环境部署
- Homestead
- Valet
- 核心概念
- 服务容器
- 服务提供者
- Facades
- Contracts
- HTTP层
- 路由
- 中间件
- CSRF 保护
- 控制器
- 请求
- 响应
- 视图
- Session
- 表单验证
- 前端
- Blade 模板
- 本地化
- 前端指南
- 编辑资源 Mix
- 安全
- 用户认证
- Passport OAuth 认证
- 用户授权
- 加密解密
- 哈希
- 重置密码
- 综合话题
- Artisan 命令行
- 广播系统
- 缓存系统
- 集合
- 错误与日志
- 事件系统
- 文件存储
- 辅助函数
- 邮件发送
- 消息通知
- 扩展包开发
- 队列
- 任务调度
- 数据库
- 快速入门
- 查询构造器
- 分页
- 数据库迁移
- 数据填充
- Redis
- Eloquent ORM
- 快速入门
- 模型关联
- Eloquent 集合
- 修改器
- 序列化
- 测试
- 快速入门
- HTTP 测试
- 浏览器测试 Dusk
- 数据库测试
- 测试模拟器
- 官方扩展包
- Cashier 交易工具包
- Envoy 部署工具
- Scout 全文搜索
- Socialite 社会化登录