// Copyright (c) 2010 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. #include "net/base/proxy_server.h" #include #include "base/strings/string_util.h" #include "base/trace_event/memory_usage_estimator.h" #include "net/base/url_util.h" #include "net/http/http_util.h" namespace net { namespace { // Parses the proxy type from a PAC string, to a ProxyServer::Scheme. // This mapping is case-insensitive. If no type could be matched // returns SCHEME_INVALID. ProxyServer::Scheme GetSchemeFromPacTypeInternal(base::StringPiece type) { if (base::LowerCaseEqualsASCII(type, "proxy")) return ProxyServer::SCHEME_HTTP; if (base::LowerCaseEqualsASCII(type, "socks")) { // Default to v4 for compatibility. This is because the SOCKS4 vs SOCKS5 // notation didn't originally exist, so if a client returns SOCKS they // really meant SOCKS4. return ProxyServer::SCHEME_SOCKS4; } if (base::LowerCaseEqualsASCII(type, "socks4")) return ProxyServer::SCHEME_SOCKS4; if (base::LowerCaseEqualsASCII(type, "socks5")) return ProxyServer::SCHEME_SOCKS5; if (base::LowerCaseEqualsASCII(type, "direct")) return ProxyServer::SCHEME_DIRECT; if (base::LowerCaseEqualsASCII(type, "https")) return ProxyServer::SCHEME_HTTPS; if (base::LowerCaseEqualsASCII(type, "quic")) return ProxyServer::SCHEME_QUIC; return ProxyServer::SCHEME_INVALID; } // Parses the proxy scheme from a URL-like representation, to a // ProxyServer::Scheme. This corresponds with the values used in // ProxyServer::ToURI(). If no type could be matched, returns SCHEME_INVALID. ProxyServer::Scheme GetSchemeFromURIInternal(base::StringPiece type) { if (base::LowerCaseEqualsASCII(type, "http")) return ProxyServer::SCHEME_HTTP; if (base::LowerCaseEqualsASCII(type, "socks4")) return ProxyServer::SCHEME_SOCKS4; if (base::LowerCaseEqualsASCII(type, "socks")) return ProxyServer::SCHEME_SOCKS5; if (base::LowerCaseEqualsASCII(type, "socks5")) return ProxyServer::SCHEME_SOCKS5; if (base::LowerCaseEqualsASCII(type, "direct")) return ProxyServer::SCHEME_DIRECT; if (base::LowerCaseEqualsASCII(type, "https")) return ProxyServer::SCHEME_HTTPS; if (base::LowerCaseEqualsASCII(type, "quic")) return ProxyServer::SCHEME_QUIC; return ProxyServer::SCHEME_INVALID; } } // namespace ProxyServer::ProxyServer(Scheme scheme, const HostPortPair& host_port_pair) : scheme_(scheme), host_port_pair_(host_port_pair) { if (scheme_ == SCHEME_DIRECT || scheme_ == SCHEME_INVALID) { // |host_port_pair| isn't relevant for these special schemes, so none should // have been specified. It is important for this to be consistent since we // do raw field comparisons in the equality and comparison functions. DCHECK(host_port_pair.Equals(HostPortPair())); host_port_pair_ = HostPortPair(); } } ProxyServer::ProxyServer(Scheme scheme, const HostPortPair& host_port_pair, bool is_trusted_proxy) : ProxyServer(scheme, host_port_pair) { if (is_trusted_proxy) { is_trusted_proxy_ = true; // TODO(https://crbug.com/778010): Update this when cross-origin server // push is allowed for QUIC proxies. DCHECK_EQ(SCHEME_HTTPS, scheme_); } } const HostPortPair& ProxyServer::host_port_pair() const { // Doesn't make sense to call this if the URI scheme doesn't // have concept of a host. DCHECK(is_valid()); DCHECK(!is_direct()); return host_port_pair_; } // static ProxyServer ProxyServer::FromURI(base::StringPiece uri, Scheme default_scheme) { // We will default to |default_scheme| if no scheme specifier was given. Scheme scheme = default_scheme; // Trim the leading/trailing whitespace. uri = HttpUtil::TrimLWS(uri); // Check for [ "://"] size_t colon = uri.find(':'); if (colon != base::StringPiece::npos && uri.size() - colon >= 3 && uri[colon + 1] == '/' && uri[colon + 2] == '/') { scheme = GetSchemeFromURIInternal(uri.substr(0, colon)); uri = uri.substr(colon + 3); // Skip past the "://" } // Now parse the [":"]. return FromSchemeHostAndPort(scheme, uri); } std::string ProxyServer::ToURI() const { switch (scheme_) { case SCHEME_DIRECT: return "direct://"; case SCHEME_HTTP: // Leave off "http://" since it is our default scheme. return host_port_pair().ToString(); case SCHEME_SOCKS4: return std::string("socks4://") + host_port_pair().ToString(); case SCHEME_SOCKS5: return std::string("socks5://") + host_port_pair().ToString(); case SCHEME_HTTPS: return std::string("https://") + host_port_pair().ToString(); case SCHEME_QUIC: return std::string("quic://") + host_port_pair().ToString(); default: // Got called with an invalid scheme. NOTREACHED(); return std::string(); } } // static ProxyServer ProxyServer::FromPacString(base::StringPiece pac_string) { // Trim the leading/trailing whitespace. pac_string = HttpUtil::TrimLWS(pac_string); // Input should match: // "DIRECT" | ( 1*(LWS) ) // Start by finding the first space (if any). size_t space = 0; for (; space < pac_string.size(); space++) { if (HttpUtil::IsLWS(pac_string[space])) { break; } } // Everything to the left of the space is the scheme. Scheme scheme = GetSchemeFromPacTypeInternal(pac_string.substr(0, space)); // And everything to the right of the space is the // [":" ]. return FromSchemeHostAndPort(scheme, pac_string.substr(space)); } std::string ProxyServer::ToPacString() const { switch (scheme_) { case SCHEME_DIRECT: return "DIRECT"; case SCHEME_HTTP: return std::string("PROXY ") + host_port_pair().ToString(); case SCHEME_SOCKS4: // For compatibility send SOCKS instead of SOCKS4. return std::string("SOCKS ") + host_port_pair().ToString(); case SCHEME_SOCKS5: return std::string("SOCKS5 ") + host_port_pair().ToString(); case SCHEME_HTTPS: return std::string("HTTPS ") + host_port_pair().ToString(); case SCHEME_QUIC: return std::string("QUIC ") + host_port_pair().ToString(); default: // Got called with an invalid scheme. NOTREACHED(); return std::string(); } } // static int ProxyServer::GetDefaultPortForScheme(Scheme scheme) { switch (scheme) { case SCHEME_HTTP: return 80; case SCHEME_SOCKS4: case SCHEME_SOCKS5: return 1080; case SCHEME_HTTPS: case SCHEME_QUIC: return 443; case SCHEME_INVALID: case SCHEME_DIRECT: break; } return -1; } // static ProxyServer::Scheme ProxyServer::GetSchemeFromURI(const std::string& scheme) { return GetSchemeFromURIInternal(scheme); } size_t ProxyServer::EstimateMemoryUsage() const { return base::trace_event::EstimateMemoryUsage(host_port_pair_); } // static ProxyServer ProxyServer::FromSchemeHostAndPort( Scheme scheme, base::StringPiece host_and_port) { // Trim leading/trailing space. host_and_port = HttpUtil::TrimLWS(host_and_port); if (scheme == SCHEME_DIRECT && !host_and_port.empty()) return ProxyServer(); // Invalid -- DIRECT cannot have a host/port. HostPortPair host_port_pair; if (scheme != SCHEME_INVALID && scheme != SCHEME_DIRECT) { std::string host; int port = -1; // If the scheme has a host/port, parse it. bool ok = ParseHostAndPort(host_and_port, &host, &port); if (!ok) return ProxyServer(); // Invalid -- failed parsing [":"] // Choose a default port number if none was given. if (port == -1) port = GetDefaultPortForScheme(scheme); host_port_pair = HostPortPair(host, static_cast(port)); } return ProxyServer(scheme, host_port_pair); } } // namespace net