Skip to content

Commit 1c9cf87

Browse files
committed
Merge of PR 297
Fixing limits on response header size * Fixed bug in handling over long response headers. When the 64 KB limit of nghttp2 was exceeded, the request was not reset and the client was left hanging, waiting for it. Now the stream is reset. * Added new directive `H2MaxHeaderBlockLen` to set the limit on response header sizes.
1 parent dc5f41c commit 1c9cf87

File tree

9 files changed

+337
-2
lines changed

9 files changed

+337
-2
lines changed

ChangeLog

+5
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,8 @@
1+
* Fixed bug in handling over long response headers. When the 64 KB limit
2+
of nghttp2 was exceeded, the request was not reset and the client was
3+
left hanging, waiting for it. Now the stream is reset.
4+
* Added new directive `H2MaxHeaderBlockLen` to set the limit on response
5+
header sizes.
16
* Fixed handling of Timeout vs. KeepAliveTimeout when first request on a
27
connection was reset.
38

README.md

+9
Original file line numberDiff line numberDiff line change
@@ -135,6 +135,7 @@ The following configuration directives are available:
135135
* [H2EarlyHint](#h2earlyhint)
136136
* [H2EarlyHints](#h2earlyhints)
137137
* [H2MaxDataFrameLen](#h2maxdataframelen)
138+
* [H2MaxHeaderBlockLen](#h2maxheaderblocklen)
138139
* [H2MaxSessionStreams](#h2maxsessionstreams)
139140
* [H2MaxWorkerIdleSeconds](#h2maxworkeridleseconds)
140141
* [H2MaxWorkers](#h2maxworkers)
@@ -225,6 +226,14 @@ H2MaxDataFrameLen limits the maximum amount of response body bytes placed into a
225226

226227
The module, by default, tries to use the maximum size possible, which is somewhat around 16KB. This sets the maximum. When less response data is available, smaller frames will be sent.
227228

229+
### H2MaxHeaderBlockLen
230+
```
231+
Syntax: H2MaxHeaderBlockLen n
232+
Default: H2MaxHeaderBlockLen 0
233+
Context: server config, virtual host
234+
```
235+
H2MaxHeaderBlockLen set the limit on the maximum size of the uncompressed headers of a response. This is the accumulated size of all headers in a response. Setting this to 0 means the nghttp2 default of 64 KB applies.
236+
228237
### H2MaxSessionStreams
229238

230239
```

mod_http2/h2_config.c

+21
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,7 @@ typedef struct h2_config {
7777
int output_buffered;
7878
apr_interval_time_t stream_timeout;/* beam timeout */
7979
int max_data_frame_len; /* max # bytes in a single h2 DATA frame */
80+
int max_hd_block_len; /* max # bytes in a response header block */
8081
int proxy_requests; /* act as forward proxy */
8182
int h2_websockets; /* if mod_h2 negotiating WebSockets */
8283
} h2_config;
@@ -117,6 +118,7 @@ static h2_config defconf = {
117118
1, /* stream output buffered */
118119
-1, /* beam timeout */
119120
0, /* max DATA frame len, 0 == no extra limit */
121+
0, /* max header block len, 0 == no extra limit */
120122
0, /* forward proxy */
121123
0, /* WebSockets negotiation, enabled */
122124
};
@@ -165,6 +167,7 @@ void *h2_config_create_svr(apr_pool_t *pool, server_rec *s)
165167
conf->output_buffered = DEF_VAL;
166168
conf->stream_timeout = DEF_VAL;
167169
conf->max_data_frame_len = DEF_VAL;
170+
conf->max_hd_block_len = DEF_VAL;
168171
conf->proxy_requests = DEF_VAL;
169172
conf->h2_websockets = DEF_VAL;
170173
return conf;
@@ -216,6 +219,7 @@ static void *h2_config_merge(apr_pool_t *pool, void *basev, void *addv)
216219
n->padding_always = H2_CONFIG_GET(add, base, padding_always);
217220
n->stream_timeout = H2_CONFIG_GET(add, base, stream_timeout);
218221
n->max_data_frame_len = H2_CONFIG_GET(add, base, max_data_frame_len);
222+
n->max_hd_block_len = H2_CONFIG_GET(add, base, max_hd_block_len);
219223
n->proxy_requests = H2_CONFIG_GET(add, base, proxy_requests);
220224
n->h2_websockets = H2_CONFIG_GET(add, base, h2_websockets);
221225
return n;
@@ -313,6 +317,8 @@ static apr_int64_t h2_srv_config_geti64(const h2_config *conf, h2_config_var_t v
313317
return H2_CONFIG_GET(conf, &defconf, proxy_requests);
314318
case H2_CONF_WEBSOCKETS:
315319
return H2_CONFIG_GET(conf, &defconf, h2_websockets);
320+
case H2_CONF_MAX_HEADER_BLOCK_LEN:
321+
return H2_CONFIG_GET(conf, &defconf, max_hd_block_len);
316322
default:
317323
return DEF_VAL;
318324
}
@@ -381,6 +387,8 @@ static void h2_srv_config_seti(h2_config *conf, h2_config_var_t var, int val)
381387
case H2_CONF_WEBSOCKETS:
382388
H2_CONFIG_SET(conf, h2_websockets, val);
383389
break;
390+
case H2_CONF_MAX_HEADER_BLOCK_LEN:
391+
H2_CONFIG_SET(conf, max_hd_block_len, val);
384392
default:
385393
break;
386394
}
@@ -650,6 +658,17 @@ static const char *h2_conf_set_max_data_frame_len(cmd_parms *cmd,
650658
return NULL;
651659
}
652660

661+
static const char *h2_conf_set_max_hd_block_len(cmd_parms *cmd,
662+
void *dirconf, const char *value)
663+
{
664+
int val = (int)apr_atoi64(value);
665+
if (val < 0) {
666+
return "value must be 0 or larger";
667+
}
668+
CONFIG_CMD_SET(cmd, dirconf, H2_CONF_MAX_HEADER_BLOCK_LEN, val);
669+
return NULL;
670+
}
671+
653672
static const char *h2_conf_set_session_extra_files(cmd_parms *cmd,
654673
void *dirconf, const char *value)
655674
{
@@ -1071,6 +1090,8 @@ const command_rec h2_cmds[] = {
10711090
RSRC_CONF, "set stream timeout"),
10721091
AP_INIT_TAKE1("H2MaxDataFrameLen", h2_conf_set_max_data_frame_len, NULL,
10731092
RSRC_CONF, "maximum number of bytes in a single HTTP/2 DATA frame"),
1093+
AP_INIT_TAKE1("H2MaxHeaderBlockLen", h2_conf_set_max_hd_block_len, NULL,
1094+
RSRC_CONF, "maximum number of bytes in a response header block"),
10741095
AP_INIT_TAKE2("H2EarlyHint", h2_conf_add_early_hint, NULL,
10751096
OR_FILEINFO|OR_AUTHCFG, "add a a 'Link:' header for a 103 Early Hints response."),
10761097
AP_INIT_TAKE1("H2ProxyRequests", h2_conf_set_proxy_requests, NULL,

mod_http2/h2_config.h

+1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@ typedef enum {
4646
H2_CONF_MAX_DATA_FRAME_LEN,
4747
H2_CONF_PROXY_REQUESTS,
4848
H2_CONF_WEBSOCKETS,
49+
H2_CONF_MAX_HEADER_BLOCK_LEN,
4950
} h2_config_var_t;
5051

5152
struct apr_hash_t;

mod_http2/h2_session.c

+29
Original file line numberDiff line numberDiff line change
@@ -624,6 +624,29 @@ static int on_frame_send_cb(nghttp2_session *ngh2,
624624
return 0;
625625
}
626626

627+
static int on_frame_not_send_cb(nghttp2_session *ngh2,
628+
const nghttp2_frame *frame,
629+
int ngh2_err,
630+
void *user_data)
631+
{
632+
h2_session *session = user_data;
633+
int stream_id = frame->hd.stream_id;
634+
h2_stream *stream;
635+
char buffer[256];
636+
637+
stream = get_stream(session, stream_id);
638+
h2_util_frame_print(frame, buffer, sizeof(buffer)/sizeof(buffer[0]));
639+
ap_log_cerror(APLOG_MARK, APLOG_ERR, 0, session->c1,
640+
H2_SSSN_LOG(APLOGNO(), session,
641+
"not sent FRAME[%s], error %d: %s"),
642+
buffer, ngh2_err, nghttp2_strerror(ngh2_err));
643+
if(stream) {
644+
h2_stream_rst(stream, NGHTTP2_PROTOCOL_ERROR);
645+
return 0;
646+
}
647+
return NGHTTP2_ERR_CALLBACK_FAILURE;
648+
}
649+
627650
#ifdef H2_NG2_INVALID_HEADER_CB
628651
static int on_invalid_header_cb(nghttp2_session *ngh2,
629652
const nghttp2_frame *frame,
@@ -699,6 +722,7 @@ static apr_status_t init_callbacks(conn_rec *c, nghttp2_session_callbacks **pcb)
699722
NGH2_SET_CALLBACK(*pcb, on_header, on_header_cb);
700723
NGH2_SET_CALLBACK(*pcb, send_data, on_send_data_cb);
701724
NGH2_SET_CALLBACK(*pcb, on_frame_send, on_frame_send_cb);
725+
NGH2_SET_CALLBACK(*pcb, on_frame_not_send, on_frame_not_send_cb);
702726
#ifdef H2_NG2_INVALID_HEADER_CB
703727
NGH2_SET_CALLBACK(*pcb, on_invalid_header, on_invalid_header_cb);
704728
#endif
@@ -988,6 +1012,11 @@ apr_status_t h2_session_create(h2_session **psession, conn_rec *c, request_rec *
9881012
* handle them, just like the HTTP/1.1 parser does. */
9891013
nghttp2_option_set_no_rfc9113_leading_and_trailing_ws_validation(options, 1);
9901014
#endif
1015+
1016+
if(h2_config_sgeti(s, H2_CONF_MAX_HEADER_BLOCK_LEN) > 0)
1017+
nghttp2_option_set_max_send_header_block_length(options,
1018+
h2_config_sgeti(s, H2_CONF_MAX_HEADER_BLOCK_LEN));
1019+
9911020
rv = nghttp2_session_server_new2(&session->ngh2, callbacks,
9921021
session, options);
9931022
nghttp2_session_callbacks_del(callbacks);

mod_http2/h2_stream.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -1721,10 +1721,10 @@ static apr_status_t stream_do_response(h2_stream *stream)
17211721
if (nghttp2_is_fatal(ngrv)) {
17221722
rv = APR_EGENERAL;
17231723
h2_session_dispatch_event(stream->session,
1724-
H2_SESSION_EV_PROTO_ERROR, ngrv, nghttp2_strerror(rv));
1724+
H2_SESSION_EV_PROTO_ERROR, ngrv, nghttp2_strerror(ngrv));
17251725
ap_log_cerror(APLOG_MARK, APLOG_ERR, rv, c1,
17261726
APLOGNO(10402) "submit_response: %s",
1727-
nghttp2_strerror(rv));
1727+
nghttp2_strerror(ngrv));
17281728
goto cleanup;
17291729
}
17301730

test/modules/http2/env.py

+3
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,9 @@ def __init__(self, env: HttpdTestEnv, extras: Dict[str, Any] = None):
113113
"<Location \"/h2test/error\">",
114114
" SetHandler h2test-error",
115115
"</Location>",
116+
"<Location \"/h2test/tweak\">",
117+
" SetHandler h2test-tweak",
118+
"</Location>",
116119
]
117120
}))
118121

0 commit comments

Comments
 (0)