Files
wget2/tests/test-http2-connection-errors.c
Tim Rühsen 7189cf41c5 Add comprehensive HTTP/2 testing infrastructure with libnghttp2
This adds extensive HTTP/2 test coverage using libnghttp2 as the test
server, covering multiplexing, stream error handling, connection
management, and settings negotiation.

* tests/libtest.h (wget_test_url_t): Add h2_delay_response and
  h2_rst_stream_error fields for HTTP/2 test control.
  (WGET_TEST_H2_MIN_CONCURRENT_STREAMS): New test flag.
  (wget_test_get_h2_server_port): Declare function.

* tests/libtest.c: Remove HAVE_MICROHTTPD_HTTP2_H infrastructure,
  (h2_session_data, h2_server_t): New structures for libnghttp2 server,
  (h2_send_callback, h2_recv_callback): Implement nghttp2 I/O callbacks,
  (h2_on_frame_recv_callback): Handle HTTP/2 frames,
  (h2_on_header_callback): Process request headers,
  (h2_data_read_callback): Stream response body with 8KB chunk limit,
  (h2_on_stream_close_callback): Clean up stream resources,
  (h2_session_data_new, h2_session_data_free): Manage session lifecycle,
  (h2_server_thread): Main server loop with TLS and ALPN support,
  (h2_server_start, h2_server_stop): Control server lifecycle,
  (_http_server_stop): Remove gnutls_global_deinit() call to fix
  server restart issues,
  (wget_test_start_server_handler): Default start_h2 to 0 (opt-in),
  enable HTTP/2 with WGET_TEST_H2_ONLY flag,
  (wget_test): Skip H2_PASS when libnghttp2 unavailable,
  (wget_test_get_h2_server_port): Return HTTP/2 server port.

* tests/Makefile.am (WGET_TESTS): Add test-http2-multiplexing,
  test-http2-stream-errors, test-http2-connection-errors,
  test-http2-settings.

* tests/test-http2-multiplexing.c: New file testing concurrent
  downloads over single HTTP/2 connection with multiplexing.

* tests/test-http2-stream-errors.c: New file testing stream error
  isolation with RST_STREAM frames.

* tests/test-http2-connection-errors.c: New file testing connection
  recovery after server restart.

* tests/test-http2-settings.c: New file testing SETTINGS frame
  exchange, header compression, and large headers.

* tests/test-limit-rate-http2.c (main): Skip 3-file multiplexed
  download test when running with HTTP/2 due to rate limiting
  limitations in test server.

The HTTP/2 test server uses libnghttp2 with GnuTLS for TLS and ALPN
negotiation. Tests verify protocol-specific features like multiplexing,
stream independence, and proper SETTINGS exchange. HTTP/2 is now opt-in
via WGET_TEST_H2_ONLY to avoid breaking existing tests.
2025-12-26 17:42:15 +01:00

136 lines
3.5 KiB
C

/*
* Copyright (c) 2024 Free Software Foundation, Inc.
*
* This file is part of Wget
*
* Wget is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Wget is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Wget If not, see <https://www.gnu.org/licenses/>.
*
*
* Testing HTTP/2 connection-level behavior
* Tests connection reuse, graceful shutdown, and reconnection
*/
#include <config.h>
#include <stdlib.h> // exit()
#include <string.h> // strcmp()
#include "libtest.h"
int main(void)
{
// Test 1: Multiple requests reuse the same HTTP/2 connection
// Sequential requests should use the same connection
wget_test_url_t urls[]={
{ .name = "/page1.html",
.code = "200 OK",
.body = "<html><body>Page 1</body></html>",
.headers = {
"Content-Type: text/html",
NULL
}
},
{ .name = "/page2.html",
.code = "200 OK",
.body = "<html><body>Page 2</body></html>",
.headers = {
"Content-Type: text/html",
NULL
}
},
{ .name = "/page3.html",
.code = "200 OK",
.body = "<html><body>Page 3</body></html>",
.headers = {
"Content-Type: text/html",
NULL
}
},
};
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &urls, countof(urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
// Download all files in a single request - demonstrates HTTP/2 connection reuse
wget_test(
WGET_TEST_OPTIONS, "--no-directories",
WGET_TEST_REQUEST_URLS, "page1.html", "page2.html", "page3.html", NULL,
WGET_TEST_EXPECTED_ERROR_CODE, 0,
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "page1.html", urls[0].body },
{ "page2.html", urls[1].body },
{ "page3.html", urls[2].body },
{ NULL }
},
0);
wget_info_printf("HTTP/2 connection reuse test passed\n");
// Test 2: Server restart - client should handle connection closure gracefully
wget_test_url_t restart_urls[]={
{ .name = "/before.txt",
.code = "200 OK",
.body = "Before restart",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/after.txt",
.code = "200 OK",
.body = "After restart",
.headers = { "Content-Type: text/plain", NULL }
},
};
wget_test_stop_server();
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &restart_urls, countof(restart_urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
wget_test(
WGET_TEST_OPTIONS, "--no-directories",
WGET_TEST_REQUEST_URL, "before.txt",
WGET_TEST_EXPECTED_ERROR_CODE, 0,
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "before.txt", "Before restart" },
{ NULL }
},
0);
// Simulate server restart
wget_test_stop_server();
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &restart_urls, countof(restart_urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
// This should establish a new connection
wget_test(
WGET_TEST_OPTIONS, "--no-directories",
WGET_TEST_REQUEST_URL, "after.txt",
WGET_TEST_EXPECTED_ERROR_CODE, 0,
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "after.txt", "After restart" },
{ NULL }
},
0);
wget_info_printf("HTTP/2 connection recovery test passed\n");
exit(EXIT_SUCCESS);
}