From 622ef2bb6d68c21c6e58be247de389ad707de279 Mon Sep 17 00:00:00 2001 From: "nsylvain@google.com" Date: Wed, 30 Jul 2008 20:19:33 +0000 Subject: Change the proxy failover to be less aggressive. We now clear the list of bad proxies when we get a new configuration or when a direct connection fails and we try again to get the pac. git-svn-id: svn://svn.chromium.org/chrome/trunk/src@134 0039d316-1c4b-4281-b951-d872f2087c98 --- net/http/http_proxy_service.cc | 19 +++-- net/http/http_proxy_service_unittest.cc | 143 ++++++++++++++++++++++++++++---- 2 files changed, 140 insertions(+), 22 deletions(-) diff --git a/net/http/http_proxy_service.cc b/net/http/http_proxy_service.cc index d851a6d..7adcc8c 100644 --- a/net/http/http_proxy_service.cc +++ b/net/http/http_proxy_service.cc @@ -385,10 +385,6 @@ int HttpProxyService::ReconsiderProxyAfterError(const GURL& url, HttpProxyInfo* result, CompletionCallback* callback, PacRequest** pac_request) { - bool was_direct = result->is_direct(); - if (!was_direct && result->Fallback(&http_proxy_retry_info_)) - return OK; - // Check to see if we have a new config since ResolveProxy was called. We // want to re-run ResolveProxy in two cases: 1) we have a new config, or 2) a // direct connection failed and we never tried the current config. @@ -407,8 +403,18 @@ int HttpProxyService::ReconsiderProxyAfterError(const GURL& url, re_resolve = true; } } - if (re_resolve) + if (re_resolve) { + // If we have a new config or the config was never tried, we delete the + // list of bad proxies and we try again. + http_proxy_retry_info_.clear(); return ResolveProxy(url, result, callback, pac_request); + } + + // We don't have new proxy settings to try, fallback to the next proxy + // in the list. + bool was_direct = result->is_direct(); + if (!was_direct && result->Fallback(&http_proxy_retry_info_)) + return OK; if (!config_.auto_detect && !config_.proxy_server.empty()) { // If auto detect is on, then we should try a DIRECT connection @@ -452,6 +458,9 @@ void HttpProxyService::UpdateConfig() { config_ = latest; config_is_bad_ = false; + + // We have a new config, we should clear the list of bad proxies. + http_proxy_retry_info_.clear(); } bool HttpProxyService::ShouldBypassProxyForURL(const GURL& url) { diff --git a/net/http/http_proxy_service_unittest.cc b/net/http/http_proxy_service_unittest.cc index c047d9c..10e3c74 100644 --- a/net/http/http_proxy_service_unittest.cc +++ b/net/http/http_proxy_service_unittest.cc @@ -37,15 +37,16 @@ namespace { class MockProxyResolver : public net::HttpProxyResolver { public: MockProxyResolver() : fail_get_proxy_for_url(false) { + config.reset(new net::HttpProxyConfig); } virtual int GetProxyConfig(net::HttpProxyConfig* results) { - *results = config; + *results = *(config.get()); return net::OK; } virtual int GetProxyForURL(const std::wstring& query_url, const std::wstring& pac_url, net::HttpProxyInfo* results) { - if (pac_url != config.pac_url) + if (pac_url != config->pac_url) return net::ERR_INVALID_ARGUMENT; if (fail_get_proxy_for_url) return net::ERR_FAILED; @@ -56,7 +57,7 @@ class MockProxyResolver : public net::HttpProxyResolver { } return net::OK; } - net::HttpProxyConfig config; + scoped_ptr config; net::HttpProxyInfo info; // info is only returned if query_url in GetProxyForURL matches this: @@ -83,7 +84,7 @@ TEST(HttpProxyServiceTest, Direct) { TEST(HttpProxyServiceTest, PAC) { MockProxyResolver resolver; - resolver.config.pac_url = L"http://foopy/proxy.pac"; + resolver.config->pac_url = L"http://foopy/proxy.pac"; resolver.info.UseNamedProxy(L"foopy"); resolver.info_predicate_query_host = "www.google.com"; @@ -100,7 +101,7 @@ TEST(HttpProxyServiceTest, PAC) { TEST(HttpProxyServiceTest, PAC_FailoverToDirect) { MockProxyResolver resolver; - resolver.config.pac_url = L"http://foopy/proxy.pac"; + resolver.config->pac_url = L"http://foopy/proxy.pac"; resolver.info.UseNamedProxy(L"foopy:8080"); resolver.info_predicate_query_host = "www.google.com"; @@ -124,7 +125,7 @@ TEST(HttpProxyServiceTest, PAC_FailsToDownload) { // Test what happens when we fail to download the PAC URL. MockProxyResolver resolver; - resolver.config.pac_url = L"http://foopy/proxy.pac"; + resolver.config->pac_url = L"http://foopy/proxy.pac"; resolver.info.UseNamedProxy(L"foopy:8080"); resolver.info_predicate_query_host = "www.google.com"; resolver.fail_get_proxy_for_url = true; @@ -157,7 +158,7 @@ TEST(HttpProxyServiceTest, ProxyFallback) { // are bad. MockProxyResolver resolver; - resolver.config.pac_url = L"http://foopy/proxy.pac"; + resolver.config->pac_url = L"http://foopy/proxy.pac"; resolver.info.UseNamedProxy(L"foopy1:8080;foopy2:9090"); resolver.info_predicate_query_host = "www.google.com"; resolver.fail_get_proxy_for_url = false; @@ -184,7 +185,7 @@ TEST(HttpProxyServiceTest, ProxyFallback) { // Create a new resolver that returns 3 proxies. The second one is already // known to be bad. - resolver.config.pac_url = L"http://foopy/proxy.pac"; + resolver.config->pac_url = L"http://foopy/proxy.pac"; resolver.info.UseNamedProxy(L"foopy3:7070;foopy1:8080;foopy2:9090"); resolver.info_predicate_query_host = "www.google.com"; resolver.fail_get_proxy_for_url = false; @@ -211,13 +212,121 @@ TEST(HttpProxyServiceTest, ProxyFallback) { // TODO(nsylvain): Test that the proxy can be retried after the delay. } +TEST(HttpProxyServiceTest, ProxyFallback_NewSettings) { + // Test proxy failover when new settings are available. + + MockProxyResolver resolver; + resolver.config->pac_url = L"http://foopy/proxy.pac"; + resolver.info.UseNamedProxy(L"foopy1:8080;foopy2:9090"); + resolver.info_predicate_query_host = "www.google.com"; + resolver.fail_get_proxy_for_url = false; + + net::HttpProxyService service(&resolver); + + GURL url("http://www.google.com/"); + + // Get the proxy information. + net::HttpProxyInfo info; + int rv = service.ResolveProxy(url, &info, NULL, NULL); + EXPECT_EQ(rv, net::OK); + EXPECT_FALSE(info.is_direct()); + + // The first item is valid. + EXPECT_EQ(info.proxy_server(), L"foopy1:8080"); + + // Fake an error on the proxy, and also a new configuration on the proxy. + resolver.config.reset(new net::HttpProxyConfig); + resolver.config->pac_url = L"http://foopy-new/proxy.pac"; + + rv = service.ReconsiderProxyAfterError(url, &info, NULL, NULL); + EXPECT_EQ(rv, net::OK); + + // The first proxy is still there since the configuration changed. + EXPECT_EQ(info.proxy_server(), L"foopy1:8080"); + + // We fake another error. It should now ignore the first one. + rv = service.ReconsiderProxyAfterError(url, &info, NULL, NULL); + EXPECT_EQ(rv, net::OK); + EXPECT_EQ(info.proxy_server(), L"foopy2:9090"); + + // We simulate a new configuration. + resolver.config.reset(new net::HttpProxyConfig); + resolver.config->pac_url = L"http://foopy-new2/proxy.pac"; + + // We fake anothe error. It should go back to the first proxy. + rv = service.ReconsiderProxyAfterError(url, &info, NULL, NULL); + EXPECT_EQ(rv, net::OK); + EXPECT_EQ(info.proxy_server(), L"foopy1:8080"); +} + +TEST(HttpProxyServiceTest, ProxyFallback_BadConfig) { + // Test proxy failover when the configuration is bad. + + MockProxyResolver resolver; + resolver.config->pac_url = L"http://foopy/proxy.pac"; + resolver.info.UseNamedProxy(L"foopy1:8080;foopy2:9090"); + resolver.info_predicate_query_host = "www.google.com"; + resolver.fail_get_proxy_for_url = false; + + net::HttpProxyService service(&resolver); + + GURL url("http://www.google.com/"); + + // Get the proxy information. + net::HttpProxyInfo info; + int rv = service.ResolveProxy(url, &info, NULL, NULL); + EXPECT_EQ(rv, net::OK); + EXPECT_FALSE(info.is_direct()); + + // The first item is valid. + EXPECT_EQ(info.proxy_server(), L"foopy1:8080"); + + // Fake a proxy error. + rv = service.ReconsiderProxyAfterError(url, &info, NULL, NULL); + EXPECT_EQ(rv, net::OK); + + // The first proxy is ignored, and the second one is selected. + EXPECT_FALSE(info.is_direct()); + EXPECT_EQ(info.proxy_server(), L"foopy2:9090"); + + // Fake a PAC failure. + net::HttpProxyInfo info2; + resolver.fail_get_proxy_for_url = true; + rv = service.ResolveProxy(url, &info2, NULL, NULL); + EXPECT_EQ(rv, net::OK); + + // No proxy servers are returned. It's a direct connection. + EXPECT_TRUE(info2.is_direct()); + + // The PAC is now fixed and will return a proxy server. + // It should also clear the list of bad proxies. + resolver.fail_get_proxy_for_url = false; + + // Try to resolve, it will still return "direct" because we have no reason + // to check the config since everything works. + net::HttpProxyInfo info3; + rv = service.ResolveProxy(url, &info3, NULL, NULL); + EXPECT_EQ(rv, net::OK); + EXPECT_TRUE(info3.is_direct()); + + // But if the direct connection fails, we check if the ProxyInfo tried to + // resolve the proxy before, and if not (like in this case), we give the + // PAC another try. + rv = service.ReconsiderProxyAfterError(url, &info3, NULL, NULL); + EXPECT_EQ(rv, net::OK); + + // The first proxy is still there since the list of bad proxies got cleared. + EXPECT_FALSE(info3.is_direct()); + EXPECT_EQ(info3.proxy_server(), L"foopy1:8080"); +} + TEST(HttpProxyServiceTest, ProxyBypassList) { // Test what happens when a proxy bypass list is specified. MockProxyResolver resolver; - resolver.config.proxy_server = L"foopy1:8080;foopy2:9090"; - resolver.config.auto_detect = false; - resolver.config.proxy_bypass = L""; + resolver.config->proxy_server = L"foopy1:8080;foopy2:9090"; + resolver.config->auto_detect = false; + resolver.config->proxy_bypass = L""; net::HttpProxyService service(&resolver); GURL url("http://www.google.com/"); @@ -234,7 +343,7 @@ TEST(HttpProxyServiceTest, ProxyBypassList) { EXPECT_EQ(rv, net::OK); EXPECT_TRUE(info1.is_direct()); - resolver.config.proxy_bypass = L";*.org"; + resolver.config->proxy_bypass = L";*.org"; net::HttpProxyService service2(&resolver); GURL test_url2("http://www.webkit.org"); net::HttpProxyInfo info2; @@ -242,7 +351,7 @@ TEST(HttpProxyServiceTest, ProxyBypassList) { EXPECT_EQ(rv, net::OK); EXPECT_TRUE(info2.is_direct()); - resolver.config.proxy_bypass = L";*.org;7*"; + resolver.config->proxy_bypass = L";*.org;7*"; net::HttpProxyService service3(&resolver); GURL test_url3("http://74.125.19.147"); net::HttpProxyInfo info3; @@ -250,7 +359,7 @@ TEST(HttpProxyServiceTest, ProxyBypassList) { EXPECT_EQ(rv, net::OK); EXPECT_TRUE(info3.is_direct()); - resolver.config.proxy_bypass = L";*.org;"; + resolver.config->proxy_bypass = L";*.org;"; net::HttpProxyService service4(&resolver); GURL test_url4("http://www.msn.com"); net::HttpProxyInfo info4; @@ -261,8 +370,8 @@ TEST(HttpProxyServiceTest, ProxyBypassList) { TEST(HttpProxyServiceTest, PerProtocolProxyTests) { MockProxyResolver resolver; - resolver.config.proxy_server = L"http=foopy1:8080;https=foopy2:8080"; - resolver.config.auto_detect = false; + resolver.config->proxy_server = L"http=foopy1:8080;https=foopy2:8080"; + resolver.config->auto_detect = false; net::HttpProxyService service1(&resolver); GURL test_url1("http://www.msn.com"); @@ -288,7 +397,7 @@ TEST(HttpProxyServiceTest, PerProtocolProxyTests) { EXPECT_FALSE(info3.is_direct()); EXPECT_TRUE(info3.proxy_server() == L"foopy2:8080"); - resolver.config.proxy_server = L"foopy1:8080"; + resolver.config->proxy_server = L"foopy1:8080"; net::HttpProxyService service4(&resolver); GURL test_url4("www.microsoft.com"); net::HttpProxyInfo info4; -- cgit v1.1