Laravel 中间件 middleware

简介

Laravel 中间件提供了一种方便的机制来过滤进入应用的 HTTP 请求,例如,Laravel 包含验证用户身份权限的中间件。如果用户没有通过身份验证,中间件会重定向到登录页,引导用户登录。反之,中间件将允许该请求继续传递到应用程序。

当然,除了身份验证以外,中间件还可以被用来执行各式各样的任务,如:CORS 中间件负责为所有即将离开应用的响应添加适当的头信息;日志中间件可以记录所有传入应用的请求。

Laravel 已经内置了一些中间件,包括身份验证、CSRF 保护等。所有的中间件都放在 app/Http/Middleware 目录内。

创建中间件

使用 make:middleware 这个 Artisan 命令创建新的中间件:

1
php artisan make:middleware CheckAge

该命令将会在 app/Http/Middleware 目录内新建一个 CheckAge 类。在这个中间件内,我们仅允许请求的 age 参数大于 200 时访问该路由,否则,会将用户请求重定向到 home URI 。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
<?php

namespace App\Http\Middleware;

use Closure;

class CheckAge
{
/**
* 处理传入的请求
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @return mixed
*/
public function handle($request, Closure $next)
{
if ($request->age <= 200) {
return redirect('home');
}

return $next($request);
}
}

如你所见,若请求参数 age 小于等于 200,中间件将返回给客户端 HTTP 重定向,反之应用程序才会继续处理该请求。若将请求继续传递到应用程序(即允许通过中间件验证),只需将 $request 作为参数调用 $next 回调函数。

最好将中间件想象为一系列的「层」,HTTP 请求必须经过它们才会触发您的应用程序。每一层都可以检测接收的请求,甚至可以完全拒绝请求访问您的应用。

前置中间件 / 后置中间件

中间件运行在请求之前或之后取决于中间件本身。例如,以下中间件会在请求被应用处理 之前 执行一些任务

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<?php

namespace App\Http\Middleware;

use Closure;

class BeforeMiddleware
{
public function handle($request, Closure $next)
{
// 执行动作

return $next($request);
}
}

然而,这个中间件会在请求被应用处理 之后 执行它的任务:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<?php

namespace App\Http\Middleware;

use Closure;

class AfterMiddleware
{
public function handle($request, Closure $next)
{
$response = $next($request);

// 执行动作

return $response;
}
}

注册中间件

全局中间件

如果你希望访问你应用的每个 HTTP 请求都经过某个中间件,只需将该中间件类列入 app/Http/Kernel.php 类里的 $middleware 属性。

1
2
3
4
5
6
7
8
// App\Http\Kernel 类内

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

为路由指定中间件

如果你想为特殊的路由指定中间件,首先应该在 app/Http/Kernel.php 文件内为该中间件指定一个 键值。默认情况下 Kernel 类的 $routeMiddleware 属性已经包含了 Laravel 内置的中间件条目。加入自定义的中间件,只需把它附加到此列表并指定你定义的键值即可。例如:

1
2
3
4
5
6
7
8
9
10
// App\Http\Kernel 类内

protected $routeMiddleware = [
'auth' => \Illuminate\Auth\Middleware\Authenticate::class,
'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class,
'bindings' => \Illuminate\Routing\Middleware\SubstituteBindings::class,
'can' => \Illuminate\Auth\Middleware\Authorize::class,
'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class,
'throttle' => \Illuminate\Routing\Middleware\ThrottleRequests::class,
];

一旦在 HTTP kernel 文件内定义了中间件,即可使用 middleware 方法将中间件分配给路由:

1
2
3
Route::get('admin/profile', function () {
//
})->middleware('auth');

为路由指定多个中间件:

1
2
3
Route::get('/', function () {
//
})->middleware('first', 'second');

也可使用完整类名指定中间件:

1
2
3
4
5
se App\Http\Middleware\CheckAge;

Route::get('admin/profile', function () {
//
})->middleware(CheckAge::class);

中间件组

有时您可能想要将多个中间件分组到同一个键值下,从而使它们更方便地分配给路由,你可以使用 HTTP kernel 的 $middlewareGroups 属性来实现。

Laravel 带有开箱即用的 webapi 中间件,包含了可能应用到 Web UI 和 API 路由的通用中间件:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/**
* 应用的路由中间件组
*
* @var array
*/
protected $middlewareGroups = [
'web' => [
\App\Http\Middleware\EncryptCookies::class,
\Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
\Illuminate\Session\Middleware\StartSession::class,
\Illuminate\View\Middleware\ShareErrorsFromSession::class,
\App\Http\Middleware\VerifyCsrfToken::class,
\Illuminate\Routing\Middleware\SubstituteBindings::class,
],

'api' => [
'throttle:60,1',
'auth:api',
],
];

中间件组可以像单个中间件一样使用相同的语法指定给路由和控制器。重申,中间件组仅仅是为了使一次将多个中间件指定给某个路由的实现变得更加方便。

1
2
3
4
5
6
7
Route::get('/', function () {
//
})->middleware('web');

Route::group(['middleware' => ['web']], function () {
//
});

tip: 开箱即用的 web 中间件组被自动应用于 RouteServiceProvider 中定义的 routes/web.php 路由组。

中间件参数

中间件也可以接受其他附加的参数。例如,如果应用需要在运行特定操作前验证该用户具备该操作的权限的「角色」,你可以新建一个 CheckRole 中间件,该中间件接收「角色」名字作为附加参数。

附加的中间件参数将在 $next 参数之后被传入:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<?php

namespace App\Http\Middleware;

use Closure;

class CheckRole
{
/**
* 处理传入的请求
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @param string $role
* @return mixed
*/
public function handle($request, Closure $next, $role)
{
if (! $request->user()->hasRole($role)) {
// Redirect...
}

return $next($request);
}

}

定义路由时,指定中间件参数可以通过冒号 : 来隔开中间件与参数,多个参数可以使用逗号分隔:

1
2
3
Route::put('post/{id}', function ($id) {
//
})->middleware('role:editor');

Terminable 中间件

有些时候中间件需要在 HTTP 响应发送到浏览器后运行来处理一些任务。比如,Laravel 内置的「session」中间件存储的 session 数据是在响应被发送到浏览器之后才进行写入的。想实现这一点,你需要在中间件中定义一个 terminate 方法,它会在响应发送后自动被调用:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<?php

namespace Illuminate\Session\Middleware;

use Closure;

class StartSession
{
public function handle($request, Closure $next)
{
return $next($request);
}

public function terminate($request, $response)
{
// Store the session data...
}
}

terminate 方法必需接收请求及响应两个参数。一旦定义了 terminable 中间件,你便需要将它增加到 HTTP kernel 文件的全局中间件清单列表中

中间件的 terminate 调用时,Laravel 会从 服务容器 中解析一个全新的中间件实例。如果你想在 handleterminate 被调用时使用同一个中间件实例,可使用容器的 singleton 方法向容器注册中间件。

0%