1
0
mirror of https://github.com/apache/httpd.git synced 2025-08-08 15:02:10 +03:00

*) mod_http2: preserve the port number given in a HTTP/1.1

request that was Upgraded to HTTP/2. Fixes PR65881.



git-svn-id: https://svn.apache.org/repos/asf/httpd/httpd/trunk@1898146 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Stefan Eissing
2022-02-17 10:06:31 +00:00
parent d27a32da69
commit 6137cd7b74
4 changed files with 63 additions and 6 deletions

View File

@@ -0,0 +1,3 @@
*) mod_http2: preserve the port number given in a HTTP/1.1
request that was Upgraded to HTTP/2. Fixes PR65881.
[Stefan Eissing]

View File

@@ -90,7 +90,19 @@ apr_status_t h2_request_rcreate(h2_request **preq, apr_pool_t *pool,
return APR_EINVAL;
}
if (!ap_strchr_c(authority, ':') && r->server && r->server->port) {
/* The authority we carry in h2_request is the 'authority' part of
* the URL for the request. r->hostname has stripped any port info that
* might have been present. Do we need to add it?
*/
if (!ap_strchr_c(authority, ':')) {
if (r->parsed_uri.port_str) {
/* Yes, it was there, add it again. */
authority = apr_psprintf(pool, "%s:%s", authority, r->parsed_uri.port_str);
}
else if (!r->parsed_uri.hostname && r->server && r->server->port) {
/* If there was no hostname in the parsed URL, the URL was relative.
* In that case, we restore port from our server->port, if it
* is known and not the default port for the scheme. */
apr_port_t defport = apr_uri_port_of_scheme(scheme);
if (defport != r->server->port) {
/* port info missing and port is not default for scheme: append */
@@ -98,6 +110,7 @@ apr_status_t h2_request_rcreate(h2_request **preq, apr_pool_t *pool,
(int)r->server->port);
}
}
}
req = apr_pcalloc(pool, sizeof(*req));
req->method = apr_pstrdup(pool, r->method);

View File

@@ -7,6 +7,7 @@ print()
print("{")
print(" \"https\" : \"%s\"," % (os.getenv('HTTPS', '')))
print(" \"host\" : \"%s\"," % (os.getenv('SERVER_NAME', '')))
print(" \"port\" : \"%s\"," % (os.getenv('SERVER_PORT', '')))
print(" \"protocol\" : \"%s\"," % (os.getenv('SERVER_PROTOCOL', '')))
print(" \"ssl_protocol\" : \"%s\"," % (os.getenv('SSL_PROTOCOL', '')))
print(" \"h2\" : \"%s\"," % (os.getenv('HTTP2', '')))

View File

@@ -0,0 +1,40 @@
import pytest
from .env import H2Conf
class TestProxyPort:
@pytest.fixture(autouse=True, scope='class')
def _class_scope(self, env):
conf = H2Conf(env, extras={
'base': [
f'Listen {env.proxy_port}',
'Protocols h2c http/1.1',
'LogLevel proxy_http2:trace2 proxy:trace2',
],
f'cgi.{env.http_tld}': [
"Header unset Server",
"Header always set Server cgi",
]
})
conf.add_vhost_cgi(proxy_self=False, h2proxy_self=False)
conf.start_vhost(domains=[f"test1.{env.http_tld}"], port=env.proxy_port)
conf.add([
'Protocols h2c',
'RewriteEngine On',
'RewriteRule "^/(.*)$" "h2c://%{HTTP_HOST}/$1"[NC,P]',
'ProxyPassMatch / "h2c://$1/"',
])
conf.end_vhost()
conf.install()
assert env.apache_restart() == 0
# Test PR 65881
# h2c upgraded request via a dynamic proxy onto another port
def test_h2_502_01(self, env):
url = f'http://localhost:{env.http_port}/hello.py'
r = env.curl_get(url, 5, options=['--http2',
'--proxy', f'localhost:{env.proxy_port}'])
assert r.response['status'] == 200
assert r.json['port'] == f'{env.http_port}'