blob: 12a83280c9c1cb95ad250d5e8a717a2bfca71076 [file] [log] [blame]
//
// Copyright (C) 2011 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#include "update_engine/chrome_browser_proxy_resolver.h"
#include <utility>
#include <base/bind.h>
#include <base/memory/ptr_util.h>
#include <base/strings/string_tokenizer.h>
#include <base/strings/string_util.h>
#include "network_proxy/dbus-proxies.h"
namespace chromeos_update_engine {
using base::StringTokenizer;
using std::deque;
using std::string;
namespace {
// Timeout for D-Bus calls in milliseconds.
constexpr int kTimeoutMs = 5000;
} // namespace
ChromeBrowserProxyResolver::ChromeBrowserProxyResolver(
org::chromium::NetworkProxyServiceInterfaceProxyInterface* dbus_proxy)
: dbus_proxy_(dbus_proxy),
next_request_id_(kProxyRequestIdNull + 1),
weak_ptr_factory_(this) {}
ChromeBrowserProxyResolver::~ChromeBrowserProxyResolver() = default;
// static
deque<string> ChromeBrowserProxyResolver::ParseProxyString(
const string& input) {
deque<string> ret;
// Some of this code taken from
// http://src.chromium.org/svn/trunk/src/net/proxy/proxy_server.cc and
// http://src.chromium.org/svn/trunk/src/net/proxy/proxy_list.cc
StringTokenizer entry_tok(input, ";");
while (entry_tok.GetNext()) {
string token = entry_tok.token();
base::TrimWhitespaceASCII(token, base::TRIM_ALL, &token);
// Start by finding the first space (if any).
string::iterator space;
for (space = token.begin(); space != token.end(); ++space) {
if (base::IsAsciiWhitespace(*space)) {
break;
}
}
string scheme = base::ToLowerASCII(string(token.begin(), space));
// Chrome uses "socks" to mean socks4 and "proxy" to mean http.
if (scheme == "socks")
scheme += "4";
else if (scheme == "proxy")
scheme = "http";
else if (scheme != "https" &&
scheme != "socks4" &&
scheme != "socks5" &&
scheme != "direct")
continue; // Invalid proxy scheme
string host_and_port = string(space, token.end());
base::TrimWhitespaceASCII(host_and_port, base::TRIM_ALL, &host_and_port);
if (scheme != "direct" && host_and_port.empty())
continue; // Must supply host/port when non-direct proxy used.
ret.push_back(scheme + "://" + host_and_port);
}
if (ret.empty() || *ret.rbegin() != kNoProxy)
ret.push_back(kNoProxy);
return ret;
}
ProxyRequestId ChromeBrowserProxyResolver::GetProxiesForUrl(
const string& url, const ProxiesResolvedFn& callback) {
const ProxyRequestId id = next_request_id_++;
dbus_proxy_->ResolveProxyAsync(
url,
base::Bind(&ChromeBrowserProxyResolver::OnResolveProxyResponse,
weak_ptr_factory_.GetWeakPtr(), id),
base::Bind(&ChromeBrowserProxyResolver::OnResolveProxyError,
weak_ptr_factory_.GetWeakPtr(), id),
kTimeoutMs);
pending_callbacks_[id] = callback;
return id;
}
bool ChromeBrowserProxyResolver::CancelProxyRequest(ProxyRequestId request) {
return pending_callbacks_.erase(request) != 0;
}
void ChromeBrowserProxyResolver::OnResolveProxyResponse(
ProxyRequestId request_id,
const std::string& proxy_info,
const std::string& error_message) {
if (!error_message.empty())
LOG(WARNING) << "Got error resolving proxy: " << error_message;
RunCallback(request_id, ParseProxyString(proxy_info));
}
void ChromeBrowserProxyResolver::OnResolveProxyError(ProxyRequestId request_id,
brillo::Error* error) {
LOG(WARNING) << "Failed to resolve proxy: "
<< (error ? error->GetMessage() : "[null]");
RunCallback(request_id, deque<string>{kNoProxy});
}
void ChromeBrowserProxyResolver::RunCallback(
ProxyRequestId request_id,
const std::deque<std::string>& proxies) {
auto it = pending_callbacks_.find(request_id);
if (it == pending_callbacks_.end())
return;
ProxiesResolvedFn callback = it->second;
pending_callbacks_.erase(it);
callback.Run(proxies);
}
} // namespace chromeos_update_engine