// // 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 <map> #include <string> #include <utility> #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::make_pair; using std::pair; 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 (auto& timer : timers_) { MessageLoop::current()->CancelTask(timer.second); timer.second = MessageLoop::kTaskIdNull; } } bool ChromeBrowserProxyResolver::GetProxiesForUrl(const string& url, ProxiesResolvedFn callback, void* data) { 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; } callbacks_.insert(make_pair(url, make_pair(callback, data))); MessageLoop::TaskId timer = MessageLoop::current()->PostDelayedTask( FROM_HERE, base::Bind(&ChromeBrowserProxyResolver::HandleTimeout, base::Unretained(this), url), TimeDelta::FromSeconds(timeout)); timers_.insert(make_pair(url, timer)); return true; } bool ChromeBrowserProxyResolver::DeleteUrlState( const string& source_url, bool delete_timer, pair<ProxiesResolvedFn, void*>* callback) { { CallbacksMap::iterator it = callbacks_.lower_bound(source_url); TEST_AND_RETURN_FALSE(it != callbacks_.end()); TEST_AND_RETURN_FALSE(it->first == source_url); if (callback) *callback = it->second; callbacks_.erase(it); } { TimeoutsMap::iterator it = timers_.lower_bound(source_url); TEST_AND_RETURN_FALSE(it != timers_.end()); TEST_AND_RETURN_FALSE(it->first == source_url); if (delete_timer) MessageLoop::current()->CancelTask(it->second); timers_.erase(it); } return true; } 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) { pair<ProxiesResolvedFn, void*> callback; TEST_AND_RETURN(DeleteUrlState(source_url, true, &callback)); if (!error_message.empty()) { LOG(WARNING) << "ProxyResolved error: " << error_message; } (*callback.first)(ParseProxyString(proxy_info), callback.second); } void ChromeBrowserProxyResolver::HandleTimeout(string source_url) { LOG(INFO) << "Timeout handler called. Seems Chrome isn't responding."; pair<ProxiesResolvedFn, void*> callback; TEST_AND_RETURN(DeleteUrlState(source_url, false, &callback)); deque<string> proxies; proxies.push_back(kNoProxy); (*callback.first)(proxies, callback.second); } 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