PHPスクリプトから500内部サーバーエラーエラーを送信する方法


80

特定の条件下で、PHPスクリプトから「500InternalServerError」を送信する必要があります。スクリプトは、サードパーティのアプリによって呼び出されることになっています。スクリプトには、通常のの代わりに応答コードdie("this happend")を送信する必要があるステートメントがいくつか含まれ500 Internal Server Errorてい200 OKます。サードパーティのスクリプトは、200 OK応答コードを受信しないなどの特定の条件下で要求を再送信します。

質問の2番目の部分:スクリプトを次のように設定する必要があります。

<?php
    custom_header( "500 Internal Server Error" );

    if ( that_happened ) {
        die( "that happened" )
    }

    if ( something_else_happened ) {
        die( "something else happened" )
    }

    update_database( );

    // the script can also fail on the above line
    // e.g. a mysql error occurred

    remove_header( "500" );
?>

200最後の行が実行された後にのみヘッダーを送信する必要があります。

編集

副次的な質問:次のような奇妙な500ヘッダーを送信できますか?

HTTP/1.1 500 No Record Found
HTTP/1.1 500 Script Generated Error (E_RECORD_NOT_FOUND)
HTTP/1.1 500 Conditions Failed on Line 23

そのようなエラーはWebサーバーによってログに記録されますか?


uがヘッダーを送信し、後でヘッダーを削除する場合は実行できません
ajreal 2010年

1
副次的な質問:それは完全に合法です。理由フレーズは、マシンの消費を目的としたものではなく、何でもかまいません。重要なのは3桁のステータスコードだけです。(RFC2616 6.1.1:「ここにリストされている理由フレーズは推奨事項にすぎません。プロトコルに影響を与えることなく、ローカルの同等のものに置き換えることができます。」)
Piskvorは

回答:


168
header($_SERVER['SERVER_PROTOCOL'] . ' 500 Internal Server Error', true, 500);

参考までに、このソリューションはX-Padを送信します。Apacheの一部のバージョンではブラウザーのバグヘッダーを回避します。stackoverflow.com/questions/8711584/…http_response_code()このヘッダーを省略します。
Anthony Rutledge 2015年

3
これは、ヘッダーが最初にない場合でも機能します(動作の証明など)。
私は私です

44

PHP 5.4にはhttp_response_codeという関数があるため、PHP 5.4を使用している場合は、次のようにすることができます。

http_response_code(500);

5.4でPHPのバージョンを実行している場合は、この関数(Gist)のポリフィルを作成しました。


フォローアップの質問に答えるために、HTTP 1.1RFCは次のように述べています。

ここにリストされている理由フレーズは推奨事項にすぎません。プロトコルに影響を与えることなく、ローカルの同等のフレーズに置き換えることができます。

つまり、コード自体の後に必要なテキスト(キャリッジリターンまたはラインフィードを除く)を使用でき、それが機能します。ただし、一般的には、通常、より適切な応答コードを使用する必要があります。たとえば、レコードが見つからない場合に500を使用する代わりに、404(見つかりません)を送信できます。「条件が失敗しました」(検証エラーだと思います)などの場合は、422(処理不能)などを送信できます。エンティティ)。


出力が開始されると、この関数も機能しなくなることに注意することが重要です。
rob74 2015年

@ rob74 True — PHPが出力の送信を開始すると、ヘッダー関連のすべての関数が機能しなくなります。出力の送信を開始する前に応答コードを変更する必要があるかどうかわからない場合は、出力バッファリングが適切なソリューションです。
inxilpro 2015年

33

次の関数を使用して、ステータス変更を送信できます。

function header_status($statusCode) {
    static $status_codes = null;

    if ($status_codes === null) {
        $status_codes = array (
            100 => 'Continue',
            101 => 'Switching Protocols',
            102 => 'Processing',
            200 => 'OK',
            201 => 'Created',
            202 => 'Accepted',
            203 => 'Non-Authoritative Information',
            204 => 'No Content',
            205 => 'Reset Content',
            206 => 'Partial Content',
            207 => 'Multi-Status',
            300 => 'Multiple Choices',
            301 => 'Moved Permanently',
            302 => 'Found',
            303 => 'See Other',
            304 => 'Not Modified',
            305 => 'Use Proxy',
            307 => 'Temporary Redirect',
            400 => 'Bad Request',
            401 => 'Unauthorized',
            402 => 'Payment Required',
            403 => 'Forbidden',
            404 => 'Not Found',
            405 => 'Method Not Allowed',
            406 => 'Not Acceptable',
            407 => 'Proxy Authentication Required',
            408 => 'Request Timeout',
            409 => 'Conflict',
            410 => 'Gone',
            411 => 'Length Required',
            412 => 'Precondition Failed',
            413 => 'Request Entity Too Large',
            414 => 'Request-URI Too Long',
            415 => 'Unsupported Media Type',
            416 => 'Requested Range Not Satisfiable',
            417 => 'Expectation Failed',
            422 => 'Unprocessable Entity',
            423 => 'Locked',
            424 => 'Failed Dependency',
            426 => 'Upgrade Required',
            500 => 'Internal Server Error',
            501 => 'Not Implemented',
            502 => 'Bad Gateway',
            503 => 'Service Unavailable',
            504 => 'Gateway Timeout',
            505 => 'HTTP Version Not Supported',
            506 => 'Variant Also Negotiates',
            507 => 'Insufficient Storage',
            509 => 'Bandwidth Limit Exceeded',
            510 => 'Not Extended'
        );
    }

    if ($status_codes[$statusCode] !== null) {
        $status_string = $statusCode . ' ' . $status_codes[$statusCode];
        header($_SERVER['SERVER_PROTOCOL'] . ' ' . $status_string, true, $statusCode);
    }
}

あなたはそれをそのように使うことができます:

<?php
header_status(500);

if (that_happened) {
    die("that happened")
}

if (something_else_happened) {
    die("something else happened")
}

update_database();

header_status(200);

21
数バイトのメモリを節約することが、Webアプリケーションコードを何らかの方法で作成する正当な理由になることはめったにありません。
ダン・グロスマン

1
;アフターを逃していませんdie()か?
ヘンリーライト2014年

2
リストがありません418 =>「私はティーポットです」を参照してください:en.wikipedia.org/wiki/Hyper_Text_Coffee_Pot_Control_Protocol
Louis Loudog Trottier 2017

16

あなたはただ置くことができます:

header("HTTP/1.0 500 Internal Server Error");

あなたの条件の中で:

if (that happened) {
    header("HTTP/1.0 500 Internal Server Error");
}

データベースクエリに関しては、次のように実行できます。

$result = mysql_query("..query string..") or header("HTTP/1.0 500 Internal Server Error");

このコードは、htmlタグ(または出力)の前に配置する必要があることを覚えておく必要があります。


11
header()を呼び出した後、必ずexit / die / return / somethingを実行してください。PHPはコードの実行を継続しますが、これはおそらく望ましくありません。
David Goodwin

8

次のように簡略化できます。

if ( that_happened || something_else_happened )
{
    header('X-Error-Message: Incorrect username or password', true, 500);
    die;
}

次のヘッダーが返されます。

HTTP/1.1 500 Internal Server Error
...
X-Error-Message: Incorrect username or password
...

追加:何がうまくいかなかったかを正確に知る必要がある場合は、次のようにします。

if ( that_happened )
{
    header('X-Error-Message: Incorrect username', true, 500);
    die('Incorrect username');
}

if ( something_else_happened )
{
    header('X-Error-Message: Incorrect password', true, 500);
    die('Incorrect password');
}

5
さて、何'x'になるはずですか?
コアXii 2010年

1
+1の最初のパラメータはheader空でない文字列でなければならないことに注意してください。それ'x'は本当に重要ではありません。
theazureshadow 2010年

1
@Core Xii、@ theazureshadowが指摘したように、最初のパラメーターはnullであってはなりません。つまり、header( 'something'、true、500)を呼び出すと、正しいヘッダー「HTTP / 1.0500内部サーバーエラー」が返されます。私を怠惰と呼ぶかもしれませんが、実際のヘッダー文字列を処理するよりもエラーコードを渡す方が簡単です:)詳細については、php.net / manual / en /function.header.phpを参照してください。
デヴィッド・Kuridža

マニュアルはこれについてあまり明確ではありません。つまり、ステータスコードを強制すると、PHPは文字列引数を正しい値に自動的に上書きするということですか?それでは、なぜマニュアルは、文字列でない場合にのみ有効になると言っているのでしょうか?
コアXii 2010年

1
一部のPHP構成(私のホスティングプロバイダーが例に使用しているが、ローカルセットアップには使用していない)では、このトリックは機能しないことに注意してください!Apacheは「x」を正しいヘッダー文字列として認識せず、「不正なヘッダー」エラーで失敗します。
mjsarfatti 2012年

2

コードは次のようになります。

<?php
if ( that_happened ) {
    header("HTTP/1.0 500 Internal Server Error");
    die();
}

if ( something_else_happened ) {
    header("HTTP/1.0 500 Internal Server Error");
    die();
}

// Your function should return FALSE if something goes wrong
if ( !update_database() ) {
    header("HTTP/1.0 500 Internal Server Error");
    die();
}

// the script can also fail on the above line
// e.g. a mysql error occurred


header('HTTP/1.1 200 OK');
?>

何か問題が発生した場合は実行を停止すると思います。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.