// // 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