0 x00 preface
Middleware is a feature provided by many PHP frameworks, and I was surprised and excited when I first encountered it. Because it was so powerful, before middleware we had to write permission validation and common operations in controller methods, and then the controller became bloated, reducing readability and maintainability. But with middleware we can write these operations in the middleware, and then use different combinations of middleware to not only fulfill the requirements but also reduce the coupling of the code. Since middleware is so great, how does it work? Reading Laravel and Slim’s source code (an agonizing, torturous and unnerving exercise), I found that the emphasis was on placing multiple middleware closures (some frame-ware that were not implemented through closures but fell into the Callable category), Collectively called closures for the sake of writing) the process of packaging it into a closure by array_reduce or loop.
0 x01 preheating
This picture is often associated with middleware
This may look amazing and scary, but if you look closely, does this process actually look like a nested call to Middleware2(Middleware1(App()))? This is obviously wrong because PHP executes App() first instead of Middleware2. But what about a nested closure? Sample code is shown below:
$allMiddleware = function (a) {
echo 'start middleware2' . PHP_EOL;
(function (a) {
echo 'start middleware1' . PHP_EOL;
// app
(function (a) {
echo 'app'. PHP_EOL; }) ();// end app
echo 'end middleware1'. PHP_EOL; }) ();echo 'end middleware2' . PHP_EOL;
};
$allMiddleware();
/ / output
// start middleware2
// start middleware1
// app
// end middleware1
// end middleware2
Copy the code
0 x02 thinking
Hey, the above code turns out as expected, but it seems a bit simple and ugly. But the $allMiddleware in the code above is the result of middleware combinations, so now that we have reached the threshold, how can we automatically combine the following middleware closures?
// Database middleware
$db = function (Closure $next) {
echo 'Database connection established successfully' . PHP_EOL;
$next();
echo 'Database connection closed successfully' . PHP_EOL;
};
// Thumbs up middleware
$like = function (Closure $next) {
echo 'thumb up + 1' . PHP_EOL;
$next();
echo 'thumb up + 2' . PHP_EOL;
};
// Content closure
$app = function (a) {
echo 'Article content' . PHP_EOL;
};
Copy the code
What the hell is $next? In the case of Middleware2, its $next is the closure that Middleware1 and APP are packaged into (see the innermost two layers above).
0 x03 answer
The answer code is as follows:
/ / array_reduce implementation
$allMiddleware = [$like, $db];
$go = array_reduce($allMiddleware, function ($next, $middleware) {
return function (a) use ($next, $middleware) {
$middleware($next);
};
}, $app);
$go();
/ / foreach implementation
$allMiddleware = [$like, $db];
$next = $app;
foreach ($allMiddleware as $middleware) {
$next = function (a) use ($next, $middleware) {
return $middleware($next);
};
}
$next();
Copy the code
Both implementations work the same way, so only the foreach version is explained. Start by grouping all middleware into an array and setting $Next to $app, and then loop $Next and middleware into a new closure and assign it to $Next, so that $Next will merge the previous closures into one. The result is then obtained by executing the final $next. While the words may be hard to follow, the best way is to mentally execute the code myself and then draw it (which is how I learned when I first saw the code). If you don’t know what the array_reduce function does, click here.
0x04 better version
In order to understand the previous middleware only has $next parameter, but the actual framework middleware will have a similar $request parameter and support the return value of $response (such as Laravel), the following is a middleware similar to Laravel (just mimic, draw a picture of the tiger-like) implementation code. One law, ten thousand law, this dish force does not explain (in case involving my knowledge blind area GG, O(∩_∩)O ha ha ~).
$db = function ($request, Closure $next) {
echo 'Database connection established successfully' . PHP_EOL;
$response = $next($request);
echo 'Database connection closed successfully' . PHP_EOL;
return $response;
};
$like = function ($request, Closure $next) {
echo 'thumb up + 1' . PHP_EOL;
$response = $next($request);
echo 'thumb up + 2' . PHP_EOL;
return $response;
};
$app = function ($request) {
echo $request . PHP_EOL;
return 'A boring return value';
};
$allMiddleware = [$like, $db];
$next = $app;
foreach ($allMiddleware as $middleware) {
$next = function ($request) use ($middleware, $next) {
return $middleware($request, $next);
};
}
$response = $next('O (studying studying) O');
echo $response;
Copy the code
0 x05 summary
(⊙ V ⊙) Well, a little bit of content goes a long way. And if there is an error in the article, in the hope that we can point to, if have any questions can discuss with each other: -d.
My blog post