Edited at

StackPHPとミドルウェアについて調べてみた

More than 3 years have passed since last update.

ちょっとStackPHPについてを調べたので、まとめを書いてみます。

Middlewareとは何だろうと思い、調べているとStackPHPのウェブサイトやスライド(Do You Stack Up? (slideshare))が出てきます。考え方もコードも実に簡単なのですが、今ひとつ重要性がピンと来ませんでした。

が、このブログ「HttpKernel middlewares」を読んだら腑に落ちました。ということで、このブログを元に自分なりに理解した内容をまとめます。


HTTPの動作を理解する


HTTP

HTTPの基本は何でしょう?

簡単にPHPで表すと…

function call($request) {

/* do something */
return new Response($contents);
}

これだけ!

HTTPのリクエスト($request)を受け取り、結果をResponseとして返す。

レスポンスを返さなくても、リクエストを操作する場合もあります。例えば、IPアドレスから送信元の国を探してリクエストに設定する、などの機能も考えられます。

この基本動作を元にしてミドルウェアを構築します。


MiddleWare

先のコードをミドルウェアに変えます。

class MiddleWare {

protected $app;
function __construct($app) {
$this->app = $app;
}
function call(Request $request) {
/* do your stuff */
$response = $this->app->call($request); // call next middleware
/* do more stuff if necessary */
return $response;
}
}

あまり変わってませんが…

ポイントはcallメソッドの中で次のミドルウェアを呼び出していること($this->app->call)です。

このコードだと、こんな風に走らせることになります。

$app = new MiddleWare(new YourMiddleware(...));

$response = $app->call(Request::createFromGlobals());

こうやってミドルウェアを次々と呼び出すことで、処理を自由に積み重ねることが出来ます。必要なミドルウェアを適当に途中に挟んでも、大きな問題なく動くのも理解できます。

ちなみにミドルウェアでレスポンスを返すことも出来ます。認証ミドルウェアなら、認証してない場合は別URLに戻すレスポンスを返す、などできます。逆に、レスポンスを元にログをとったりキャッシュに保存したりすることも自由自在です。


StackPHPについて

このHttpの基本をもとにStackPHPは構築されています。


Middlewareサンプル

StackPHPでのミドルウェアの書き方です。

class MyMiddleWare implements HttpKernelInterface {

protected $app;
function __construct($app) {
$this->app = $app;
}
function handle(Request $request, $type, $catch) {
/* do your stuff */
$response = $this->app->handle($request, $type, $catch); // call next middleware
/* do more stuff if necessary */
return $response;
}
}

あまり変わってませんね。

大事な点は、StackPHPではSymfonyのRequestResponseクラスを使います。同じくミドルウェアにはHttpKernelInterfaceを利用するので、handleというメソッドで処理を行います。

コンストラクターの最初の引数に、次のミドルウェアを受け取るのも大事なポイントです。これでスタック可能になります。


アプリケーションの構築

実際のStackPHPの使い方です。

Stack/Builderを利用します。githubに書いてあるコードを少し簡単にしてあります。ちなみにSilexをウェブアプリとして使う場合です。

$app = new Silex\Application();

/* set up $app */

$stack = (new Stack\Builder())
->push('Stack\Session')
->push('More\Middleware')
->resolve($app); // 最後はresolveを呼ぶこと

// ミドルウェアとアプリ実行
$request = Request::createFromGlobals();
$response = $stack->handle($request);
$response->send();
$stack->terminate($request, $response);

最後にresolveを呼ぶことで、スタックを構築します。


たった3つのルール

まとめると、次の簡単な3つのルールを守るだけでミドルウェアを作れます。


  1. SymfonyのHttpKernelInterfaceを実装する。

  2. コンストラクターの最初の引数として次のミドルウェアを受け取る。

  3. handleメソッド内で次のミドルウェアを呼び出す。

これだけの簡単なルールで、


  • 安全で再利用しやすい、

  • フレームワークに依存しない、

ミドルウェアができる、というのがStackPHPの特徴です。


StackPHPの未来

StackPHPは、LaravelやDrupalにも採用されているので、これからも様々なミドルウェアが登場すると思います。しかもフレームワークに依存しないので、さまざまな場合で使えるようになると思います。


Symfonyへの依存問題

フレームワークのクラスに依存しないのは確かですが、Symfonyのインターフェースだけには依存します。

これが原因で問題が起きたことはあるようです。CakePHP3の開発が進んだ段階でStackPHPの採用について検討をしたようですが、すでに独自のリクエストやレスポンスクラスを採用した後だったため、Symfonyのクラスに依存させると問題が多くて断念したという話ですCakePHP3: Improve middleware (github issue #2544)


PHP-FIGによる標準化

今は、PHP-Figにより、Httpのインターフェース標準化作業が進められています(PHP-FigによるHTTPプロトコル)。

標準が決まれば、StackPHPも採用すると考えられます。すると、さらに互換性が高くなり、StackPHPの採用も進むと考えられます。


豊富なミドルウェア

StackPHPのウェブサイトにあるミドルウェアの一覧です。豊富、かどうかはおいておいて、今後も増えてゆくといいですね。