| // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // A binary wrapper for QuicClient. |
| // Connects to a host using QUIC, sends a request to the provided URL, and |
| // displays the response. |
| // |
| // Some usage examples: |
| // |
| // Standard request/response: |
| // quic_client www.google.com |
| // quic_client www.google.com --quiet |
| // quic_client www.google.com --port=443 |
| // |
| // Use a specific version: |
| // quic_client www.google.com --quic_version=23 |
| // |
| // Send a POST instead of a GET: |
| // quic_client www.google.com --body="this is a POST body" |
| // |
| // Append additional headers to the request: |
| // quic_client www.google.com --headers="Header-A: 1234; Header-B: 5678" |
| // |
| // Connect to a host different to the URL being requested: |
| // quic_client mail.google.com --host=www.google.com |
| // |
| // Connect to a specific IP: |
| // IP=`dig www.google.com +short | head -1` |
| // quic_client www.google.com --host=${IP} |
| // |
| // Send repeated requests and change ephemeral port between requests |
| // quic_client www.google.com --num_requests=10 |
| // |
| // Try to connect to a host which does not speak QUIC: |
| // quic_client www.example.com |
| // |
| // This tool is available as a built binary at: |
| // /google/data/ro/teams/quic/tools/quic_client |
| // After submitting changes to this file, you will need to follow the |
| // instructions at go/quic_client_binary_update |
| |
| #include "quiche/quic/tools/quic_toy_client.h" |
| |
| #include <fstream> |
| #include <iostream> |
| #include <memory> |
| #include <string> |
| #include <utility> |
| #include <vector> |
| |
| #include "absl/strings/escaping.h" |
| #include "absl/strings/str_split.h" |
| #include "absl/strings/string_view.h" |
| #include "quiche/quic/core/crypto/quic_client_session_cache.h" |
| #include "quiche/quic/core/quic_packets.h" |
| #include "quiche/quic/core/quic_server_id.h" |
| #include "quiche/quic/core/quic_utils.h" |
| #include "quiche/quic/core/quic_versions.h" |
| #include "quiche/quic/platform/api/quic_default_proof_providers.h" |
| #include "quiche/quic/platform/api/quic_ip_address.h" |
| #include "quiche/quic/platform/api/quic_socket_address.h" |
| #include "quiche/quic/tools/fake_proof_verifier.h" |
| #include "quiche/quic/tools/quic_url.h" |
| #include "quiche/common/platform/api/quiche_command_line_flags.h" |
| #include "quiche/common/quiche_text_utils.h" |
| #include "quiche/spdy/core/http2_header_block.h" |
| |
| namespace { |
| |
| using quiche::QuicheTextUtils; |
| |
| } // namespace |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, host, "", |
| "The IP or hostname to connect to. If not provided, the host " |
| "will be derived from the provided URL."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(int32_t, port, 0, "The port to connect to."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(std::string, ip_version_for_host_lookup, "", |
| "Only used if host address lookup is needed. " |
| "4=ipv4; 6=ipv6; otherwise=don't care."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(std::string, body, "", |
| "If set, send a POST with this body."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, body_hex, "", |
| "If set, contents are converted from hex to ascii, before " |
| "sending as body of a POST. e.g. --body_hex=\"68656c6c6f\""); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, headers, "", |
| "A semicolon separated list of key:value pairs to " |
| "add to request headers."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(bool, quiet, false, |
| "Set to true for a quieter output experience."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, quic_version, "", |
| "QUIC version to speak, e.g. 21. If not set, then all available " |
| "versions are offered in the handshake. Also supports wire versions " |
| "such as Q043 or T099."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, connection_options, "", |
| "Connection options as ASCII tags separated by commas, " |
| "e.g. \"ABCD,EFGH\""); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, client_connection_options, "", |
| "Client connection options as ASCII tags separated by commas, " |
| "e.g. \"ABCD,EFGH\""); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(bool, quic_ietf_draft, false, |
| "Use the IETF draft version. This also enables " |
| "required internal QUIC flags."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, version_mismatch_ok, false, |
| "If true, a version mismatch in the handshake is not considered a " |
| "failure. Useful for probing a server to determine if it speaks " |
| "any version of QUIC."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, force_version_negotiation, false, |
| "If true, start by proposing a version that is reserved for version " |
| "negotiation."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, multi_packet_chlo, false, |
| "If true, add a transport parameter to make the ClientHello span two " |
| "packets. Only works with QUIC+TLS."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, redirect_is_success, true, |
| "If true, an HTTP response code of 3xx is considered to be a " |
| "successful response, otherwise a failure."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(int32_t, initial_mtu, 0, |
| "Initial MTU of the connection."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| int32_t, num_requests, 1, |
| "How many sequential requests to make on a single connection."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, disable_certificate_verification, false, |
| "If true, don't verify the server certificate."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, default_client_cert, "", |
| "The path to the file containing PEM-encoded client default certificate to " |
| "be sent to the server, if server requested client certs."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| std::string, default_client_cert_key, "", |
| "The path to the file containing PEM-encoded private key of the client's " |
| "default certificate for signing, if server requested client certs."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, drop_response_body, false, |
| "If true, drop response body immediately after it is received."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| bool, disable_port_changes, false, |
| "If true, do not change local port after each request."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(bool, one_connection_per_request, false, |
| "If true, close the connection after each " |
| "request. This allows testing 0-RTT."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(int32_t, server_connection_id_length, -1, |
| "Length of the server connection ID used."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(int32_t, client_connection_id_length, -1, |
| "Length of the client connection ID used."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(int32_t, max_time_before_crypto_handshake_ms, |
| 10000, |
| "Max time to wait before handshake completes."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG( |
| int32_t, max_inbound_header_list_size, 128 * 1024, |
| "Max inbound header list size. 0 means default."); |
| |
| DEFINE_QUICHE_COMMAND_LINE_FLAG(std::string, interface_name, "", |
| "Interface name to bind QUIC UDP sockets to."); |
| |
| namespace quic { |
| namespace { |
| |
| // Creates a ClientProofSource which only contains a default client certificate. |
| // Return nullptr for failure. |
| std::unique_ptr<ClientProofSource> CreateTestClientProofSource( |
| absl::string_view default_client_cert_file, |
| absl::string_view default_client_cert_key_file) { |
| std::ifstream cert_stream(std::string{default_client_cert_file}, |
| std::ios::binary); |
| std::vector<std::string> certs = |
| CertificateView::LoadPemFromStream(&cert_stream); |
| if (certs.empty()) { |
| std::cerr << "Failed to load client certs." << std::endl; |
| return nullptr; |
| } |
| |
| std::ifstream key_stream(std::string{default_client_cert_key_file}, |
| std::ios::binary); |
| std::unique_ptr<CertificatePrivateKey> private_key = |
| CertificatePrivateKey::LoadPemFromStream(&key_stream); |
| if (private_key == nullptr) { |
| std::cerr << "Failed to load client cert key." << std::endl; |
| return nullptr; |
| } |
| |
| auto proof_source = std::make_unique<DefaultClientProofSource>(); |
| proof_source->AddCertAndKey( |
| {"*"}, |
| quiche::QuicheReferenceCountedPointer<ClientProofSource::Chain>( |
| new ClientProofSource::Chain(certs)), |
| std::move(*private_key)); |
| |
| return proof_source; |
| } |
| |
| } // namespace |
| |
| QuicToyClient::QuicToyClient(ClientFactory* client_factory) |
| : client_factory_(client_factory) {} |
| |
| int QuicToyClient::SendRequestsAndPrintResponses( |
| std::vector<std::string> urls) { |
| QuicUrl url(urls[0], "https"); |
| std::string host = quiche::GetQuicheCommandLineFlag(FLAGS_host); |
| if (host.empty()) { |
| host = url.host(); |
| } |
| int port = quiche::GetQuicheCommandLineFlag(FLAGS_port); |
| if (port == 0) { |
| port = url.port(); |
| } |
| |
| quic::ParsedQuicVersionVector versions = quic::CurrentSupportedVersions(); |
| |
| if (quiche::GetQuicheCommandLineFlag(FLAGS_quic_ietf_draft)) { |
| quic::QuicVersionInitializeSupportForIetfDraft(); |
| versions = {}; |
| for (const ParsedQuicVersion& version : AllSupportedVersions()) { |
| if (version.HasIetfQuicFrames() && |
| version.handshake_protocol == quic::PROTOCOL_TLS1_3) { |
| versions.push_back(version); |
| } |
| } |
| } |
| |
| std::string quic_version_string = |
| quiche::GetQuicheCommandLineFlag(FLAGS_quic_version); |
| if (!quic_version_string.empty()) { |
| versions = quic::ParseQuicVersionVectorString(quic_version_string); |
| } |
| |
| if (versions.empty()) { |
| std::cerr << "No known version selected." << std::endl; |
| return 1; |
| } |
| |
| for (const quic::ParsedQuicVersion& version : versions) { |
| quic::QuicEnableVersion(version); |
| } |
| |
| if (quiche::GetQuicheCommandLineFlag(FLAGS_force_version_negotiation)) { |
| versions.insert(versions.begin(), |
| quic::QuicVersionReservedForNegotiation()); |
| } |
| |
| const int32_t num_requests( |
| quiche::GetQuicheCommandLineFlag(FLAGS_num_requests)); |
| std::unique_ptr<quic::ProofVerifier> proof_verifier; |
| if (quiche::GetQuicheCommandLineFlag( |
| FLAGS_disable_certificate_verification)) { |
| proof_verifier = std::make_unique<FakeProofVerifier>(); |
| } else { |
| proof_verifier = quic::CreateDefaultProofVerifier(url.host()); |
| } |
| std::unique_ptr<quic::SessionCache> session_cache; |
| if (num_requests > 1 && |
| quiche::GetQuicheCommandLineFlag(FLAGS_one_connection_per_request)) { |
| session_cache = std::make_unique<QuicClientSessionCache>(); |
| } |
| |
| QuicConfig config; |
| std::string connection_options_string = |
| quiche::GetQuicheCommandLineFlag(FLAGS_connection_options); |
| if (!connection_options_string.empty()) { |
| config.SetConnectionOptionsToSend( |
| ParseQuicTagVector(connection_options_string)); |
| } |
| std::string client_connection_options_string = |
| quiche::GetQuicheCommandLineFlag(FLAGS_client_connection_options); |
| if (!client_connection_options_string.empty()) { |
| config.SetClientConnectionOptions( |
| ParseQuicTagVector(client_connection_options_string)); |
| } |
| if (quiche::GetQuicheCommandLineFlag(FLAGS_multi_packet_chlo)) { |
| // Make the ClientHello span multiple packets by adding a custom transport |
| // parameter. |
| constexpr auto kCustomParameter = |
| static_cast<TransportParameters::TransportParameterId>(0x173E); |
| std::string custom_value(2000, '?'); |
| config.custom_transport_parameters_to_send()[kCustomParameter] = |
| custom_value; |
| } |
| config.set_max_time_before_crypto_handshake( |
| QuicTime::Delta::FromMilliseconds(quiche::GetQuicheCommandLineFlag( |
| FLAGS_max_time_before_crypto_handshake_ms))); |
| |
| int address_family_for_lookup = AF_UNSPEC; |
| if (quiche::GetQuicheCommandLineFlag(FLAGS_ip_version_for_host_lookup) == |
| "4") { |
| address_family_for_lookup = AF_INET; |
| } else if (quiche::GetQuicheCommandLineFlag( |
| FLAGS_ip_version_for_host_lookup) == "6") { |
| address_family_for_lookup = AF_INET6; |
| } |
| |
| // Build the client, and try to connect. |
| std::unique_ptr<QuicSpdyClientBase> client = client_factory_->CreateClient( |
| url.host(), host, address_family_for_lookup, port, versions, config, |
| std::move(proof_verifier), std::move(session_cache)); |
| |
| if (client == nullptr) { |
| std::cerr << "Failed to create client." << std::endl; |
| return 1; |
| } |
| |
| if (!quiche::GetQuicheCommandLineFlag(FLAGS_default_client_cert).empty() && |
| !quiche::GetQuicheCommandLineFlag(FLAGS_default_client_cert_key) |
| .empty()) { |
| std::unique_ptr<ClientProofSource> proof_source = |
| CreateTestClientProofSource( |
| quiche::GetQuicheCommandLineFlag(FLAGS_default_client_cert), |
| quiche::GetQuicheCommandLineFlag(FLAGS_default_client_cert_key)); |
| if (proof_source == nullptr) { |
| std::cerr << "Failed to create client proof source." << std::endl; |
| return 1; |
| } |
| client->crypto_config()->set_proof_source(std::move(proof_source)); |
| } |
| |
| int32_t initial_mtu = quiche::GetQuicheCommandLineFlag(FLAGS_initial_mtu); |
| client->set_initial_max_packet_length( |
| initial_mtu != 0 ? initial_mtu : quic::kDefaultMaxPacketSize); |
| client->set_drop_response_body( |
| quiche::GetQuicheCommandLineFlag(FLAGS_drop_response_body)); |
| const int32_t server_connection_id_length = |
| quiche::GetQuicheCommandLineFlag(FLAGS_server_connection_id_length); |
| if (server_connection_id_length >= 0) { |
| client->set_server_connection_id_length(server_connection_id_length); |
| } |
| const int32_t client_connection_id_length = |
| quiche::GetQuicheCommandLineFlag(FLAGS_client_connection_id_length); |
| if (client_connection_id_length >= 0) { |
| client->set_client_connection_id_length(client_connection_id_length); |
| } |
| const size_t max_inbound_header_list_size = |
| quiche::GetQuicheCommandLineFlag(FLAGS_max_inbound_header_list_size); |
| if (max_inbound_header_list_size > 0) { |
| client->set_max_inbound_header_list_size(max_inbound_header_list_size); |
| } |
| const std::string interface_name = |
| quiche::GetQuicheCommandLineFlag(FLAGS_interface_name); |
| if (!interface_name.empty()) { |
| client->set_interface_name(interface_name); |
| } |
| if (!client->Initialize()) { |
| std::cerr << "Failed to initialize client." << std::endl; |
| return 1; |
| } |
| if (!client->Connect()) { |
| quic::QuicErrorCode error = client->session()->error(); |
| if (error == quic::QUIC_INVALID_VERSION) { |
| std::cerr << "Failed to negotiate version with " << host << ":" << port |
| << ". " << client->session()->error_details() << std::endl; |
| // 0: No error. |
| // 20: Failed to connect due to QUIC_INVALID_VERSION. |
| return quiche::GetQuicheCommandLineFlag(FLAGS_version_mismatch_ok) ? 0 |
| : 20; |
| } |
| std::cerr << "Failed to connect to " << host << ":" << port << ". " |
| << quic::QuicErrorCodeToString(error) << " " |
| << client->session()->error_details() << std::endl; |
| return 1; |
| } |
| std::cerr << "Connected to " << host << ":" << port << std::endl; |
| |
| // Construct the string body from flags, if provided. |
| std::string body = quiche::GetQuicheCommandLineFlag(FLAGS_body); |
| if (!quiche::GetQuicheCommandLineFlag(FLAGS_body_hex).empty()) { |
| QUICHE_DCHECK(quiche::GetQuicheCommandLineFlag(FLAGS_body).empty()) |
| << "Only set one of --body and --body_hex."; |
| body = absl::HexStringToBytes( |
| quiche::GetQuicheCommandLineFlag(FLAGS_body_hex)); |
| } |
| |
| // Construct a GET or POST request for supplied URL. |
| spdy::Http2HeaderBlock header_block; |
| header_block[":method"] = body.empty() ? "GET" : "POST"; |
| header_block[":scheme"] = url.scheme(); |
| header_block[":authority"] = url.HostPort(); |
| header_block[":path"] = url.PathParamsQuery(); |
| |
| // Append any additional headers supplied on the command line. |
| const std::string headers = quiche::GetQuicheCommandLineFlag(FLAGS_headers); |
| for (absl::string_view sp : absl::StrSplit(headers, ';')) { |
| QuicheTextUtils::RemoveLeadingAndTrailingWhitespace(&sp); |
| if (sp.empty()) { |
| continue; |
| } |
| std::vector<absl::string_view> kv = |
| absl::StrSplit(sp, absl::MaxSplits(':', 1)); |
| QuicheTextUtils::RemoveLeadingAndTrailingWhitespace(&kv[0]); |
| QuicheTextUtils::RemoveLeadingAndTrailingWhitespace(&kv[1]); |
| header_block[kv[0]] = kv[1]; |
| } |
| |
| // Make sure to store the response, for later output. |
| client->set_store_response(true); |
| |
| for (int i = 0; i < num_requests; ++i) { |
| // Send the request. |
| client->SendRequestAndWaitForResponse(header_block, body, /*fin=*/true); |
| |
| // Print request and response details. |
| if (!quiche::GetQuicheCommandLineFlag(FLAGS_quiet)) { |
| std::cout << "Request:" << std::endl; |
| std::cout << "headers:" << header_block.DebugString(); |
| if (!quiche::GetQuicheCommandLineFlag(FLAGS_body_hex).empty()) { |
| // Print the user provided hex, rather than binary body. |
| std::cout << "body:\n" |
| << QuicheTextUtils::HexDump(absl::HexStringToBytes( |
| quiche::GetQuicheCommandLineFlag(FLAGS_body_hex))) |
| << std::endl; |
| } else { |
| std::cout << "body: " << body << std::endl; |
| } |
| std::cout << std::endl; |
| |
| if (!client->preliminary_response_headers().empty()) { |
| std::cout << "Preliminary response headers: " |
| << client->preliminary_response_headers() << std::endl; |
| std::cout << std::endl; |
| } |
| |
| std::cout << "Response:" << std::endl; |
| std::cout << "headers: " << client->latest_response_headers() |
| << std::endl; |
| std::string response_body = client->latest_response_body(); |
| if (!quiche::GetQuicheCommandLineFlag(FLAGS_body_hex).empty()) { |
| // Assume response is binary data. |
| std::cout << "body:\n" |
| << QuicheTextUtils::HexDump(response_body) << std::endl; |
| } else { |
| std::cout << "body: " << response_body << std::endl; |
| } |
| std::cout << "trailers: " << client->latest_response_trailers() |
| << std::endl; |
| std::cout << "early data accepted: " << client->EarlyDataAccepted() |
| << std::endl; |
| } |
| |
| if (!client->connected()) { |
| std::cerr << "Request caused connection failure. Error: " |
| << quic::QuicErrorCodeToString(client->session()->error()) |
| << std::endl; |
| return 1; |
| } |
| |
| int response_code = client->latest_response_code(); |
| if (response_code >= 200 && response_code < 300) { |
| std::cout << "Request succeeded (" << response_code << ")." << std::endl; |
| } else if (response_code >= 300 && response_code < 400) { |
| if (quiche::GetQuicheCommandLineFlag(FLAGS_redirect_is_success)) { |
| std::cout << "Request succeeded (redirect " << response_code << ")." |
| << std::endl; |
| } else { |
| std::cout << "Request failed (redirect " << response_code << ")." |
| << std::endl; |
| return 1; |
| } |
| } else { |
| std::cout << "Request failed (" << response_code << ")." << std::endl; |
| return 1; |
| } |
| |
| if (i + 1 < num_requests) { // There are more requests to perform. |
| if (quiche::GetQuicheCommandLineFlag(FLAGS_one_connection_per_request)) { |
| std::cout << "Disconnecting client between requests." << std::endl; |
| client->Disconnect(); |
| if (!client->Initialize()) { |
| std::cerr << "Failed to reinitialize client between requests." |
| << std::endl; |
| return 1; |
| } |
| if (!client->Connect()) { |
| std::cerr << "Failed to reconnect client between requests." |
| << std::endl; |
| return 1; |
| } |
| } else if (!quiche::GetQuicheCommandLineFlag( |
| FLAGS_disable_port_changes)) { |
| // Change the ephemeral port. |
| if (!client->ChangeEphemeralPort()) { |
| std::cerr << "Failed to change ephemeral port." << std::endl; |
| return 1; |
| } |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| } // namespace quic |