草庐IT

php - HTTP2 header 与旧版浏览器的兼容性

coder 2024-04-11 原文

今天听说现代浏览器很快就会实现http2协议(protocol)。更多信息:https://en.wikipedia.org/wiki/HTTP/2 ,我知道维基百科不是最好的资源,但它会提供一些关于正在发生的事情的线索。问题是:


旧浏览器将如何响应 http2 header ?

我的意思是在 php ( http://php.net ) 上 (26.02.2015) 头函数 (http://php.net/manual/en/function.header.php) 中仍然有一个链接到 http1.1 规范 (http://www.faqs.org/rfcs/rfc2616)。我知道在 http2 中我必须做的就是将 header 从例如 HTTP/1.1 404 Not Found 更改为类似于 HTTP/2.0 404 Not Found。但是旧的浏览器将如何响应呢?这对 web 开发人员和 php 编码人员来说是透明的,并在浏览器/服务器端实现,还是有一些关于兼容性的重要事情/线索?

准备好后立即使用 http2 header 是个好主意吗?

我不想伤害任何人,但我知道这样一种浏览器,它的名字以字母 I 开头,第二个以字母 E 开头,即总是会搞砸一点。恐怕新规范会彻底毁掉这个浏览器的所有旧版本,以及这个 http2。我们 - 开发人员必须编写无论在何处都能正常工作的网站,神奇的 http2 将在数以百万计的补丁/升级/数月与旧机器的兼容性问题之后实现。


在格式正确的问题中一定有一些代码,所以这里是:):

<?php
    header("HTTP/2.0 404 Not Found"); // Am I correct? It will look like this?
?>

在这种情况下旧浏览器怎么办?

在 http2 生效后立即使用它是个好主意吗?


附加文档:

  1. http2 规范草案:https://datatracker.ietf.org/doc/html/draft-ietf-httpbis-http2-17
  2. 维基百科几句话:https://en.wikipedia.org/wiki/HTTP/2
  3. PHP header 函数:http://php.net/manual/en/function.header.php
  4. W3.org 关于 http2:http://www.w3.org/Protocols/HTTP/HTTP2.html

最佳答案

在典型情况下,支持 HTTP/2 的客户端联系服务器最初使用 HTTP/1.1 连接,使用 Upgrade: header 指示 HTTP/2 支持的可用性。

这看起来像这样:

 GET / HTTP/1.1
 Host: server.example.com
 Connection: Upgrade, HTTP2-Settings
 Upgrade: h2c
 HTTP2-Settings: <base64url encoding of HTTP/2 SETTINGS payload>

来源:HTTP/2 draft 17, Sec 3.2

我在那里直接引用了规范,因此来自真实客户端的真实请求看起来会稍微复杂一些——它也会包含通常的 HTTP/1.1 header ,这样服务器就不想升级到 HTTP/2可以简单地继续请求。不支持 HTTP/2 的服务器会简单地忽略 Upgrade: h2cHTTP2-Settings: ... header 。

将连接升级到 HTTP/2 的服务器响应:

 HTTP/1.1 101 Switching Protocols
 Connection: Upgrade
 Upgrade: h2c
  
 PRI * HTTP/2.0
  
 SM
  
  

来源:HTTP/2 草案 17,第 3.2 节

PRI * 开头的部分称为“客户端连接前言”,在任何 HTTP/2 连接开始时发送。它被设计为任何 HTTP/1.0 或/1.1 服务器都不会响应的字符串。这意味着即使在浏览器有理由相信服务器在连接之前支持 HTTP/2 的非典型场景中(例如,我们正在谈论的可能是通过其他协议(protocol)进行服务广告的 Intranet 环境),HTTP/2 客户端将打开与 "PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n" 的连接并立即被 HTTP/1.1 400 Bad Request 拒绝 通过任何 HTTP/1.X 服务器。 (例如,客户端然后可能会将请求降级为 HTTP/1.1 并继续。)

重写 PHP 应用程序以直接使用 HTTP/2 比更改 header('HTTP/X.X...') 数字要花费更多的工作。 Saikyr 是对的,HTTP/2 是一个二进制协议(protocol);这意味着我们将使用一个新的库直接从 PHP 编写 HTTP/2。

不再发送状态代码和消息以及协议(protocol)版本。相反,使用了伪 header ——基本上只是 header 前面带有 : 以确保不会与 HTTP/1.X header 冲突。然而, header 不会以明文形式发送——已经制定了一种压缩方案,用于将 header 转换为二进制代码。 HPACK draft 12. (我还没有读过它;它不像 HTTP/2 那样引人入胜。)

因此,代替 header("HTTP/2 404 Not Found");,您将执行如下操作:

 \HTTP2::setHeader(':status','404'); 

或者代替 header("Location: $PROTO://$HOST$PATH");

 \HTTP2::setHeader('location', "$PROTO://$HOST$PATH"); 

我还不知道 PHP 中有这样的支持,所以我有点猜测。我不确定 PHP 是否可以获得对 Apache 连接的足够控制以从 HTTP/2 升级并管理连接本身。随着对 Apache 的支持增加,PHP 可能有机会在库级别进行链接。无论新库是什么,它可能仍然能够编写 HTTP/1.1 连接的 header ,具体取决于客户端——但它将需要重写软件,除非他们决定使用 header()重新解析 HTTP/2 输出的每个参数(这并非不可想象,但在我看来它会导致比它解决的问题更多的问题)。

可以添加 HTTP/2 支持的一种方法是在服务器/传输级别,将 HTTP/1.1 header 转换为 HTTP/2。可以想象,Apache 模块可以解析 HTML 输出并检测链接的样式表、脚本和图像,从而在响应 GET 请求时向客户端生成 PUSH_PROMISE。

直接在 PHP 中支持 HTTP/2 的主要优势是允许链接到的资源“推送”给愿意支持该功能的客户。许多基于 PHP 的 CMS 环境将能够提供样式表、脚本和图像的列表,而无需重新解析为客户端准备的输出。

我很期待看到我们能做些什么!

关于php - HTTP2 header 与旧版浏览器的兼容性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28740791/

有关php - HTTP2 header 与旧版浏览器的兼容性的更多相关文章

  1. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  2. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

  3. Get https://registry-1.docker.io/v2/: net/http: request canceled while waiting - 2

    1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里

  4. ruby-on-rails - Rails - 从命名路由中提取 HTTP 动词 - 2

    Rails中有没有一种方法可以提取与路由关联的HTTP动词?例如,给定这样的路线:将“users”匹配到:“users#show”,通过:[:get,:post]我能实现这样的目标吗?users_path.respond_to?(:get)(显然#respond_to不是正确的方法)我最接近的是通过执行以下操作,但它似乎并不令人满意。Rails.application.routes.routes.named_routes["users"].constraints[:request_method]#=>/^GET$/对于上下文,我有一个设置cookie然后执行redirect_to:ba

  5. ruby-on-rails - Heroku 吃掉了我的自定义 HTTP header - 2

    我正在使用Heroku(heroku.com)来部署我的Rails应用程序,并且正在构建一个iPhone客户端来与之交互。我的目的是将手机的唯一设备标识符作为HTTPheader传递给应用程序以进行身份​​验证。当我在本地测试时,我的header通过得很好,但在Heroku上它似乎去掉了我的自定义header。我用ruby​​脚本验证:url=URI.parse('http://#{myapp}.heroku.com/')#url=URI.parse('http://localhost:3000/')req=Net::HTTP::Post.new(url.path)#boguspara

  6. ruby-on-rails - 使用 HTTP.get_response 检索 Facebook 访问 token 时出现 Rails EOF 错误 - 2

    我试图在我的网站上实现使用Facebook登录功能,但在尝试从Facebook取回访问token时遇到障碍。这是我的代码:ifparams[:error_reason]=="user_denied"thenflash[:error]="TologinwithFacebook,youmustclick'Allow'toletthesiteaccessyourinformation"redirect_to:loginelsifparams[:code]thentoken_uri=URI.parse("https://graph.facebook.com/oauth/access_token

  7. ruby - HTTP 请求中的用户代理,Ruby - 2

    我是Ruby的新手。我试过查看在线文档,但没有找到任何有效的方法。我想在以下HTTP请求botget_response()和get()中包含一个用户代理。有人可以指出我正确的方向吗?#PreliminarycheckthatProggitisupcheck=Net::HTTP.get_response(URI.parse(proggit_url))ifcheck.code!="200"puts"ErrorcontactingProggit"returnend#Attempttogetthejsonresponse=Net::HTTP.get(URI.parse(proggit_url)

  8. ruby - 如何使用 Ruby HTTP::Net 处理 404 错误? - 2

    我正在尝试解析网页,但有时会收到404错误。这是我用来获取网页的代码:result=Net::HTTP::getURI.parse(URI.escape(url))如何测试result是否为404错误代码? 最佳答案 像这样重写你的代码:uri=URI.parse(url)result=Net::HTTP.start(uri.host,uri.port){|http|http.get(uri.path)}putsresult.codeputsresult.body这将打印状态码和正文。

  9. ruby - 让 bundler 使用 http : instead of git:? - 2

    我正在安装gitlabhq,并且在Gemfile中有对某些资源的“git://...”的引用。但是,我在公司防火墙后面,所以我必须使用http://。我可以手动编辑Gemfile,但我想知道是否有另一种方法告诉bundler使用http://作为git存储库? 最佳答案 您可以通过运行gitconfig--globalurl."https://".insteadOfgit://或通过将以下内容添加到~/.gitconfig:[url"https://"]insteadOf=git://

  10. ruby - 如何捕获所有 HTTP 流量(本地代理) - 2

    我希望访问我机器上的所有HTTP流量(我的Windows机器-不是服务器)。据我了解,拥有一个本地代理是所有流量路线的必经之路。我一直在谷歌搜索但未能找到任何资源(关于Ruby)来帮助我。非常感谢任何提示或链接。 最佳答案 WEBrick中有一个HTTP代理(Rubystdlib的一部分)和here's一个实现示例。如果你喜欢生活在边缘,还有em-proxy伊利亚·格里戈里克。这postIlya暗示它似乎确实需要一些调整来解决您的问题。 关于ruby-如何捕获所有HTTP流量(本地代理)

随机推荐