|  | // 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 <iostream> | 
|  |  | 
|  | #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 <url>` | 
|  | 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<quic::ProofSource> CreateProofSource( | 
|  | const base::FilePath& cert_path, | 
|  | const base::FilePath& key_path) { | 
|  | std::unique_ptr<net::ProofSourceChromium> 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=<port>               specify the port to listen on\n" | 
|  | "--mode=<cache|proxy>        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=<directory>\n" | 
|  | "                            The directory containing cached response " | 
|  | "data to load\n" | 
|  | "--quic_proxy_backend_url=<http/https>://<hostname_ip>:<port_number> \n" | 
|  | "                            The URL for the single backend server " | 
|  | "hostname \n" | 
|  | "                            For example, \"http://xyz.com:80\"\n" | 
|  | "--certificate_file=<file>   path to the certificate chain\n" | 
|  | "--key_file=<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::QuicSimpleServerBackend> 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<quic::QuicMemoryCacheBackend>(); | 
|  | 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<net::QuicHttpProxyBackend>(); | 
|  | 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; | 
|  | } |