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.
5 #include "chrome/browser/chromeos/proxy_config_service_impl.h"
9 #include "base/format_macros.h"
10 #include "base/json/json_writer.h"
11 #include "base/logging.h"
12 #include "base/message_loop/message_loop.h"
13 #include "base/prefs/testing_pref_service.h"
14 #include "base/strings/stringprintf.h"
15 #include "chrome/browser/chromeos/settings/cros_settings.h"
16 #include "chrome/browser/chromeos/settings/device_settings_service.h"
17 #include "chrome/browser/chromeos/ui_proxy_config.h"
18 #include "chrome/common/pref_names.h"
19 #include "chromeos/dbus/dbus_thread_manager.h"
20 #include "chromeos/dbus/shill_profile_client.h"
21 #include "chromeos/dbus/shill_service_client.h"
22 #include "chromeos/network/network_handler.h"
23 #include "chromeos/network/network_state.h"
24 #include "chromeos/network/network_state_handler.h"
25 #include "content/public/test/test_browser_thread.h"
26 #include "net/proxy/proxy_config_service_common_unittest.h"
27 #include "testing/gtest/include/gtest/gtest.h"
28 #include "third_party/cros_system_api/dbus/service_constants.h"
30 using content::BrowserThread
;
37 UIProxyConfig::Mode mode
;
40 std::string bypass_rules
;
43 // Builds an identifier for each test in an array.
44 #define TEST_DESC(desc) base::StringPrintf("at line %d <%s>", __LINE__, desc)
46 // Shortcuts to declare enums within chromeos's ProxyConfig.
47 #define MK_MODE(mode) UIProxyConfig::MODE_##mode
49 // Inspired from net/proxy/proxy_config_service_linux_unittest.cc.
50 const struct TestParams
{
51 // Short description to identify the test
52 std::string description
;
56 // Expected outputs from fields of net::ProxyConfig (via IO).
59 net::ProxyRulesExpectation proxy_rules
;
62 TEST_DESC("No proxying"),
65 MK_MODE(DIRECT
), // mode
71 net::ProxyRulesExpectation::Empty(), // proxy_rules
75 TEST_DESC("Auto detect"),
78 MK_MODE(AUTO_DETECT
), // mode
84 net::ProxyRulesExpectation::Empty(), // proxy_rules
88 TEST_DESC("Valid PAC URL"),
91 MK_MODE(PAC_SCRIPT
), // mode
92 "http://wpad/wpad.dat", // pac_url
97 GURL("http://wpad/wpad.dat"), // pac_url
98 net::ProxyRulesExpectation::Empty(), // proxy_rules
102 TEST_DESC("Invalid PAC URL"),
105 MK_MODE(PAC_SCRIPT
), // mode
106 "wpad.dat", // pac_url
110 false, // auto_detect
112 net::ProxyRulesExpectation::Empty(), // proxy_rules
116 TEST_DESC("Single-host in proxy list"),
119 MK_MODE(SINGLE_PROXY
), // mode
121 "www.google.com", // server
125 false, // auto_detect
127 net::ProxyRulesExpectation::Single( // proxy_rules
128 "www.google.com:80", // single proxy
129 "<local>"), // bypass rules
133 TEST_DESC("Single-host, different port"),
136 MK_MODE(SINGLE_PROXY
), // mode
138 "www.google.com:99", // server
142 false, // auto_detect
144 net::ProxyRulesExpectation::Single( // proxy_rules
145 "www.google.com:99", // single
146 "<local>"), // bypass rules
150 TEST_DESC("Tolerate a scheme"),
153 MK_MODE(SINGLE_PROXY
), // mode
155 "http://www.google.com:99", // server
159 false, // auto_detect
161 net::ProxyRulesExpectation::Single( // proxy_rules
162 "www.google.com:99", // single proxy
163 "<local>"), // bypass rules
167 TEST_DESC("Per-scheme proxy rules"),
170 MK_MODE(PROXY_PER_SCHEME
), // mode
172 "http=www.google.com:80;https=https://www.foo.com:110;"
173 "ftp=ftp.foo.com:121;socks=socks5://socks.com:888", // server
177 false, // auto_detect
179 net::ProxyRulesExpectation::PerSchemeWithSocks( // proxy_rules
180 "www.google.com:80", // http
181 "https://www.foo.com:110", // https
182 "ftp.foo.com:121", // ftp
183 "socks5://socks.com:888", // fallback proxy
184 "<local>"), // bypass rules
188 TEST_DESC("Bypass rules"),
191 MK_MODE(SINGLE_PROXY
), // mode
193 "www.google.com", // server
194 "*.google.com, *foo.com:99, 1.2.3.4:22, 127.0.0.1/8", // bypass_rules
198 false, // auto_detect
200 net::ProxyRulesExpectation::Single( // proxy_rules
201 "www.google.com:80", // single proxy
203 "*.google.com,*foo.com:99,1.2.3.4:22,127.0.0.1/8,<local>"),
207 const char* kUserProfilePath
= "user_profile";
211 class ProxyConfigServiceImplTest
: public testing::Test
{
213 ProxyConfigServiceImplTest()
214 : ui_thread_(BrowserThread::UI
, &loop_
),
215 io_thread_(BrowserThread::IO
, &loop_
) {}
217 virtual void SetUp() {
218 DBusThreadManager::InitializeWithStub();
219 NetworkHandler::Initialize();
223 PrefProxyConfigTrackerImpl::RegisterPrefs(pref_service_
.registry());
225 // Create a ProxyConfigServiceImpl like for the system request context.
226 config_service_impl_
.reset(
227 new ProxyConfigServiceImpl(NULL
, // no profile prefs
229 proxy_config_service_
=
230 config_service_impl_
->CreateTrackingProxyConfigService(
231 scoped_ptr
<net::ProxyConfigService
>());
233 // CreateTrackingProxyConfigService triggers update of initial prefs proxy
234 // config by tracker to chrome proxy config service, so flush all pending
235 // tasks so that tests start fresh.
236 loop_
.RunUntilIdle();
239 void SetUpNetwork() {
240 ShillProfileClient::TestInterface
* profile_test
=
241 DBusThreadManager::Get()->GetShillProfileClient()->GetTestInterface();
242 ShillServiceClient::TestInterface
* service_test
=
243 DBusThreadManager::Get()->GetShillServiceClient()->GetTestInterface();
245 service_test
->ClearServices();
247 // Sends a notification about the added profile.
248 profile_test
->AddProfile(kUserProfilePath
, "user_hash");
250 service_test
->AddService("stub_wifi2", "wifi2_PSK",
251 shill::kTypeWifi
, shill::kStateOnline
,
252 true /* visible */, true /* watch */);
253 service_test
->SetServiceProperty("stub_wifi2",
254 shill::kGuidProperty
,
255 base::StringValue("stub_wifi2"));
256 profile_test
->AddService(kUserProfilePath
, "stub_wifi2");
258 loop_
.RunUntilIdle();
261 virtual void TearDown() {
262 config_service_impl_
->DetachFromPrefService();
263 loop_
.RunUntilIdle();
264 config_service_impl_
.reset();
265 proxy_config_service_
.reset();
266 NetworkHandler::Shutdown();
267 DBusThreadManager::Shutdown();
270 void InitConfigWithTestInput(const Input
& input
,
271 base::DictionaryValue
* result
) {
272 base::DictionaryValue
* new_config
= NULL
;
273 switch (input
.mode
) {
274 case MK_MODE(DIRECT
):
275 new_config
= ProxyConfigDictionary::CreateDirect();
277 case MK_MODE(AUTO_DETECT
):
278 new_config
= ProxyConfigDictionary::CreateAutoDetect();
280 case MK_MODE(PAC_SCRIPT
):
282 ProxyConfigDictionary::CreatePacScript(input
.pac_url
, false);
284 case MK_MODE(SINGLE_PROXY
):
285 case MK_MODE(PROXY_PER_SCHEME
):
287 ProxyConfigDictionary::CreateFixedServers(input
.server
,
291 result
->Swap(new_config
);
295 void SetConfig(base::DictionaryValue
* pref_proxy_config_dict
) {
296 std::string proxy_config
;
297 if (pref_proxy_config_dict
)
298 base::JSONWriter::Write(pref_proxy_config_dict
, &proxy_config
);
300 NetworkStateHandler
* network_state_handler
=
301 NetworkHandler::Get()->network_state_handler();
302 const NetworkState
* network
= network_state_handler
->DefaultNetwork();
303 ASSERT_TRUE(network
);
304 DBusThreadManager::Get()->GetShillServiceClient()->GetTestInterface()->
305 SetServiceProperty(network
->path(),
306 shill::kProxyConfigProperty
,
307 base::StringValue(proxy_config
));
310 // Synchronously gets the latest proxy config.
311 void SyncGetLatestProxyConfig(net::ProxyConfig
* config
) {
312 *config
= net::ProxyConfig();
313 // Let message loop process all messages. This will run
314 // ChromeProxyConfigService::UpdateProxyConfig, which is posted on IO from
315 // PrefProxyConfigTrackerImpl::OnProxyConfigChanged.
316 loop_
.RunUntilIdle();
317 net::ProxyConfigService::ConfigAvailability availability
=
318 proxy_config_service_
->GetLatestProxyConfig(config
);
320 EXPECT_EQ(net::ProxyConfigService::CONFIG_VALID
, availability
);
323 base::MessageLoop loop_
;
324 scoped_ptr
<net::ProxyConfigService
> proxy_config_service_
;
325 scoped_ptr
<ProxyConfigServiceImpl
> config_service_impl_
;
326 TestingPrefServiceSimple pref_service_
;
329 ScopedTestDeviceSettingsService test_device_settings_service_
;
330 ScopedTestCrosSettings test_cros_settings_
;
331 content::TestBrowserThread ui_thread_
;
332 content::TestBrowserThread io_thread_
;
335 TEST_F(ProxyConfigServiceImplTest
, NetworkProxy
) {
336 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(tests
); ++i
) {
337 SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS
"] %s", i
,
338 tests
[i
].description
.c_str()));
340 base::DictionaryValue test_config
;
341 InitConfigWithTestInput(tests
[i
].input
, &test_config
);
342 SetConfig(&test_config
);
344 net::ProxyConfig config
;
345 SyncGetLatestProxyConfig(&config
);
347 EXPECT_EQ(tests
[i
].auto_detect
, config
.auto_detect());
348 EXPECT_EQ(tests
[i
].pac_url
, config
.pac_url());
349 EXPECT_TRUE(tests
[i
].proxy_rules
.Matches(config
.proxy_rules()));
353 TEST_F(ProxyConfigServiceImplTest
, DynamicPrefsOverride
) {
354 // Groupings of 3 test inputs to use for managed, recommended and network
355 // proxies respectively. Only valid and non-direct test inputs are used.
356 const size_t proxies
[][3] = {
378 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(proxies
); ++i
) {
379 const TestParams
& managed_params
= tests
[proxies
[i
][0]];
380 const TestParams
& recommended_params
= tests
[proxies
[i
][1]];
381 const TestParams
& network_params
= tests
[proxies
[i
][2]];
383 SCOPED_TRACE(base::StringPrintf(
384 "Test[%" PRIuS
"] managed=[%s], recommended=[%s], network=[%s]", i
,
385 managed_params
.description
.c_str(),
386 recommended_params
.description
.c_str(),
387 network_params
.description
.c_str()));
389 base::DictionaryValue managed_config
;
390 InitConfigWithTestInput(managed_params
.input
, &managed_config
);
391 base::DictionaryValue recommended_config
;
392 InitConfigWithTestInput(recommended_params
.input
, &recommended_config
);
393 base::DictionaryValue network_config
;
394 InitConfigWithTestInput(network_params
.input
, &network_config
);
396 // Managed proxy pref should take effect over recommended proxy and
397 // non-existent network proxy.
399 pref_service_
.SetManagedPref(prefs::kProxy
, managed_config
.DeepCopy());
400 pref_service_
.SetRecommendedPref(prefs::kProxy
,
401 recommended_config
.DeepCopy());
402 net::ProxyConfig actual_config
;
403 SyncGetLatestProxyConfig(&actual_config
);
404 EXPECT_EQ(managed_params
.auto_detect
, actual_config
.auto_detect());
405 EXPECT_EQ(managed_params
.pac_url
, actual_config
.pac_url());
406 EXPECT_TRUE(managed_params
.proxy_rules
.Matches(
407 actual_config
.proxy_rules()));
409 // Recommended proxy pref should take effect when managed proxy pref is
411 pref_service_
.RemoveManagedPref(prefs::kProxy
);
412 SyncGetLatestProxyConfig(&actual_config
);
413 EXPECT_EQ(recommended_params
.auto_detect
, actual_config
.auto_detect());
414 EXPECT_EQ(recommended_params
.pac_url
, actual_config
.pac_url());
415 EXPECT_TRUE(recommended_params
.proxy_rules
.Matches(
416 actual_config
.proxy_rules()));
418 // Network proxy should take take effect over recommended proxy pref.
419 SetConfig(&network_config
);
420 SyncGetLatestProxyConfig(&actual_config
);
421 EXPECT_EQ(network_params
.auto_detect
, actual_config
.auto_detect());
422 EXPECT_EQ(network_params
.pac_url
, actual_config
.pac_url());
423 EXPECT_TRUE(network_params
.proxy_rules
.Matches(
424 actual_config
.proxy_rules()));
426 // Managed proxy pref should take effect over network proxy.
427 pref_service_
.SetManagedPref(prefs::kProxy
, managed_config
.DeepCopy());
428 SyncGetLatestProxyConfig(&actual_config
);
429 EXPECT_EQ(managed_params
.auto_detect
, actual_config
.auto_detect());
430 EXPECT_EQ(managed_params
.pac_url
, actual_config
.pac_url());
431 EXPECT_TRUE(managed_params
.proxy_rules
.Matches(
432 actual_config
.proxy_rules()));
434 // Network proxy should take effect over recommended proxy pref when managed
435 // proxy pref is removed.
436 pref_service_
.RemoveManagedPref(prefs::kProxy
);
437 SyncGetLatestProxyConfig(&actual_config
);
438 EXPECT_EQ(network_params
.auto_detect
, actual_config
.auto_detect());
439 EXPECT_EQ(network_params
.pac_url
, actual_config
.pac_url());
440 EXPECT_TRUE(network_params
.proxy_rules
.Matches(
441 actual_config
.proxy_rules()));
443 // Removing recommended proxy pref should have no effect on network proxy.
444 pref_service_
.RemoveRecommendedPref(prefs::kProxy
);
445 SyncGetLatestProxyConfig(&actual_config
);
446 EXPECT_EQ(network_params
.auto_detect
, actual_config
.auto_detect());
447 EXPECT_EQ(network_params
.pac_url
, actual_config
.pac_url());
448 EXPECT_TRUE(network_params
.proxy_rules
.Matches(
449 actual_config
.proxy_rules()));
453 } // namespace chromeos