PHP Curl - 在不允许时接收到 HTTP/0.9

新手上路,请多包涵

当我尝试向苹果推送服务发送 HTTP/2.0 后请求时,我偶然发现了一个奇怪的行为:

         $http2ch = curl_init();
        curl_setopt($http2ch, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_2_0);
        curl_setopt($http2ch, CURLOPT_URL, 'https://api.push.apple.com/3/device/megauniquedevicetokendummy');
        curl_setopt($http2ch, CURLOPT_PORT, 443);
        curl_setopt($http2ch, CURLOPT_HTTPHEADER, $httpHeader);
        curl_setopt($http2ch, CURLOPT_POST, true);
        curl_setopt($http2ch, CURLOPT_POSTFIELDS, $body);
        curl_setopt($http2ch, CURLOPT_RETURNTRANSFER, true);
        curl_setopt($http2ch, CURLOPT_TIMEOUT, 30);
        curl_setopt($http2ch, CURLOPT_HEADER, 1);

       $result = curl_exec($http2ch);
       if ($result === false) {
           throw new \Exception("Curl failed: " . curl_error($http2ch) . " | " . curl_getinfo($http2ch, CURLINFO_HTTP_CODE));
       }

异常随消息一起抛出:Curl failed: Received HTTP/0.9 when not allowed | 0

我在上面截取的代码的第二行明确告诉 curl 使用 HTTP/2.0。有谁知道该错误消息的含义以及为什么 curl 使用如此旧的 HTTP 版本?

我使用的是 PHP 7.2 和 curl 版本 7.66.0。

原文由 rockZ 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 3.4k
2 个回答

我想到了。确保 curl 是使用 nghttp2 编译的。

如果不确定,可以在终端上使用 curl --version 检查它

如果你没有找到 nghttp2/{version} 你需要用 nghttp2 再次编译 curl。

curl --version 缺少 nghttp2 的示例:

 curl 7.66.0 (amd64-portbld-freebsd12.0) libcurl/7.66.0 OpenSSL/1.1.1d zlib/1.2.11

curl --version nghttp2 可用的示例:

 curl 7.64.1 (x86_64-apple-darwin19.0) libcurl/7.64.1 (SecureTransport) LibreSSL/2.8.3 zlib/1.2.11 nghttp2/1.39.2

原文由 rockZ 发布,翻译遵循 CC BY-SA 4.0 许可协议

当服务器是 grpc 服务器时也会发生这种情况。当 curl 针对 grpc 服务器或其他非 HTTP 服务器运行时,如果 curl 没有响应 curl 期望的有效 HTTP 状态行,curl 将打印“Received HTTP/0.9 when not allowed”。

如果 curl 打印出诸如“未知协议”之类的东西而不是假设它是 0.9 可能会更好,因为如今访问诸如 grpc 服务器之类的东西比实际的 HTTP 0.9 服务器要普遍得多。

原文由 d-chord 发布,翻译遵循 CC BY-SA 4.0 许可协议

推荐问题
logo
Stack Overflow 翻译
子站问答
访问
宣传栏