中间件
中间件一般用于拦截请求或者响应。例如执行控制器前统一验证用户身份,如用户未登录时跳转到登录页面,例如响应中增加某个header头。例如统计某个uri请求占比等等。
中间件洋葱模型
┌──────────────────────────────────────────────────────┐
│ middleware1 │
│ ┌──────────────────────────────────────────┐ │
│ │ middleware2 │ │
│ │ ┌──────────────────────────────┐ │ │
│ │ │ middleware3 │ │ │
│ │ │ ┌──────────────────┐ │ │ │
│ │ │ │ │ │ │ │
── Reqeust ───────────────────> Controller ── Response ───────────────────────────> Client
│ │ │ │ │ │ │ │
│ │ │ └──────────────────┘ │ │ │
│ │ │ │ │ │
│ │ └──────────────────────────────┘ │ │
│ │ │ │
│ └──────────────────────────────────────────┘ │
│ │
└──────────────────────────────────────────────────────┘
中间件和控制器组成了一个经典的洋葱模型,中间件类似一层一层的洋葱表皮,控制器是洋葱芯。如图所示请求像箭一样穿越中间件1、2、3到达控制器,控制器返回了一个响应,然后响应又以3、2、1的顺序穿出中间件最终返回给客户端。也就是说在每个中间件里我们既可以拿到请求,也可以获得响应。
请求拦截
有时候我们不想某个请求到达控制器层,例如我们在middleware2发现当前用户并没有登录,则我们可以直接拦截请求并返回一个登录响应。那么这个流程类似下面这样
┌────────────────────────────────────────────────────────────┐
│ middleware1 │
│ ┌────────────────────────────────────────────────┐ │
│ │ middleware2 │ │
│ │ ┌──────────────────────────────┐ │ │
│ │ │ middleware3 │ │ │
│ │ │ ┌──────────────────┐ │ │ │
│ │ │ │ │ │ │ │
── Reqeust ─────────┐ │ │ Controller │ │ │ │
│ │ Response │ │ │ │ │ │
<───────────────────┘ │ └──────────────────┘ │ │ │
│ │ │ │ │ │
│ │ └──────────────────────────────┘ │ │
│ │ │ │
│ └────────────────────────────────────────────────┘ │
│ │
└────────────────────────────────────────────────────────────┘
如图所示请求到达middleware2后生成了一个登录响应,响应从middleware2穿越回中间件1然后返回给客户端。
中间件接口
中间件必须实现Webman\MiddlewareInterface
接口。
interface MiddlewareInterface
{
/**
* Process an incoming server request.
*
* Processes an incoming server request in order to produce a response.
* If unable to produce the response itself, it may delegate to the provided
* request handler to do so.
*/
public function process(Request $request, callable $handler): Response;
}
也就是必须实现process
方法,process
方法必须返回一个support\Response
对象,默认这个对象由$handler($request)
生成(请求将继续向洋葱芯穿越),也可以是response()
json()
xml()
redirect()
等助手函数生成的响应(请求停止继续向洋葱芯穿越)。
中间件中获取请求及响应
在中间件中我们可以获得请求,也可以获得执行控制器后的响应,所以中间件内部分为三个部分。
- 请求穿越阶段,也就是请求处理前的阶段
- 控制器处理请求阶段,也就是请求处理阶段
- 响应穿出阶段,也就是请求处理后的阶段
三个阶段在中间件里的体现如下
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class Test implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
echo '这里是请求穿越阶段,也就是请求处理前';
$response = $handler($request); // 继续向洋葱芯穿越,直至执行控制器得到响应
echo '这里是响应穿出阶段,也就是请求处理后';
return $response;
}
}
示例:身份验证中间件
创建文件app/middleware/AuthCheckTest.php
(如目录不存在请自行创建) 如下:
<?php
namespace app\middleware;
use ReflectionClass;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class AuthCheckTest implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
if (session('user')) {
// 已经登录,请求继续向洋葱芯穿越
return $handler($request);
}
// 通过反射获取控制器哪些方法不需要登录
$controller = new ReflectionClass($request->controller);
$noNeedLogin = $controller->getDefaultProperties()['noNeedLogin'] ?? [];
// 访问的方法需要登录
if (!in_array($request->action, $noNeedLogin)) {
// 拦截请求,返回一个重定向响应,请求停止向洋葱芯穿越
return redirect('/user/login');
}
// 不需要登录,请求继续向洋葱芯穿越
return $handler($request);
}
}
新建控制器 app/controller/UserController.php
<?php
namespace app\controller;
use support\Request;
class UserController
{
/**
* 不需要登录的方法
*/
protected $noNeedLogin = ['login'];
public function login(Request $request)
{
$request->session()->set('user', ['id' => 10, 'name' => 'webman']);
return json(['code' => 0, 'msg' => 'login ok']);
}
public function info()
{
return json(['code' => 0, 'msg' => 'ok', 'data' => session('user')]);
}
}
注意
$noNeedLogin
里记录了当前控制器不需要登录就可以访问的方法
在 config/middleware.php
中添加全局中间件如下:
return [
// 全局中间件
'' => [
// ... 这里省略其它中间件
app\middleware\AuthCheckTest::class,
]
];
有了身份验证中间件,我们就可以在控制器层专心的写业务代码,不用就用户是否登录而担心。
示例:跨域请求中间件
创建文件app/middleware/AccessControlTest.php
(如目录不存在请自行创建) 如下:
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class AccessControlTest implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
// 如果是options请求则返回一个空响应,否则继续向洋葱芯穿越,并得到一个响应
$response = $request->method() == 'OPTIONS' ? response('') : $handler($request);
// 给响应添加跨域相关的http头
$response->withHeaders([
'Access-Control-Allow-Credentials' => 'true',
'Access-Control-Allow-Origin' => $request->header('origin', '*'),
'Access-Control-Allow-Methods' => $request->header('access-control-request-method', '*'),
'Access-Control-Allow-Headers' => $request->header('access-control-request-headers', '*'),
]);
return $response;
}
}
提示
跨域可能会产生OPTIONS请求,我们不想OPTIONS请求进入到控制器,所以我们为OPTIONS请求直接返回了一个空的响应(response('')
)实现请求拦截。
如果你的接口需要设置路由,请使用Route::any(..)
或者Route::add(['POST', 'OPTIONS'], ..)
设置。
在 config/middleware.php
中添加全局中间件如下:
return [
// 全局中间件
'' => [
// ... 这里省略其它中间件
app\middleware\AccessControlTest::class,
]
];
注意
如果ajax请求自定义了header头,需要在中间件里Access-Control-Allow-Headers
字段加入这个自定义header头,否则会报Request header field XXXX is not allowed by Access-Control-Allow-Headers in preflight response.
说明
- 中间件分为全局中间件、应用中间件(应用中间件仅在多应用模式下有效,参见多应用)、路由中间件
- 中间件配置文件位置在
config/middleware.php
- 全局中间件配置在key
''
下 - 应用中间件配置在具体的应用名下,例如
return [
// 全局中间件
'' => [
app\middleware\AuthCheckTest::class,
app\middleware\AccessControlTest::class,
],
// api应用中间件(应用中间件仅在多应用模式下有效)
'api' => [
app\middleware\ApiOnly::class,
]
];
控制器中间件和方法中间件
利用注解,我们可以给某个控制器或者控制器的某个方法设置中间件。
注意
此特性需要webman-framework >= 1.6.11
<?php
namespace app\controller;
use app\middleware\Controller1Middleware;
use app\middleware\Controller2Middleware;
use app\middleware\Method1Middleware;
use app\middleware\Method2Middleware;
use support\annotation\Middleware;
use support\Request;
#[Middleware(Controller1Middleware::class, Controller2Middleware::class)]
class IndexController
{
#[Middleware(Method1Middleware::class, Method2Middleware::class)]
public function index(Request $request): string
{
return 'hello';
}
}
路由中间件
我们可以给某个一个或某一组路由设置中间件。
例如在config/route.php
中添加如下配置:
<?php
use support\Request;
use Webman\Route;
Route::any('/admin', [app\admin\controller\IndexController::class, 'index'])->middleware([
app\middleware\MiddlewareA::class,
app\middleware\MiddlewareB::class,
]);
Route::group('/blog', function () {
Route::any('/create', function () {return response('create');});
Route::any('/edit', function () {return response('edit');});
Route::any('/view/{id}', function ($r, $id) {response("view $id");});
})->middleware([
app\middleware\MiddlewareA::class,
app\middleware\MiddlewareB::class,
]);
中间件构造函数传参
注意
此特性需要webman-framework >= 1.4.8
1.4.8版本之后,配置文件支持直接实例化中间件或者匿名函数,这样可以方便的通过构造函数向中间件传参。
例如config/middleware.php
里也可以这样配置
return [
// 全局中间件
'' => [
new app\middleware\AuthCheckTest($param1, $param2, ...),
function(){
return new app\middleware\AccessControlTest($param1, $param2, ...);
},
],
// api应用中间件(应用中间件仅在多应用模式下有效)
'api' => [
app\middleware\ApiOnly::class,
]
];
同理路由中间件也可以通过构造函数向中间件传递参数,例如config/route.php
里
Route::any('/admin', [app\admin\controller\IndexController::class, 'index'])->middleware([
new app\middleware\MiddlewareA($param1, $param2, ...),
function(){
return new app\middleware\MiddlewareB($param1, $param2, ...);
},
]);
中间件里使用参数示例
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class MiddlewareA implements MiddlewareInterface
{
protected $param1;
protected $param2;
public function __construct($param1, $param2)
{
$this->param1 = $param1;
$this->param2 = $param2;
}
public function process(Request $request, callable $handler) : Response
{
var_dump($this->param1, $this->param2);
return $handler($request);
}
}
中间件执行顺序
- 中间件执行顺序为
全局中间件
->应用中间件
->控制器中间件
->路由中间件
->方法中间件
。 - 当同一个层次有多个中间件时,按照同层次中间件实际配置顺序执行。
- 404请求默认不会触发任何中间件(不过仍然可以通过
Route::fallback(function(){})->middleware()
添加中间件)。
路由向中间件传参(route->setParams)
路由配置 config/route.php
<?php
use support\Request;
use Webman\Route;
Route::any('/test', [app\controller\IndexController::class, 'index'])->setParams(['some_key' =>'some value']);
中间件(假设为全局中间件)
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class Hello implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
// 默认路由 $request->route 为null,所以需要判断 $request->route 是否为空
if ($route = $request->route) {
$value = $route->param('some_key');
var_export($value);
}
return $handler($request);
}
}
中间件向控制器传参
有时候控制器需要使用中间件里产生的数据,这时我们可以通过给$request
对象添加属性的方式向控制器传参。例如:
中间件
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class Hello implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
$request->data = 'some value';
return $handler($request);
}
}
控制器:
<?php
namespace app\controller;
use support\Request;
class FooController
{
public function index(Request $request)
{
return response($request->data);
}
}
中间件获取当前请求路由信息
注意
需要 webman-framework >= 1.3.2
我们可以使用 $request->route
获取路由对象,通过调用对应的方法获取相应信息。
路由配置
<?php
use support\Request;
use Webman\Route;
Route::any('/user/{uid}', [app\controller\UserController::class, 'view']);
中间件
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class Hello implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
$route = $request->route;
// 如果请求没有匹配任何路由(默认路由除外),则 $request->route 为 null
// 假设浏览器访问地址 /user/111,则会打印如下信息
if ($route) {
var_export($route->getPath()); // /user/{uid}
var_export($route->getMethods()); // ['GET', 'POST', 'PUT', 'DELETE', 'PATCH', 'HEAD','OPTIONS']
var_export($route->getName()); // user_view
var_export($route->getMiddleware()); // []
var_export($route->getCallback()); // ['app\\controller\\UserController', 'view']
var_export($route->param()); // ['uid'=>111]
var_export($route->param('uid')); // 111
}
return $handler($request);
}
}
注意
$route->param()
方法需要 webman-framework >= 1.3.16
中间件获取异常
注意
需要 webman-framework >= 1.3.15
业务处理过程中可能会产生异常,在中间件里使用 $response->exception()
获取异常。
路由配置
<?php
use support\Request;
use Webman\Route;
Route::any('/user/{uid}', function (Request $request, $uid) {
throw new \Exception('exception test');
});
中间件:
<?php
namespace app\middleware;
use Webman\MiddlewareInterface;
use Webman\Http\Response;
use Webman\Http\Request;
class Hello implements MiddlewareInterface
{
public function process(Request $request, callable $handler) : Response
{
$response = $handler($request);
$exception = $response->exception();
if ($exception) {
echo $exception->getMessage();
}
return $response;
}
}
超全局中间件
注意
此特性要求 webman-framework >= 1.5.16
主项目的全局中间件只影响主项目,不会对应用插件产生影响。有时候我们想要加一个影响全局包括所有插件的中间件,则可以使用超全局中间件。
在config/middleware.php
中配置如下:
return [
'@' => [ // 给主项目及所有插件增加全局中间件
app\middleware\MiddlewareGlobl::class,
],
'' => [], // 只给主项目增加全局中间件
];
提示
@
超全局中间件不仅可以在主项目配置,也可以在某个插件里配置,例如plugin/ai/config/middleware.php
里配置@
超全局中间件,则也会影响主项目及所有插件。
给某个插件增加中间件
注意
此特性要求 webman-framework >= 1.5.16
有时候我们想给某个应用插件增加一个中间件,又不想改插件的代码(因为升级会被覆盖),这时候我们可以在主项目中给它配置中间件。
在config/middleware.php
中配置如下:
return [
'plugin.ai' => [], // 给ai插件增加中间件
'plugin.ai.admin' => [], // 给ai插件的admin模块(plugin\ai\app\admin目录)增加中间件
];
提示
当然也可以在某个插件中加类似的配置去影响其它插件,例如plugin/foo/config/middleware.php
里加入如上配置,则会影响ai插件。