0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

FuelPHPの例外エイリアスがキャッチできない理由を探る

Posted at

以下に清書した記事をお送りします。


FuelPHPのAutoloaderとエイリアスの例外キャッチに関する検証

FuelPHPでエイリアスによる例外キャッチの動作について気になる点があったため、検証を行いました。弊社ではFuelPHPフレームワークを使用しており、404エラーなどにHttpNotFoundExceptionを利用しています。この例外はFuel\Core\HttpNotFoundExceptionのエイリアスとして定義されているはずですが、Fuel\Core\HttpNotFoundExceptionを投げた場合にHttpNotFoundExceptionでキャッチできないケースが発生しました。これを踏まえ、具体的に調査を行いました。

検証環境

本検証はPHP8.0の環境で実施しました。

$ php -v
PHP 8.0.30 (cli) (built: Apr 10 2024 07:34:47) ( NTS gcc x86_64 )

検証内容

エイリアスとオリジナルの例外がどのようにキャッチされるかを以下の4パターンで検証しました。

<?php

/**
 * @internal
 * @coversNothing
 */
class Controller_Test extends Controller
{
    // パターン1: エイリアスのHttpNotFoundExceptionをエイリアスでキャッチ
    public function action_alias_to_alias(): Response
    {
        try {
            throw new HttpNotFoundException();
        } catch (HttpNotFoundException) {
            return Response::forge('HttpNotFoundException を HttpNotFoundException でキャッチできた!' . PHP_EOL);
        } catch (Exception) {
            return Response::forge('HttpNotFoundException を HttpNotFoundException でキャッチできなかった…' . PHP_EOL);
        }
        return Response::forge('例外をキャッチできなかった…');
    }

    // パターン2: オリジナルのFuel\Core\HttpNotFoundExceptionをオリジナルでキャッチ
    public function action_original_to_original(): Response
    {
        try {
            throw new Fuel\Core\HttpNotFoundException();
        } catch (Fuel\Core\HttpNotFoundException) {
            return Response::forge('Fuel\Core\HttpNotFoundException を Fuel\Core\HttpNotFoundException でキャッチできた!' . PHP_EOL);
        } catch (Exception) {
            return Response::forge('Fuel\Core\HttpNotFoundException を Fuel\Core\HttpNotFoundException でキャッチできなかった…' . PHP_EOL);
        }
        return Response::forge('例外をキャッチできなかった…');
    }

    // パターン3: エイリアスのHttpNotFoundExceptionをオリジナルでキャッチ
    public function action_alias_to_original(): Response
    {
        try {
            throw new HttpNotFoundException();
        } catch (Fuel\Core\HttpNotFoundException) {
            return Response::forge('HttpNotFoundException を Fuel\Core\HttpNotFoundException でキャッチできた!' . PHP_EOL);
        } catch (Exception) {
            return Response::forge('HttpNotFoundException を Fuel\Core\HttpNotFoundException でキャッチできなかった…' . PHP_EOL);
        }
        return Response::forge('例外をキャッチできなかった…');
    }

    // パターン4: オリジナルのFuel\Core\HttpNotFoundExceptionをエイリアスでキャッチ
    public function action_original_to_alias(): Response
    {
        try {
            throw new Fuel\Core\HttpNotFoundException();
        } catch (HttpNotFoundException) {
            return Response::forge('Fuel\Core\HttpNotFoundException を HttpNotFoundException でキャッチできた!' . PHP_EOL);
        } catch (Exception) {
            return Response::forge('Fuel\Core\HttpNotFoundException を HttpNotFoundException でキャッチできなかった…' . PHP_EOL);
        }
        return Response::forge('例外をキャッチできなかった…');
    }
}

実行結果

各パターンの実行結果は以下の通りです。

$ curl -k https://localhost/test/alias_to_alias
HttpNotFoundException を HttpNotFoundException でキャッチできた!
$ curl -k https://localhost/test/original_to_original
Fuel\Core\HttpNotFoundException を Fuel\Core\HttpNotFoundException でキャッチできた!
$ curl -k https://localhost/test/alias_to_original   
HttpNotFoundException を Fuel\Core\HttpNotFoundException でキャッチできた!
$ curl -k https://localhost/test/original_to_alias
Fuel\Core\HttpNotFoundException を HttpNotFoundException でキャッチできなかった…

この結果から、オリジナルの例外をエイリアスでキャッチすることができないことが分かりました。

追加検証:エイリアスの事前参照

catchはクラスが定義されていなくても記述可能なため、エイリアスが作成されていない可能性が考えられました。そのため、あらかじめオリジナルとエイリアスのインスタンスを生成し、オリジナルをエイリアスでキャッチする以下の検証を追加しました。

// オリジナルのFuel\Core\HttpNotFoundExceptionをエイリアスでキャッチ
public function action_prepared_original_to_alias(): Response
{
    $original = new Fuel\Core\HttpNotFoundException();
    $alias = new HttpNotFoundException();
    try {
        throw $original;
    } catch (HttpNotFoundException) {
        return Response::forge('Fuel\Core\HttpNotFoundException を HttpNotFoundException でキャッチできた!' . PHP_EOL);
    } catch (Exception) {
        return Response::forge('Fuel\Core\HttpNotFoundException を HttpNotFoundException でキャッチできなかった…' . PHP_EOL);
    }
    return Response::forge('例外をキャッチできなかった…');
}

実行結果は以下の通りで、予想通りFuelPHPではクラスの初回参照時にエイリアスが生成されることが確認できました。

$ curl -k https://localhost/test/prepared_original_to_alias
Fuel\Core\HttpNotFoundException を HttpNotFoundException でキャッチできた!

結論

FuelPHPの例外エイリアスも、PHPのclass_aliasを利用した場合と同様に動作します。エイリアスを一度も参照していない場合は、エイリアスが生成されないためキャッチできません。エイリアスを利用する場合は、最初の参照タイミングに留意する必要があります。
FuelPHPではFuel\Coreに定義されているClassはすべてエイリアスで参照可能なため、意図的にOverrideされると困るClassを参照したい場合以外は、Overrideする内容も含めて解決してくれるエイリアスを利用することを徹底するほうが良さそうです。


NOTE: 未定義の例外をキャッチに指定する場合

以下は未定義の例外クラスをキャッチに指定した場合の動作例です。

$ php -a
php > try {
php {     throw new Exception();
php { } catch (UndefinedTestException) {
php {     echo 'Catch UndefinedTestException' . PHP_EOL;
php { } catch (Exception) {
php {     echo 'Catch Exception' . PHP_EOL;
php { }
Catch Exception

未定義の例外をキャッチに指定してもエラーにはならず、次に一致する例外がキャッチされます。

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?