1 // Copyright (c) 2012 The Chromium 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 "net/proxy/proxy_config_service_mac.h"
6
7 #include <CoreFoundation/CoreFoundation.h>
8 #include <SystemConfiguration/SystemConfiguration.h>
9
10 #include "base/bind.h"
11 #include "base/logging.h"
12 #include "base/mac/foundation_util.h"
13 #include "base/mac/scoped_cftyperef.h"
14 #include "base/message_loop/message_loop.h"
15 #include "base/strings/sys_string_conversions.h"
16 #include "net/base/net_errors.h"
17 #include "net/proxy/proxy_config.h"
18 #include "net/proxy/proxy_info.h"
19 #include "net/proxy/proxy_server.h"
20
21 namespace net {
22
23 namespace {
24
25 // Utility function to pull out a boolean value from a dictionary and return it,
26 // returning a default value if the key is not present.
GetBoolFromDictionary(CFDictionaryRef dict,CFStringRef key,bool default_value)27 bool GetBoolFromDictionary(CFDictionaryRef dict,
28 CFStringRef key,
29 bool default_value) {
30 CFNumberRef number = base::mac::GetValueFromDictionary<CFNumberRef>(dict,
31 key);
32 if (!number)
33 return default_value;
34
35 int int_value;
36 if (CFNumberGetValue(number, kCFNumberIntType, &int_value))
37 return int_value;
38 else
39 return default_value;
40 }
41
GetCurrentProxyConfig(ProxyConfig * config)42 void GetCurrentProxyConfig(ProxyConfig* config) {
43 base::ScopedCFTypeRef<CFDictionaryRef> config_dict(
44 SCDynamicStoreCopyProxies(NULL));
45 DCHECK(config_dict);
46
47 // auto-detect
48
49 // There appears to be no UI for this configuration option, and we're not sure
50 // if Apple's proxy code even takes it into account. But the constant is in
51 // the header file so we'll use it.
52 config->set_auto_detect(
53 GetBoolFromDictionary(config_dict.get(),
54 kSCPropNetProxiesProxyAutoDiscoveryEnable,
55 false));
56
57 // PAC file
58
59 if (GetBoolFromDictionary(config_dict.get(),
60 kSCPropNetProxiesProxyAutoConfigEnable,
61 false)) {
62 CFStringRef pac_url_ref = base::mac::GetValueFromDictionary<CFStringRef>(
63 config_dict.get(), kSCPropNetProxiesProxyAutoConfigURLString);
64 if (pac_url_ref)
65 config->set_pac_url(GURL(base::SysCFStringRefToUTF8(pac_url_ref)));
66 }
67
68 // proxies (for now ftp, http, https, and SOCKS)
69
70 if (GetBoolFromDictionary(config_dict.get(),
71 kSCPropNetProxiesFTPEnable,
72 false)) {
73 ProxyServer proxy_server =
74 ProxyServer::FromDictionary(ProxyServer::SCHEME_HTTP,
75 config_dict.get(),
76 kSCPropNetProxiesFTPProxy,
77 kSCPropNetProxiesFTPPort);
78 if (proxy_server.is_valid()) {
79 config->proxy_rules().type =
80 ProxyConfig::ProxyRules::TYPE_PROXY_PER_SCHEME;
81 config->proxy_rules().proxies_for_ftp.SetSingleProxyServer(proxy_server);
82 }
83 }
84 if (GetBoolFromDictionary(config_dict.get(),
85 kSCPropNetProxiesHTTPEnable,
86 false)) {
87 ProxyServer proxy_server =
88 ProxyServer::FromDictionary(ProxyServer::SCHEME_HTTP,
89 config_dict.get(),
90 kSCPropNetProxiesHTTPProxy,
91 kSCPropNetProxiesHTTPPort);
92 if (proxy_server.is_valid()) {
93 config->proxy_rules().type =
94 ProxyConfig::ProxyRules::TYPE_PROXY_PER_SCHEME;
95 config->proxy_rules().proxies_for_http.SetSingleProxyServer(proxy_server);
96 }
97 }
98 if (GetBoolFromDictionary(config_dict.get(),
99 kSCPropNetProxiesHTTPSEnable,
100 false)) {
101 ProxyServer proxy_server =
102 ProxyServer::FromDictionary(ProxyServer::SCHEME_HTTP,
103 config_dict.get(),
104 kSCPropNetProxiesHTTPSProxy,
105 kSCPropNetProxiesHTTPSPort);
106 if (proxy_server.is_valid()) {
107 config->proxy_rules().type =
108 ProxyConfig::ProxyRules::TYPE_PROXY_PER_SCHEME;
109 config->proxy_rules().proxies_for_https.
110 SetSingleProxyServer(proxy_server);
111 }
112 }
113 if (GetBoolFromDictionary(config_dict.get(),
114 kSCPropNetProxiesSOCKSEnable,
115 false)) {
116 ProxyServer proxy_server =
117 ProxyServer::FromDictionary(ProxyServer::SCHEME_SOCKS5,
118 config_dict.get(),
119 kSCPropNetProxiesSOCKSProxy,
120 kSCPropNetProxiesSOCKSPort);
121 if (proxy_server.is_valid()) {
122 config->proxy_rules().type =
123 ProxyConfig::ProxyRules::TYPE_PROXY_PER_SCHEME;
124 config->proxy_rules().fallback_proxies.SetSingleProxyServer(proxy_server);
125 }
126 }
127
128 // proxy bypass list
129
130 CFArrayRef bypass_array_ref = base::mac::GetValueFromDictionary<CFArrayRef>(
131 config_dict.get(), kSCPropNetProxiesExceptionsList);
132 if (bypass_array_ref) {
133 CFIndex bypass_array_count = CFArrayGetCount(bypass_array_ref);
134 for (CFIndex i = 0; i < bypass_array_count; ++i) {
135 CFStringRef bypass_item_ref = base::mac::CFCast<CFStringRef>(
136 CFArrayGetValueAtIndex(bypass_array_ref, i));
137 if (!bypass_item_ref) {
138 LOG(WARNING) << "Expected value for item " << i
139 << " in the kSCPropNetProxiesExceptionsList"
140 " to be a CFStringRef but it was not";
141
142 } else {
143 config->proxy_rules().bypass_rules.AddRuleFromString(
144 base::SysCFStringRefToUTF8(bypass_item_ref));
145 }
146 }
147 }
148
149 // proxy bypass boolean
150
151 if (GetBoolFromDictionary(config_dict.get(),
152 kSCPropNetProxiesExcludeSimpleHostnames,
153 false)) {
154 config->proxy_rules().bypass_rules.AddRuleToBypassLocal();
155 }
156
157 // Source
158 config->set_source(PROXY_CONFIG_SOURCE_SYSTEM);
159 }
160
161 } // namespace
162
163 // Reference-counted helper for posting a task to
164 // ProxyConfigServiceMac::OnProxyConfigChanged between the notifier and IO
165 // thread. This helper object may outlive the ProxyConfigServiceMac.
166 class ProxyConfigServiceMac::Helper
167 : public base::RefCountedThreadSafe<ProxyConfigServiceMac::Helper> {
168 public:
Helper(ProxyConfigServiceMac * parent)169 explicit Helper(ProxyConfigServiceMac* parent) : parent_(parent) {
170 DCHECK(parent);
171 }
172
173 // Called when the parent is destroyed.
Orphan()174 void Orphan() {
175 parent_ = NULL;
176 }
177
OnProxyConfigChanged(const ProxyConfig & new_config)178 void OnProxyConfigChanged(const ProxyConfig& new_config) {
179 if (parent_)
180 parent_->OnProxyConfigChanged(new_config);
181 }
182
183 private:
184 friend class base::RefCountedThreadSafe<Helper>;
~Helper()185 ~Helper() {}
186
187 ProxyConfigServiceMac* parent_;
188 };
189
SetDynamicStoreNotificationKeys(SCDynamicStoreRef store)190 void ProxyConfigServiceMac::Forwarder::SetDynamicStoreNotificationKeys(
191 SCDynamicStoreRef store) {
192 proxy_config_service_->SetDynamicStoreNotificationKeys(store);
193 }
194
OnNetworkConfigChange(CFArrayRef changed_keys)195 void ProxyConfigServiceMac::Forwarder::OnNetworkConfigChange(
196 CFArrayRef changed_keys) {
197 proxy_config_service_->OnNetworkConfigChange(changed_keys);
198 }
199
ProxyConfigServiceMac(const scoped_refptr<base::SingleThreadTaskRunner> & io_thread_task_runner)200 ProxyConfigServiceMac::ProxyConfigServiceMac(
201 const scoped_refptr<base::SingleThreadTaskRunner>& io_thread_task_runner)
202 : forwarder_(this),
203 has_fetched_config_(false),
204 helper_(new Helper(this)),
205 io_thread_task_runner_(io_thread_task_runner) {
206 DCHECK(io_thread_task_runner_.get());
207 config_watcher_.reset(new NetworkConfigWatcherMac(&forwarder_));
208 }
209
~ProxyConfigServiceMac()210 ProxyConfigServiceMac::~ProxyConfigServiceMac() {
211 DCHECK(io_thread_task_runner_->BelongsToCurrentThread());
212 // Delete the config_watcher_ to ensure the notifier thread finishes before
213 // this object is destroyed.
214 config_watcher_.reset();
215 helper_->Orphan();
216 }
217
AddObserver(Observer * observer)218 void ProxyConfigServiceMac::AddObserver(Observer* observer) {
219 DCHECK(io_thread_task_runner_->BelongsToCurrentThread());
220 observers_.AddObserver(observer);
221 }
222
RemoveObserver(Observer * observer)223 void ProxyConfigServiceMac::RemoveObserver(Observer* observer) {
224 DCHECK(io_thread_task_runner_->BelongsToCurrentThread());
225 observers_.RemoveObserver(observer);
226 }
227
228 net::ProxyConfigService::ConfigAvailability
GetLatestProxyConfig(ProxyConfig * config)229 ProxyConfigServiceMac::GetLatestProxyConfig(ProxyConfig* config) {
230 DCHECK(io_thread_task_runner_->BelongsToCurrentThread());
231
232 // Lazy-initialize by fetching the proxy setting from this thread.
233 if (!has_fetched_config_) {
234 GetCurrentProxyConfig(&last_config_fetched_);
235 has_fetched_config_ = true;
236 }
237
238 *config = last_config_fetched_;
239 return has_fetched_config_ ? CONFIG_VALID : CONFIG_PENDING;
240 }
241
SetDynamicStoreNotificationKeys(SCDynamicStoreRef store)242 void ProxyConfigServiceMac::SetDynamicStoreNotificationKeys(
243 SCDynamicStoreRef store) {
244 // Called on notifier thread.
245
246 CFStringRef proxies_key = SCDynamicStoreKeyCreateProxies(NULL);
247 CFArrayRef key_array = CFArrayCreate(
248 NULL, (const void **)(&proxies_key), 1, &kCFTypeArrayCallBacks);
249
250 bool ret = SCDynamicStoreSetNotificationKeys(store, key_array, NULL);
251 // TODO(willchan): Figure out a proper way to handle this rather than crash.
252 CHECK(ret);
253
254 CFRelease(key_array);
255 CFRelease(proxies_key);
256 }
257
OnNetworkConfigChange(CFArrayRef changed_keys)258 void ProxyConfigServiceMac::OnNetworkConfigChange(CFArrayRef changed_keys) {
259 // Called on notifier thread.
260
261 // Fetch the new system proxy configuration.
262 ProxyConfig new_config;
263 GetCurrentProxyConfig(&new_config);
264
265 // Call OnProxyConfigChanged() on the IO thread to notify our observers.
266 io_thread_task_runner_->PostTask(
267 FROM_HERE,
268 base::Bind(&Helper::OnProxyConfigChanged, helper_.get(), new_config));
269 }
270
OnProxyConfigChanged(const ProxyConfig & new_config)271 void ProxyConfigServiceMac::OnProxyConfigChanged(
272 const ProxyConfig& new_config) {
273 DCHECK(io_thread_task_runner_->BelongsToCurrentThread());
274
275 // Keep track of the last value we have seen.
276 has_fetched_config_ = true;
277 last_config_fetched_ = new_config;
278
279 // Notify all the observers.
280 FOR_EACH_OBSERVER(Observer, observers_,
281 OnProxyConfigChanged(new_config, CONFIG_VALID));
282 }
283
284 } // namespace net
285