web-dev-qa-db-ja.com

どうやって捕まえますか PHP 致命的な誤り

私はset_error_handler()を使用してほとんどのPHPエラーをキャッチできますが、存在しない関数を呼び出すなどの致命的な(E_ERROR)エラーには機能しません。これらのエラーを捉える他の方法はありますか?

すべてのエラーについてmail()を呼び出そうとしていますが、PHP 5.2.3を実行しています。

524
too much php

PHP 5.2以降が必要なregister_shutdown_functionを使用して致命的なエラーを記録します。

register_shutdown_function( "fatal_handler" );

function fatal_handler() {
    $errfile = "unknown file";
    $errstr  = "shutdown";
    $errno   = E_CORE_ERROR;
    $errline = 0;

    $error = error_get_last();

    if( $error !== NULL) {
        $errno   = $error["type"];
        $errfile = $error["file"];
        $errline = $error["line"];
        $errstr  = $error["message"];

        error_mail(format_error( $errno, $errstr, $errfile, $errline));
    }
}

error_mail関数とformat_error関数を定義する必要があります。例えば:

function format_error( $errno, $errstr, $errfile, $errline ) {
    $trace = print_r( debug_backtrace( false ), true );

    $content = "
    <table>
        <thead><th>Item</th><th>Description</th></thead>
        <tbody>
            <tr>
                <th>Error</th>
                <td><pre>$errstr</pre></td>
            </tr>
            <tr>
                <th>Errno</th>
                <td><pre>$errno</pre></td>
            </tr>
            <tr>
                <th>File</th>
                <td>$errfile</td>
            </tr>
            <tr>
                <th>Line</th>
                <td>$errline</td>
            </tr>
            <tr>
                <th>Trace</th>
                <td><pre>$trace</pre></td>
            </tr>
        </tbody>
    </table>";
    return $content;
}

error_mail関数を書くには Swift Mailer を使います。

また見なさい:

608
user259973

この解決方法(PHP 5.2.0以降)を思いついたところです。

function shutDownFunction() { 
    $error = error_get_last();
    // fatal error, E_ERROR === 1
    if ($error['type'] === E_ERROR) { 
        //do your stuff     
    } 
}
register_shutdown_function('shutDownFunction');

http://www.php.net/manual/en/errorfunc.constants.phpで定義されているさまざまなエラータイプ

137
periklis

PHPは、致命的なエラーを検出して回復するための従来の手段を提供していません。これは、致命的なエラーが発生した後に処理が通常回復されるべきではないためです。 (PHP.netで説明されているテクニックの元の記事で示唆されているように)出力バッファにマッチする文字列は絶対にお勧めできません。それは単に信頼できません。

エラーハンドラメソッド内からmail()関数を呼び出すことも問題があることがわかります。もしあなたがたくさんのエラーを抱えていたら、あなたのメールサーバは仕事でロードされているでしょう、そしてあなたは異常な受信箱であなた自身を見つけることができます。これを回避するには、cronを実行してエラーログを定期的にスキャンし、それに応じて通知を送信することを検討します。 Nagios のようなシステム監視ソフトウェアも検討してください。


シャットダウン機能の登録について少し触れておくと、

それはあなたがシャットダウン機能を登録することができるというのは本当です、そしてそれは良い答えです。

ここで重要なのは、特に出力バッファに対して正規表現を使用することによってではなく、通常致命的エラーからの回復を試みるべきではないということです。私は 受け入れられた答え に答えていました、それはその後変更されたか削除されたphp.netの提案にリンクしました。

その提案は例外処理中に出力バッファに対して正規表現を使用することであり、致命的なエラー(あなたが期待しているかもしれないどんな設定されたエラーテキストに対するマッチングによっても検出される)の場合。これは推奨されるやり方ではありません(私が最初の提案を見つけることができないのもそのためです。見逃しているか、phpコミュニティがそれを見落としています)。

出力バッファリングコールバックが呼び出される前に、より新しいバージョンのPHP(5.1前後)がシャットダウン関数を早く呼び出すように思われることは注目に値するかもしれません。バージョン5以前では、この順序は逆になりました(出力バッファリングコールバックの後にシャットダウン関数が続きました)。また、5.0.5以降(質問者のバージョン5.2.3よりはるかに早い)から、登録されたシャットダウン関数が呼び出される前にオブジェクトがアンロードされるので、インメモリオブジェクトに頼ることはできません。何でも。

そのため、シャットダウン機能の登録は問題ありませんが、シャットダウン機能によって実行されるべきタスクの種類は、ほんの一握りの穏やかなシャットダウン手順に限られています。

ここで鍵となるのは、この質問につまずいて、最初に受け入れられた答えの中でアドバイスを見る人のための知恵の言葉です。あなたの出力バッファを正規表現しないでください。

113
keparo

致命的エラーを他の方法で捉えることも可能です。

ob_start('fatal_error_handler');

function fatal_error_handler($buffer){
    $error=error_get_last();
    if($error['type'] == 1){
        // type, message, file, line
        $newBuffer='<html><header><title>Fatal Error </title></header>
                    <style>                 
                    .error_content{                     
                        background: ghostwhite;
                        vertical-align: middle;
                        margin:0 auto;
                        padding:10px;
                        width:50%;                              
                     } 
                     .error_content label{color: red;font-family: Georgia;font-size: 16pt;font-style: italic;}
                     .error_content ul li{ background: none repeat scroll 0 0 FloralWhite;                   
                                border: 1px solid AliceBlue;
                                display: block;
                                font-family: monospace;
                                padding: 2%;
                                text-align: left;
                      }
                    </style>
                    <body style="text-align: center;">  
                      <div class="error_content">
                          <label >Fatal Error </label>
                          <ul>
                            <li><b>Line</b> '.$error['line'].'</li>
                            <li><b>Message</b> '.$error['message'].'</li>
                            <li><b>File</b> '.$error['file'].'</li>                             
                          </ul>

                          <a href="javascript:history.back()"> Back </a>                          
                      </div>
                    </body></html>';

        return $newBuffer;

    }

    return $buffer;

}
37
sakhunzai

私はPHP内のすべてのエラータイプを捕らえる方法を開発しました(ほとんどすべて)。私はE_CORE_ERRORについて確信が持てません(私はそのエラーだけではうまくいかないと思います)。しかし、他の致命的なエラー(E_ERROR、E_PARSE、E_COMPILE ...)では、エラーハンドラ関数を1つだけ使用してもうまくいきます。私の解決策があります:

メインファイル(index.php)に次のコードを追加してください。

<?php

define('E_FATAL',  E_ERROR | E_USER_ERROR | E_PARSE | E_CORE_ERROR | 
        E_COMPILE_ERROR | E_RECOVERABLE_ERROR);

define('ENV', 'dev');

//Custom error handling vars
define('DISPLAY_ERRORS', TRUE);
define('ERROR_REPORTING', E_ALL | E_STRICT);
define('LOG_ERRORS', TRUE);

register_shutdown_function('shut');

set_error_handler('handler');

//Function to catch no user error handler function errors...
function shut(){

    $error = error_get_last();

    if($error && ($error['type'] & E_FATAL)){
        handler($error['type'], $error['message'], $error['file'], $error['line']);
    }

}

function handler( $errno, $errstr, $errfile, $errline ) {

    switch ($errno){

        case E_ERROR: // 1 //
            $typestr = 'E_ERROR'; break;
        case E_WARNING: // 2 //
            $typestr = 'E_WARNING'; break;
        case E_PARSE: // 4 //
            $typestr = 'E_PARSE'; break;
        case E_NOTICE: // 8 //
            $typestr = 'E_NOTICE'; break;
        case E_CORE_ERROR: // 16 //
            $typestr = 'E_CORE_ERROR'; break;
        case E_CORE_WARNING: // 32 //
            $typestr = 'E_CORE_WARNING'; break;
        case E_COMPILE_ERROR: // 64 //
            $typestr = 'E_COMPILE_ERROR'; break;
        case E_CORE_WARNING: // 128 //
            $typestr = 'E_COMPILE_WARNING'; break;
        case E_USER_ERROR: // 256 //
            $typestr = 'E_USER_ERROR'; break;
        case E_USER_WARNING: // 512 //
            $typestr = 'E_USER_WARNING'; break;
        case E_USER_NOTICE: // 1024 //
            $typestr = 'E_USER_NOTICE'; break;
        case E_STRICT: // 2048 //
            $typestr = 'E_STRICT'; break;
        case E_RECOVERABLE_ERROR: // 4096 //
            $typestr = 'E_RECOVERABLE_ERROR'; break;
        case E_DEPRECATED: // 8192 //
            $typestr = 'E_DEPRECATED'; break;
        case E_USER_DEPRECATED: // 16384 //
            $typestr = 'E_USER_DEPRECATED'; break;

    }

    $message = '<b>'.$typestr.': </b>'.$errstr.' in <b>'.$errfile.'</b> on line <b>'.$errline.'</b><br/>';

    if(($errno & E_FATAL) && ENV === 'production'){

        header('Location: 500.html');
        header('Status: 500 Internal Server Error');

    }

    if(!($errno & ERROR_REPORTING))
        return;

    if(DISPLAY_ERRORS)
        printf('%s', $message);

    //Logging error on php file error log...
    if(LOG_ERRORS)
        error_log(strip_tags($message), 0);

}

ob_start();

@include 'content.php';

ob_end_flush();

?>

これが多くの人に役立つことを願っています!私はこの解決策を探していたのは長すぎるし、見つからなかった!それから私はそれを開発しました!

22

致命的なエラーをキャッチ/処理することはできませんが、ログに記録/レポートすることはできます。迅速なデバッグのために、私はこの単純なコードに対する一つの答えを修正しました

function __fatalHandler()
{
    $error = error_get_last();
//check if it's a core/fatal error, otherwise it's a normal shutdown
    if ($error !== NULL && in_array($error['type'], array(E_ERROR, E_PARSE, E_CORE_ERROR, E_CORE_WARNING, E_COMPILE_ERROR, E_COMPILE_WARNING,E_RECOVERABLE_ERROR))) {
        echo "<pre>fatal error:\n";
        print_r($error);
        echo "</pre>";
        die;
    }
}

register_shutdown_function('__fatalHandler');
18
zainengineer

あなたはそのような登録されたシャットダウン機能の中で例外を投げることができません:

<?php
function shutdown() {
    if (($error = error_get_last())) {
       ob_clean();
       throw new Exception("fatal error");
    }
}

try {
    $x = null;
    $x->method()
} catch(Exception $e) {
    # this won't work
}
?>

しかし、リクエストをキャプチャして別のページにリダイレクトすることはできます。

<?php
function shutdown() {
    if (($error = error_get_last())) {
       ob_clean();
       # raport the event, send email etc.
       header("Location: http://localhost/error-capture");
       # from /error-capture, you can use another redirect, to e.g. home page
    }
}
register_shutdown_function('shutdown');

$x = null;
$x->method()
?>
17
hipertracker

致命的なエラーまたは回復可能な致命的なエラーが、 PHP 7以降のバージョンのErrorのインスタンスをスローするようになりました 他の例外と同様に、Errorオブジェクトはtry/catchブロックを使用して捕捉できます。

例:

<?php
$variable = 'not an object';

try {
    $variable->method(); // Throws an Error object in PHP 7 or higger.
} catch (Error $e) {
    // Handle error
    echo $e->getMessage(); // Call to a member function method() on string
}

https://3v4l.org/67vbk

あるいはThrowableインターフェースを使ってすべての例外をキャッチすることもできます。

例:

<?php
    try {
        undefinedFunctionCall();
    } catch (Throwable $e) {
        // Handle error
        echo $e->getMessage(); // Call to undefined function undefinedFunctionCall()
    }

https://3v4l.org/Br0MG

より詳しい情報は: http://php.net/manual/en/language.errors.php7.php

16
LugiHaue

Php> = 5.1.0を使っているのであれば、ErrorExceptionクラスを使って次のようにしてください。

<?php
//define an error handler
function exception_error_handler($errno, $errstr, $errfile, $errline ) {
    throw new ErrorException($errstr, $errno, 0, $errfile, $errline);
}
//set ur error handle
set_error_handler("exception_error_handler");

/* Trigger exception */
try
{
  //try to do something like finding the end of the internet
}
catch(ErrorException $e)
{
  //anything you want to do with $e
}

?>
11
Cyril Tata

プロダクションの致命的なエラーを処理して、静的な503 Service Unavailable HTML出力を表示する必要があります。これは確かに「致命的なエラーを検出する」ための合理的な方法です。これは私がやったことです:

E_ERROR、E_USER_ERRORなどに "503 service unavailable" HTMLページを表示するカスタムのエラー処理関数 "error_handler"があります。これはシャットダウン関数で呼び出され、致命的なエラーを検出します。

function fatal_error_handler() {

    if (@is_array($e = @error_get_last())) {
        $code = isset($e['type']) ? $e['type'] : 0;
        $msg = isset($e['message']) ? $e['message'] : '';
        $file = isset($e['file']) ? $e['file'] : '';
        $line = isset($e['line']) ? $e['line'] : '';
        if ($code>0) error_handler($code,$msg,$file,$line);
    }
}
set_error_handler("error_handler");
register_shutdown_function('fatal_error_handler');

カスタムのerror_handler関数で、エラーがE_ERRORまたはE_USER_ERRORなどの場合は、@ob_end_clean()も呼び出します。バッファを空にしてPHPの "致命的なエラー"メッセージを取り除きます。

Error_handlerスクリプトがエラーを生成したくないので、厳密なisset()チェックと@ silencing関数に注意してください。

それでもなおkeparoと同意することで、致命的なエラーを捉えることは「致命的なエラー」の目的を無効にするので、それはあなたがさらなる処理をすることを本当に意図したものではありません。このシャットダウンプロセスではmail()関数を実行しないでください。確実にメールサーバーまたは受信トレイをバックアップします。これらの発生をファイルに記録して、これらのerror.logファイルを見つけて管理者にメールで送信するようにcronをスケジュールしてください。

8
Prof83

PHPには致命的なエラーがあります。それらはE_RECOVERABLE_ERRORとして定義されています。 PHPマニュアルはE_RECOVERABLE_ERRORを次のように記述しています。

キャッチ可能な致命的エラーおそらく危険なエラーが発生したが、エンジンが不安定な状態にならなかったことを示しています。エラーがユーザ定義のハンドルによって捕捉されない場合( set_error_handler() も参照)、アプリケーションはE_ERRORであるため中止されます。

set_error_handler() を使用してE_RECOVERABLE_ERRORをチェックすることで、これらの「致命的な」エラーを「キャッチ」できます。このエラーが発生したときにExceptionをスローすると便利です。それからtry/catchを使用できます。

この質問と回答は有用な例を提供します: PHPタイプのヒンティングで "キャッチ可能な致命的なエラー"を見つけるにはどうすればいいですか?

E_ERRORエラーは処理できますが、エンジンが不安定な状態にあるため回復できません。

7
J.Money

現在のerror_handlerメソッドを取得するためのちょっとした便利なトリック=)

<?php
register_shutdown_function('__fatalHandler');
function __fatalHandler()
{
    $error      = error_get_last();

    //check if it's a core/fatal error, otherwise it's a normal shutdown
    if($error !== NULL && $error['type'] === E_ERROR) {
        //Bit hackish, but the set_exception_handler will return the old handler
        function fakeHandler() { }
        $handler = set_exception_handler('fakeHandler');
        restore_exception_handler();
        if($handler !== null) { 
            call_user_func($handler, new ErrorException($error['message'], $error['type'], 0, $error['file'], $error['line']));
        }
        exit;
    }
}
?>

また、電話をしても気にしないでください。

<?php
ini_set('display_errors', false);
?>

Phpはエラーの表示を停止します。そうでなければエラーテキストはエラーハンドラの前にクライアントに送信されます。

5
Sander Visser

ここでのほとんどの答えは不必要に冗長なので、トップ投票の答えの私の醜いバージョンは以下のとおりです。

function errorHandler($errno, $errstr, $errfile = '', $errline = 0, $errcontext = array()) {
    //Do stuff: mail, log, etc
}

function fatalHandler() {
    $error = error_get_last();
    if($error) errorHandler($error["type"], $error["message"], $error["file"], $error["line"]);
}

set_error_handler("errorHandler")
register_shutdown_function("fatalHandler");
5
Mahn

あんまり。致命的なエラーは致命的であるため、それと呼ばれます。あなたはそれらから回復することはできません。

4
troelskn

致命的なエラーでさえキャッチされるべきである特定の状況があります(あなたは優雅に終了する前にいくつかのクリーンアップをする必要があるかもしれません、そして単に死なないでください..)。致命的なエラーを電子メールで受け取ることができるように、codeigniterアプリケーションにpre_systemフックを実装しました。これにより、報告されていない(または、既に知っていたので修正後に報告された)バグを見つけることができました。 Sendemailは、エラーがすでに報告されているかどうかを確認し、既知のエラーがスパムされないようにします。

class PHPFatalError {

    public function setHandler() {
        register_shutdown_function('handleShutdown');
    }

}

function handleShutdown() {
    if (($error = error_get_last())) {
        ob_start();
        echo "<pre>";
        var_dump($error);
        echo "</pre>";
        $message = ob_get_clean();
        sendEmail($message);
        ob_start();
        echo '{"status":"error","message":"Internal application error!"}';
        ob_flush();
        exit();
    }
}
3
tix3

致命的なエラーを引き起こす可能性のあるコードを「サンドボックス化」できるようにするために、この関数を開発しました。クロージャーregister_shutdown_functionから投げられた例外は致命的なエラー呼び出しスタックから発生しないので、私はそれを使用するための統一された方法を提供するためにこの関数の後に終了することを強制します。

function superTryCatchFinallyAndExit( Closure $try, Closure $catch = NULL, Closure $finally )
{
    $finished = FALSE;
    register_shutdown_function( function() use ( &$finished, $catch, $finally ) {
        if( ! $finished ) {
            $finished = TRUE;
            print "EXPLODE!".PHP_EOL;
            if( $catch ) {
                superTryCatchFinallyAndExit( function() use ( $catch ) {
                    $catch( new Exception( "Fatal Error!!!" ) );
                }, NULL, $finally );                
            } else {
                $finally();                
            }
        }
    } );
    try {
        $try();
    } catch( Exception $e ) {
        if( $catch ) {
            try {
                $catch( $e );
            } catch( Exception $e ) {}
        }
    }
    $finished = TRUE;
    $finally();
    exit();
}
2
Kendall Hopkins

私は、PHPのすべてのエラーを検出するための完全な解決策を備えたWikiスタイルのQ&Aを書きました。 /見ることができる/拾う/盗まれる/批評される ここ

このソリューションには、PHPが生成できるすべてのエラーをラップする5つのメソッドが含まれています。これらのメソッドは、最終的に上記のエラーを 'ErrorHandler'型付きオブジェクトに渡します。

何人かの人々がそれから何らかの用途を得ることを願っています。あなたがそれを完全に盗んでいなくても、私はその解決策があらゆる点でPHPのエラーを処理する方法の少なくとも良い例であると確信しています。

@Lucas Batistussiは創造性についてのポイントを得ました - 私は自分の解決策を共有して似たようなポイントを狙うことができると思います...

1
DigitalJedi805