// Copyright 2014 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 QuicServer. It listens forever on --port // (default 6121) until it's killed or ctrl-cd to death. #include #include "base/at_exit.h" #include "base/command_line.h" #include "base/logging.h" #include "base/message_loop/message_loop.h" #include "base/run_loop.h" #include "base/strings/string_number_conversions.h" #include "base/task/task_scheduler/task_scheduler.h" #include "net/base/ip_address.h" #include "net/base/ip_endpoint.h" #include "net/quic/crypto/proof_source_chromium.h" #include "net/third_party/quic/core/quic_packets.h" #include "net/third_party/quic/tools/quic_memory_cache_backend.h" #include "net/third_party/quic/tools/quic_simple_server_backend.h" #include "net/tools/quic/quic_http_proxy_backend.h" #include "net/tools/quic/quic_simple_server.h" // The port the quic server will listen on. int32_t FLAGS_port = 6121; // Mode of operations: currently only support in-memory cache std::string FLAGS_quic_mode = "cache"; // Specifies the directory used during QuicHttpResponseCache // construction to seed the cache. Cache directory can be // generated using `wget -p --save-headers ` std::string FLAGS_quic_response_cache_dir = ""; // URL with http/https, IP address or host name and the port number of the // backend server std::string FLAGS_quic_proxy_backend_url = ""; std::unique_ptr CreateProofSource( const base::FilePath& cert_path, const base::FilePath& key_path) { std::unique_ptr proof_source( new net::ProofSourceChromium()); CHECK(proof_source->Initialize(cert_path, key_path, base::FilePath())); return std::move(proof_source); } int main(int argc, char* argv[]) { base::TaskScheduler::CreateAndStartWithDefaultParams("quic_server"); base::AtExitManager exit_manager; base::MessageLoopForIO message_loop; base::CommandLine::Init(argc, argv); base::CommandLine* line = base::CommandLine::ForCurrentProcess(); logging::LoggingSettings settings; settings.logging_dest = logging::LOG_TO_SYSTEM_DEBUG_LOG; CHECK(logging::InitLogging(settings)); if (line->HasSwitch("h") || line->HasSwitch("help")) { const char* help_str = "Usage: quic_server [options]\n" "\n" "Options:\n" "-h, --help show this help message and exit\n" "--port= specify the port to listen on\n" "--mode= Specify mode of operation: Proxy will " "serve response from\n" " a backend server and Cache will serve it " "from a cache dir\n" "--quic_response_cache_dir=\n" " The directory containing cached response " "data to load\n" "--quic_proxy_backend_url=://: \n" " The URL for the single backend server " "hostname \n" " For example, \"http://xyz.com:80\"\n" "--certificate_file= path to the certificate chain\n" "--key_file= path to the pkcs8 private key\n"; std::cout << help_str; exit(0); } // Serve the HTTP response from backend: memory cache or http proxy std::unique_ptr quic_simple_server_backend; if (line->HasSwitch("mode")) { FLAGS_quic_mode = line->GetSwitchValueASCII("mode"); } if (FLAGS_quic_mode.compare("cache") == 0) { if (line->HasSwitch("quic_response_cache_dir")) { FLAGS_quic_response_cache_dir = line->GetSwitchValueASCII("quic_response_cache_dir"); quic_simple_server_backend = std::make_unique(); if (FLAGS_quic_response_cache_dir.empty() || quic_simple_server_backend->InitializeBackend( FLAGS_quic_response_cache_dir) != true) { LOG(ERROR) << "--quic_response_cache_dir is not valid !"; return 1; } } } else if (FLAGS_quic_mode.compare("proxy") == 0) { if (line->HasSwitch("quic_proxy_backend_url")) { FLAGS_quic_proxy_backend_url = line->GetSwitchValueASCII("quic_proxy_backend_url"); quic_simple_server_backend = std::make_unique(); if (quic_simple_server_backend->InitializeBackend( FLAGS_quic_proxy_backend_url) != true) { LOG(ERROR) << "--quic_proxy_backend_url " << FLAGS_quic_proxy_backend_url << " is not valid !"; return 1; } } } else { LOG(ERROR) << "unknown --mode. cache is a valid mode of operation"; return 1; } if (line->HasSwitch("port")) { if (!base::StringToInt(line->GetSwitchValueASCII("port"), &FLAGS_port)) { LOG(ERROR) << "--port must be an integer\n"; return 1; } } if (!line->HasSwitch("certificate_file")) { LOG(ERROR) << "missing --certificate_file"; return 1; } if (!line->HasSwitch("key_file")) { LOG(ERROR) << "missing --key_file"; return 1; } net::IPAddress ip = net::IPAddress::IPv6AllZeros(); quic::QuicConfig config; net::QuicSimpleServer server( CreateProofSource(line->GetSwitchValuePath("certificate_file"), line->GetSwitchValuePath("key_file")), config, quic::QuicCryptoServerConfig::ConfigOptions(), quic::AllSupportedVersions(), quic_simple_server_backend.get()); int rc = server.Listen(net::IPEndPoint(ip, FLAGS_port)); if (rc < 0) { return 1; } base::RunLoop().Run(); return 0; }