📡

【PHP】HTTPレスポンスをパースする

2021/03/21に公開

ポイント・結論

  • ヘッダはiconv_mime_decode_headersという神parserが存在する
  • ステータスラインにおける標準的なparserは見つからなかったので正規表現を使った
  • HTTPのレスポンスはCRLFが標準なので、CRLFでexplode, preg_matchを行う

サンプルケース

レスポンス例は適当にネットのサンプルを拾わせていただきました。
※ 下記はLFになっちゃうかもですが、HTTPレスポンスはCRLFなので、適宜CRLFに替えて下さい。

sample_http_200.txt
HTTP/1.1 200 OK
Date: Mon, 27 Jul 2009 12:28:53 GMT
Server: Apache/2.2.14 (Win32)
Last-Modified: Wed, 22 Jul 2009 19:15:56 GMT
Content-Length: 88
Content-Type: text/html
Connection: Closed

<html>
<body>
<h1>Hello, World!</h1>
</body>
</html>
sample_http_404.txt
HTTP/1.1 404 Not Found
Date: Sun, 18 Oct 2012 10:36:20 GMT
Server: Apache/2.2.14 (Win32)
Content-Length: 230
Connection: Closed
Content-Type: text/html; charset=iso-8859-1

<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML 2.0//EN">
<html>
<head>
   <title>404 Not Found</title>
</head>
<body>
   <h1>Not Found</h1>
   <p>The requested URL /t.html was not found on this server.</p>
</body>
</html>

PHP

http_parser.php
<?php

/**
 * @param string $response
 * @return array
 * @throws Exception
 */
function parse_http_response(string $response)
{
    list($headers, $body) = explode("\r\n\r\n", $response, 2);

    if (!preg_match('@HTTP/[0-9\.]+\s+([0-9]+)\s+(.*)\r\n@', $headers, $matches)) {
        throw new Exception('status line not found.');
    }

    $status_code = (int)$matches[1];
    $headers = iconv_mime_decode_headers($headers);

    return [
        'status_code' => $status_code,
        'headers' => $headers,
        'body' => $body,
    ];
}

var_dump(parse_http_response(file_get_contents(__DIR__ . '/sample_http_200.txt')));

echo PHP_EOL;

var_dump(parse_http_response(file_get_contents(__DIR__ . '/sample_http_404.txt')));

結果

php http_parser.php
array(3) {
  ["status_code"]=>
  int(200)
  ["headers"]=>
  array(6) {
    ["Date"]=>
    string(29) "Mon, 27 Jul 2009 12:28:53 GMT"
    ["Server"]=>
    string(21) "Apache/2.2.14 (Win32)"
    ["Last-Modified"]=>
    string(29) "Wed, 22 Jul 2009 19:15:56 GMT"
    ["Content-Length"]=>
    string(2) "88"
    ["Content-Type"]=>
    string(9) "text/html"
    ["Connection"]=>
    string(6) "Closed"
  }
  ["body"]=>
  string(56) "<html>
<body>
<h1>Hello, World!</h1>
</body>
</html>"
}

array(3) {
  ["status_code"]=>
  int(404)
  ["headers"]=>
  array(5) {
    ["Date"]=>
    string(29) "Sun, 18 Oct 2012 10:36:20 GMT"
    ["Server"]=>
    string(21) "Apache/2.2.14 (Win32)"
    ["Content-Length"]=>
    string(3) "230"
    ["Connection"]=>
    string(6) "Closed"
    ["Content-Type"]=>
    string(29) "text/html; charset=iso-8859-1"
  }
  ["body"]=>
  string(224) "<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML 2.0//EN">
<html>
<head>
   <title>404 Not Found</title>
</head>
<body>
   <h1>Not Found</h1>
   <p>The requested URL /t.html was not found on this server.</p>
</body>
</html>"
}

参考

PHP: iconv_mime_decode_headers - Manual
PHP: preg_match - Manual
HTTP/1.1: Response

Discussion