mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 22:36:09 +03:00
244 lines
7.8 KiB
C++
244 lines
7.8 KiB
C++
|
// 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 <algorithm>
|
||
|
|
||
|
#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 [<scheme> "://"]
|
||
|
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 <host>[":"<port>].
|
||
|
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" | ( <type> 1*(LWS) <host-and-port> )
|
||
|
|
||
|
// 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
|
||
|
// <host>[":" <port>].
|
||
|
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 <host>[":"<port>]
|
||
|
|
||
|
// Choose a default port number if none was given.
|
||
|
if (port == -1)
|
||
|
port = GetDefaultPortForScheme(scheme);
|
||
|
|
||
|
host_port_pair = HostPortPair(host, static_cast<uint16_t>(port));
|
||
|
}
|
||
|
|
||
|
return ProxyServer(scheme, host_port_pair);
|
||
|
}
|
||
|
|
||
|
} // namespace net
|