blob: fbeb71fa2ec68fa98642e5498cc96a4d3412fdb0 [file] [log] [blame]
Andrew de los Reyes000d8952011-03-02 15:21:14 -08001// Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "update_engine/chrome_browser_proxy_resolver.h"
6
Alex Vakulenkod2779df2014-06-16 13:19:00 -07007#include <deque>
Andrew de los Reyes000d8952011-03-02 15:21:14 -08008#include <map>
9#include <string>
Alex Vakulenkod2779df2014-06-16 13:19:00 -070010#include <utility>
Andrew de los Reyes000d8952011-03-02 15:21:14 -080011
Chris Sosafc661a12013-02-26 14:43:21 -080012#include <base/strings/string_tokenizer.h>
Alex Deymoc4acdf42014-05-28 21:07:10 -070013#include <base/strings/string_util.h>
Andrew de los Reyes000d8952011-03-02 15:21:14 -080014#include <dbus/dbus-glib-lowlevel.h>
Alex Deymoc4acdf42014-05-28 21:07:10 -070015#include <dbus/dbus-glib.h>
Andrew de los Reyes000d8952011-03-02 15:21:14 -080016#include <google/protobuf/stubs/common.h>
17
18#include "update_engine/dbus_constants.h"
19#include "update_engine/utils.h"
20
21namespace chromeos_update_engine {
22
Chris Sosafc661a12013-02-26 14:43:21 -080023using base::StringTokenizer;
Andrew de los Reyes000d8952011-03-02 15:21:14 -080024using google::protobuf::Closure;
Alex Deymoc4acdf42014-05-28 21:07:10 -070025using google::protobuf::NewPermanentCallback;
Andrew de los Reyes000d8952011-03-02 15:21:14 -080026using std::deque;
27using std::make_pair;
28using std::multimap;
29using std::pair;
30using std::string;
31
32#define LIB_CROS_PROXY_RESOLVE_NAME "ProxyResolved"
33#define LIB_CROS_PROXY_RESOLVE_SIGNAL_INTERFACE \
34 "org.chromium.UpdateEngineLibcrosProxyResolvedInterface"
35const char kLibCrosServiceName[] = "org.chromium.LibCrosService";
36const char kLibCrosServicePath[] = "/org/chromium/LibCrosService";
37const char kLibCrosServiceInterface[] = "org.chromium.LibCrosServiceInterface";
38const char kLibCrosServiceResolveNetworkProxyMethodName[] =
39 "ResolveNetworkProxy";
40const char kLibCrosProxyResolveName[] = LIB_CROS_PROXY_RESOLVE_NAME;
41const char kLibCrosProxyResolveSignalInterface[] =
42 LIB_CROS_PROXY_RESOLVE_SIGNAL_INTERFACE;
43const char kLibCrosProxyResolveSignalFilter[] = "type='signal', "
44 "interface='" LIB_CROS_PROXY_RESOLVE_SIGNAL_INTERFACE "', "
Andrew de los Reyes000d8952011-03-02 15:21:14 -080045 "member='" LIB_CROS_PROXY_RESOLVE_NAME "'";
46#undef LIB_CROS_PROXY_RESOLVE_SIGNAL_INTERFACE
47#undef LIB_CROS_PROXY_RESOLVE_NAME
48
49namespace {
Gilad Arnoldb752fb32014-03-03 12:23:39 -080050
Andrew de los Reyes000d8952011-03-02 15:21:14 -080051const int kTimeout = 5; // seconds
52
Alex Vakulenkod2779df2014-06-16 13:19:00 -070053} // namespace
Andrew de los Reyes000d8952011-03-02 15:21:14 -080054
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -080055ChromeBrowserProxyResolver::ChromeBrowserProxyResolver(
56 DBusWrapperInterface* dbus)
Andrew de los Reyes000d8952011-03-02 15:21:14 -080057 : dbus_(dbus), proxy_(NULL), timeout_(kTimeout) {}
58
59bool ChromeBrowserProxyResolver::Init() {
Gilad Arnoldb752fb32014-03-03 12:23:39 -080060 if (proxy_)
61 return true; // Already initialized.
62
63 // Set up signal handler. Code lifted from libcros.
64 GError* g_error = NULL;
65 DBusGConnection* bus = dbus_->BusGet(DBUS_BUS_SYSTEM, &g_error);
66 TEST_AND_RETURN_FALSE(bus);
67 DBusConnection* connection = dbus_->ConnectionGetConnection(bus);
Andrew de los Reyes000d8952011-03-02 15:21:14 -080068 TEST_AND_RETURN_FALSE(connection);
Gilad Arnoldb752fb32014-03-03 12:23:39 -080069
70 DBusError dbus_error;
71 dbus_error_init(&dbus_error);
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -080072 dbus_->DBusBusAddMatch(connection, kLibCrosProxyResolveSignalFilter,
Gilad Arnoldb752fb32014-03-03 12:23:39 -080073 &dbus_error);
74 TEST_AND_RETURN_FALSE(!dbus_error_is_set(&dbus_error));
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -080075 TEST_AND_RETURN_FALSE(dbus_->DBusConnectionAddFilter(
Andrew de los Reyes000d8952011-03-02 15:21:14 -080076 connection,
77 &ChromeBrowserProxyResolver::StaticFilterMessage,
78 this,
79 NULL));
80
Gilad Arnoldb752fb32014-03-03 12:23:39 -080081 proxy_ = dbus_->ProxyNewForName(bus, kLibCrosServiceName, kLibCrosServicePath,
82 kLibCrosServiceInterface);
Andrew de los Reyes000d8952011-03-02 15:21:14 -080083 if (!proxy_) {
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -080084 dbus_->DBusConnectionRemoveFilter(
Andrew de los Reyes000d8952011-03-02 15:21:14 -080085 connection,
86 &ChromeBrowserProxyResolver::StaticFilterMessage,
87 this);
88 }
89 TEST_AND_RETURN_FALSE(proxy_); // For the error log
90 return true;
91}
92
93ChromeBrowserProxyResolver::~ChromeBrowserProxyResolver() {
Gilad Arnoldb752fb32014-03-03 12:23:39 -080094 // Remove DBus connection filters and Kill proxy object.
95 if (proxy_) {
96 GError* gerror = NULL;
97 DBusGConnection* gbus = dbus_->BusGet(DBUS_BUS_SYSTEM, &gerror);
98 if (gbus) {
99 DBusConnection* connection = dbus_->ConnectionGetConnection(gbus);
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -0800100 dbus_->DBusConnectionRemoveFilter(
Gilad Arnoldb752fb32014-03-03 12:23:39 -0800101 connection,
102 &ChromeBrowserProxyResolver::StaticFilterMessage,
103 this);
104 }
105 dbus_->ProxyUnref(proxy_);
106 }
107
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800108 // Kill outstanding timers
109 for (TimeoutsMap::iterator it = timers_.begin(), e = timers_.end(); it != e;
110 ++it) {
111 g_source_destroy(it->second);
112 it->second = NULL;
113 }
114}
115
116bool ChromeBrowserProxyResolver::GetProxiesForUrl(const string& url,
117 ProxiesResolvedFn callback,
118 void* data) {
119 GError* error = NULL;
Andrew de los Reyes518502a2011-03-14 14:19:39 -0700120 guint timeout = timeout_;
Gilad Arnold1877c392012-02-10 11:34:33 -0800121 if (proxy_) {
Gilad Arnoldb752fb32014-03-03 12:23:39 -0800122 if (!dbus_->ProxyCall_3_0(proxy_,
123 kLibCrosServiceResolveNetworkProxyMethodName,
124 &error,
125 url.c_str(),
126 kLibCrosProxyResolveSignalInterface,
127 kLibCrosProxyResolveName)) {
Gilad Arnoldb752fb32014-03-03 12:23:39 -0800128 if (error) {
129 LOG(WARNING) << "dbus_g_proxy_call failed, continuing with no proxy: "
130 << utils::GetAndFreeGError(&error);
Gilad Arnold1877c392012-02-10 11:34:33 -0800131 } else {
Gilad Arnoldb752fb32014-03-03 12:23:39 -0800132 LOG(WARNING) << "dbus_g_proxy_call failed with no error string, "
133 "continuing with no proxy.";
Gilad Arnold1877c392012-02-10 11:34:33 -0800134 }
Gilad Arnoldb752fb32014-03-03 12:23:39 -0800135 timeout = 0;
136 }
Gilad Arnold1877c392012-02-10 11:34:33 -0800137 } else {
Brandon Philips962873e2013-03-26 13:58:51 -0700138 LOG(WARNING) << "dbus proxy object missing, continuing with no proxy.";
Andrew de los Reyes518502a2011-03-14 14:19:39 -0700139 timeout = 0;
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800140 }
Gilad Arnold1877c392012-02-10 11:34:33 -0800141
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800142 callbacks_.insert(make_pair(url, make_pair(callback, data)));
Alex Deymoc4acdf42014-05-28 21:07:10 -0700143 Closure* closure = NewPermanentCallback(
144 this,
145 &ChromeBrowserProxyResolver::HandleTimeout,
146 url);
Andrew de los Reyes518502a2011-03-14 14:19:39 -0700147 GSource* timer = g_timeout_source_new_seconds(timeout);
Alex Deymoc4acdf42014-05-28 21:07:10 -0700148 g_source_set_callback(
149 timer, utils::GlibRunClosure, closure, utils::GlibDestroyClosure);
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800150 g_source_attach(timer, NULL);
151 timers_.insert(make_pair(url, timer));
152 return true;
153}
154
155DBusHandlerResult ChromeBrowserProxyResolver::FilterMessage(
156 DBusConnection* connection,
157 DBusMessage* message) {
158 // Code lifted from libcros.
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -0800159 if (!dbus_->DBusMessageIsSignal(message,
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800160 kLibCrosProxyResolveSignalInterface,
161 kLibCrosProxyResolveName)) {
162 return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
163 }
164 // Get args
165 char* source_url = NULL;
166 char* proxy_list = NULL;
167 char* error = NULL;
168 DBusError arg_error;
169 dbus_error_init(&arg_error);
Gilad Arnold1b9d6ae2014-03-03 13:46:07 -0800170 if (!dbus_->DBusMessageGetArgs_3(message, &arg_error,
Gilad Arnoldb752fb32014-03-03 12:23:39 -0800171 &source_url,
172 &proxy_list,
173 &error)) {
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800174 LOG(ERROR) << "Error reading dbus signal.";
175 return DBUS_HANDLER_RESULT_HANDLED;
176 }
177 if (!source_url || !proxy_list) {
178 LOG(ERROR) << "Error getting url, proxy list from dbus signal.";
179 return DBUS_HANDLER_RESULT_HANDLED;
180 }
181 HandleReply(source_url, proxy_list);
182 return DBUS_HANDLER_RESULT_HANDLED;
183}
184
185bool ChromeBrowserProxyResolver::DeleteUrlState(
186 const string& source_url,
187 bool delete_timer,
188 pair<ProxiesResolvedFn, void*>* callback) {
189 {
190 CallbacksMap::iterator it = callbacks_.lower_bound(source_url);
191 TEST_AND_RETURN_FALSE(it != callbacks_.end());
192 TEST_AND_RETURN_FALSE(it->first == source_url);
193 if (callback)
194 *callback = it->second;
195 callbacks_.erase(it);
196 }
197 {
198 TimeoutsMap::iterator it = timers_.lower_bound(source_url);
199 TEST_AND_RETURN_FALSE(it != timers_.end());
200 TEST_AND_RETURN_FALSE(it->first == source_url);
201 if (delete_timer)
202 g_source_destroy(it->second);
203 timers_.erase(it);
204 }
205 return true;
206}
207
208void ChromeBrowserProxyResolver::HandleReply(const string& source_url,
209 const string& proxy_list) {
210 pair<ProxiesResolvedFn, void*> callback;
211 TEST_AND_RETURN(DeleteUrlState(source_url, true, &callback));
212 (*callback.first)(ParseProxyString(proxy_list), callback.second);
213}
214
215void ChromeBrowserProxyResolver::HandleTimeout(string source_url) {
216 LOG(INFO) << "Timeout handler called. Seems Chrome isn't responding.";
217 pair<ProxiesResolvedFn, void*> callback;
218 TEST_AND_RETURN(DeleteUrlState(source_url, false, &callback));
219 deque<string> proxies;
220 proxies.push_back(kNoProxy);
221 (*callback.first)(proxies, callback.second);
222}
223
224deque<string> ChromeBrowserProxyResolver::ParseProxyString(
225 const string& input) {
226 deque<string> ret;
227 // Some of this code taken from
228 // http://src.chromium.org/svn/trunk/src/net/proxy/proxy_server.cc and
229 // http://src.chromium.org/svn/trunk/src/net/proxy/proxy_list.cc
230 StringTokenizer entry_tok(input, ";");
231 while (entry_tok.GetNext()) {
232 string token = entry_tok.token();
Ben Chan736fcb52014-05-21 18:28:22 -0700233 base::TrimWhitespaceASCII(token, base::TRIM_ALL, &token);
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800234
235 // Start by finding the first space (if any).
236 std::string::iterator space;
237 for (space = token.begin(); space != token.end(); ++space) {
238 if (IsAsciiWhitespace(*space)) {
239 break;
240 }
241 }
242
243 string scheme = string(token.begin(), space);
244 StringToLowerASCII(&scheme);
245 // Chrome uses "socks" to mean socks4 and "proxy" to mean http.
246 if (scheme == "socks")
247 scheme += "4";
248 else if (scheme == "proxy")
249 scheme = "http";
250 else if (scheme != "https" &&
251 scheme != "socks4" &&
252 scheme != "socks5" &&
253 scheme != "direct")
254 continue; // Invalid proxy scheme
255
256 string host_and_port = string(space, token.end());
Ben Chan736fcb52014-05-21 18:28:22 -0700257 base::TrimWhitespaceASCII(host_and_port, base::TRIM_ALL, &host_and_port);
Andrew de los Reyes000d8952011-03-02 15:21:14 -0800258 if (scheme != "direct" && host_and_port.empty())
259 continue; // Must supply host/port when non-direct proxy used.
260 ret.push_back(scheme + "://" + host_and_port);
261 }
262 if (ret.empty() || *ret.rbegin() != kNoProxy)
263 ret.push_back(kNoProxy);
264 return ret;
265}
266
267} // namespace chromeos_update_engine