Files
wget2/tests/test-http2-multiplexing.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

152 lines
4.2 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 multiplexing - concurrent streams over single connection
*/
#include <config.h>
#include <stdlib.h> // exit()
#include <string.h> // strcmp(), memset()
#include "libtest.h"
// 1MB file for testing multiplexing with substantial data transfer
static char large_file[1 * 1024 * 1024];
int main(void)
{
memset(large_file, 'X', sizeof(large_file) - 1);
large_file[sizeof(large_file) - 1] = '\0';
wget_test_url_t urls[]={
{ .name = "/file1.bin",
.code = "200 OK",
.body = large_file,
.headers = {
"Content-Type: application/octet-stream",
NULL
}
},
{ .name = "/file2.bin",
.code = "200 OK",
.body = large_file,
.headers = {
"Content-Type: application/octet-stream",
NULL
}
},
{ .name = "/file3.bin",
.code = "200 OK",
.body = large_file,
.headers = {
"Content-Type: application/octet-stream",
NULL
}
},
};
// Start HTTP/2-only server
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &urls, countof(urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
// Test 1: Download 3 files concurrently
// HTTP/2 multiplexing should download all 3 files over a single connection
// with concurrent streams, which should be faster than sequential
long long start_ms = wget_get_timemillis();
wget_test(
WGET_TEST_REQUEST_URLS, "file1.bin", "file2.bin", "file3.bin", NULL,
WGET_TEST_EXPECTED_ERROR_CODE, 0,
WGET_TEST_H2_MIN_CONCURRENT_STREAMS, 2, // Expect at least 2 concurrent
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "file1.bin", large_file },
{ "file2.bin", large_file },
{ "file3.bin", large_file },
{ NULL }
},
0);
long long elapsed_ms = wget_get_timemillis() - start_ms;
// Sanity check: 3MB over HTTP/2 should complete reasonably quickly
// This is not a strict performance test, just ensuring multiplexing works
if (elapsed_ms > 30000) { // 30 seconds is very generous
wget_error_printf_exit("HTTP/2 multiplexing took too long: %lld ms "
"(expected < 30000 ms)\n", elapsed_ms);
}
wget_info_printf("HTTP/2 multiplexing test passed (%lld ms for 3MB)\n", elapsed_ms);
// Test 2: Download many small files
// This tests connection reuse and stream ID management
wget_test_url_t small_urls[]={
{ .name = "/small1.txt",
.code = "200 OK",
.body = "Content 1",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/small2.txt",
.code = "200 OK",
.body = "Content 2",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/small3.txt",
.code = "200 OK",
.body = "Content 3",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/small4.txt",
.code = "200 OK",
.body = "Content 4",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/small5.txt",
.code = "200 OK",
.body = "Content 5",
.headers = { "Content-Type: text/plain", NULL }
},
};
wget_test_stop_server();
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &small_urls, countof(small_urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
wget_test(
WGET_TEST_REQUEST_URLS, "small1.txt", "small2.txt", "small3.txt",
"small4.txt", "small5.txt", NULL,
WGET_TEST_EXPECTED_ERROR_CODE, 0,
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "small1.txt", "Content 1" },
{ "small2.txt", "Content 2" },
{ "small3.txt", "Content 3" },
{ "small4.txt", "Content 4" },
{ "small5.txt", "Content 5" },
{ NULL }
},
0);
wget_info_printf("HTTP/2 small files test passed\n");
exit(EXIT_SUCCESS);
}