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

139 lines
3.6 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 stream error handling
* Verifies that stream-level errors (RST_STREAM) don't affect other streams
*/
#include <config.h>
#include <stdlib.h> // exit()
#include <string.h> // strcmp()
#include "libtest.h"
int main(void)
{
// Test 1: One stream fails, others succeed
// In HTTP/2, individual stream errors should not affect the connection
// or other streams
wget_test_url_t urls[]={
{ .name = "/file1.txt",
.code = "200 OK",
.body = "Content of file 1",
.headers = {
"Content-Type: text/plain",
NULL
}
},
{ .name = "/error.txt",
.code = "404 Not Found",
.body = "Not Found",
.headers = {
"Content-Type: text/plain",
NULL
}
},
{ .name = "/file3.txt",
.code = "200 OK",
.body = "Content of file 3",
.headers = {
"Content-Type: text/plain",
NULL
}
},
};
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &urls, countof(urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
// Download 3 files where the middle one returns 404
// With --force-html or multiple URLs, wget2 should continue
// and successfully download the other files
wget_test(
WGET_TEST_OPTIONS, "--no-directories",
WGET_TEST_REQUEST_URLS, "file1.txt", "error.txt", "file3.txt", NULL,
WGET_TEST_EXPECTED_ERROR_CODE, 8, // Exit code 8 for server errors
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "file1.txt", "Content of file 1" },
{ "file3.txt", "Content of file 3" },
// error.txt should not be saved
{ NULL }
},
0);
wget_info_printf("HTTP/2 stream error isolation test passed\n");
// Test 2: Multiple errors with successful downloads
wget_test_url_t mixed_urls[]={
{ .name = "/ok1.txt",
.code = "200 OK",
.body = "OK 1",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/err1.txt",
.code = "500 Internal Server Error",
.body = "Error 1",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/ok2.txt",
.code = "200 OK",
.body = "OK 2",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/err2.txt",
.code = "403 Forbidden",
.body = "Error 2",
.headers = { "Content-Type: text/plain", NULL }
},
{ .name = "/ok3.txt",
.code = "200 OK",
.body = "OK 3",
.headers = { "Content-Type: text/plain", NULL }
},
};
wget_test_stop_server();
wget_test_start_server(
WGET_TEST_RESPONSE_URLS, &mixed_urls, countof(mixed_urls),
WGET_TEST_FEATURE_MHD,
WGET_TEST_H2_ONLY,
0);
wget_test(
WGET_TEST_OPTIONS, "--no-directories",
WGET_TEST_REQUEST_URLS, "ok1.txt", "err1.txt", "ok2.txt",
"err2.txt", "ok3.txt", NULL,
WGET_TEST_EXPECTED_ERROR_CODE, 8, // Exit code 8 for server errors
WGET_TEST_EXPECTED_FILES, &(wget_test_file_t []) {
{ "ok1.txt", "OK 1" },
{ "ok2.txt", "OK 2" },
{ "ok3.txt", "OK 3" },
// Error files should not be saved
{ NULL }
},
0);
wget_info_printf("HTTP/2 multiple stream errors test passed\n");
exit(EXIT_SUCCESS);
}