//
// 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 <deque>
#include <string>
#include <base/bind.h>
#include <base/strings/string_tokenizer.h>
#include <base/strings/string_util.h>
#include "update_engine/common/utils.h"
namespace chromeos_update_engine {
using base::StringTokenizer;
using base::TimeDelta;
using brillo::MessageLoop;
using std::deque;
using std::string;
const char kLibCrosServiceName[] = "org.chromium.LibCrosService";
const char kLibCrosProxyResolveName[] = "ProxyResolved";
const char kLibCrosProxyResolveSignalInterface[] =
"org.chromium.UpdateEngineLibcrosProxyResolvedInterface";
namespace {
const int kTimeout = 5; // seconds
} // namespace
ChromeBrowserProxyResolver::ChromeBrowserProxyResolver(
LibCrosProxy* libcros_proxy)
: libcros_proxy_(libcros_proxy), timeout_(kTimeout) {}
bool ChromeBrowserProxyResolver::Init() {
libcros_proxy_->ue_proxy_resolved_interface()
->RegisterProxyResolvedSignalHandler(
base::Bind(&ChromeBrowserProxyResolver::OnProxyResolvedSignal,
base::Unretained(this)),
base::Bind(&ChromeBrowserProxyResolver::OnSignalConnected,
base::Unretained(this)));
return true;
}
ChromeBrowserProxyResolver::~ChromeBrowserProxyResolver() {
// Kill outstanding timers.
for (const auto& it : callbacks_) {
MessageLoop::current()->CancelTask(it.second->timeout_id);
}
}
ProxyRequestId ChromeBrowserProxyResolver::GetProxiesForUrl(
const string& url, const ProxiesResolvedFn& callback) {
int timeout = timeout_;
brillo::ErrorPtr error;
if (!libcros_proxy_->service_interface_proxy()->ResolveNetworkProxy(
url.c_str(),
kLibCrosProxyResolveSignalInterface,
kLibCrosProxyResolveName,
&error)) {
LOG(WARNING) << "Can't resolve the proxy. Continuing with no proxy.";
timeout = 0;
}
std::unique_ptr<ProxyRequestData> request(new ProxyRequestData());
request->callback = callback;
ProxyRequestId timeout_id = MessageLoop::current()->PostDelayedTask(
FROM_HERE,
base::Bind(&ChromeBrowserProxyResolver::HandleTimeout,
base::Unretained(this),
url,
request.get()),
TimeDelta::FromSeconds(timeout));
request->timeout_id = timeout_id;
callbacks_.emplace(url, std::move(request));
// We re-use the timeout_id from the MessageLoop as the request id.
return timeout_id;
}
bool ChromeBrowserProxyResolver::CancelProxyRequest(ProxyRequestId request) {
// Finding the timeout_id in the callbacks_ structure requires a linear search
// but we expect this operation to not be so frequent and to have just a few
// proxy requests, so this should be fast enough.
for (auto it = callbacks_.begin(); it != callbacks_.end(); ++it) {
if (it->second->timeout_id == request) {
MessageLoop::current()->CancelTask(request);
callbacks_.erase(it);
return true;
}
}
return false;
}
void ChromeBrowserProxyResolver::ProcessUrlResponse(
const string& source_url, const deque<string>& proxies) {
// Call all the occurrences of the |source_url| and erase them.
auto lower_end = callbacks_.lower_bound(source_url);
auto upper_end = callbacks_.upper_bound(source_url);
for (auto it = lower_end; it != upper_end; ++it) {
ProxyRequestData* request = it->second.get();
MessageLoop::current()->CancelTask(request->timeout_id);
request->callback.Run(proxies);
}
callbacks_.erase(lower_end, upper_end);
}
void ChromeBrowserProxyResolver::OnSignalConnected(const string& interface_name,
const string& signal_name,
bool successful) {
if (!successful) {
LOG(ERROR) << "Couldn't connect to the signal " << interface_name << "."
<< signal_name;
}
}
void ChromeBrowserProxyResolver::OnProxyResolvedSignal(
const string& source_url,
const string& proxy_info,
const string& error_message) {
if (!error_message.empty()) {
LOG(WARNING) << "ProxyResolved error: " << error_message;
}
ProcessUrlResponse(source_url, ParseProxyString(proxy_info));
}
void ChromeBrowserProxyResolver::HandleTimeout(string source_url,
ProxyRequestData* request) {
LOG(INFO) << "Timeout handler called. Seems Chrome isn't responding.";
// Mark the timer_id that produced this callback as invalid to prevent
// canceling the timeout callback that already fired.
request->timeout_id = MessageLoop::kTaskIdNull;
deque<string> proxies = {kNoProxy};
ProcessUrlResponse(source_url, proxies);
}
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;
}
} // namespace chromeos_update_engine