| // |
| // 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 <vector> |
| |
| #include <gtest/gtest.h> |
| |
| #include <base/bind.h> |
| #include <brillo/make_unique_ptr.h> |
| #include <brillo/message_loops/fake_message_loop.h> |
| |
| #include "libcros/dbus-proxies.h" |
| #include "libcros/dbus-proxy-mocks.h" |
| #include "update_engine/dbus_test_utils.h" |
| |
| using ::testing::Return; |
| using ::testing::StrEq; |
| using ::testing::_; |
| using brillo::MessageLoop; |
| using org::chromium::LibCrosServiceInterfaceProxyMock; |
| using org::chromium::UpdateEngineLibcrosProxyResolvedInterfaceProxyMock; |
| using std::deque; |
| using std::string; |
| using std::vector; |
| |
| namespace chromeos_update_engine { |
| |
| class ChromeBrowserProxyResolverTest : public ::testing::Test { |
| protected: |
| ChromeBrowserProxyResolverTest() |
| : service_interface_mock_(new LibCrosServiceInterfaceProxyMock()), |
| ue_proxy_resolved_interface_mock_( |
| new UpdateEngineLibcrosProxyResolvedInterfaceProxyMock()), |
| libcros_proxy_( |
| brillo::make_unique_ptr(service_interface_mock_), |
| brillo::make_unique_ptr(ue_proxy_resolved_interface_mock_)) {} |
| |
| void SetUp() override { |
| loop_.SetAsCurrent(); |
| // The ProxyResolved signal should be subscribed to. |
| MOCK_SIGNAL_HANDLER_EXPECT_SIGNAL_HANDLER( |
| ue_proxy_resolved_signal_, |
| *ue_proxy_resolved_interface_mock_, |
| ProxyResolved); |
| |
| EXPECT_TRUE(resolver_.Init()); |
| // Run the loop once to dispatch the successfully registered signal handler. |
| EXPECT_TRUE(loop_.RunOnce(false)); |
| } |
| |
| void TearDown() override { |
| EXPECT_FALSE(loop_.PendingTasks()); |
| } |
| |
| // Send the signal to the callback passed during registration of the |
| // ProxyResolved. |
| void SendReplySignal(const string& source_url, |
| const string& proxy_info, |
| const string& error_message); |
| |
| void RunTest(bool chrome_replies, bool chrome_alive); |
| |
| brillo::FakeMessageLoop loop_{nullptr}; |
| |
| // Local pointers to the mocks. The instances are owned by the |
| // |libcros_proxy_|. |
| LibCrosServiceInterfaceProxyMock* service_interface_mock_; |
| UpdateEngineLibcrosProxyResolvedInterfaceProxyMock* |
| ue_proxy_resolved_interface_mock_; |
| |
| // The registered signal handler for the signal |
| // UpdateEngineLibcrosProxyResolvedInterface.ProxyResolved. |
| chromeos_update_engine::dbus_test_utils::MockSignalHandler< |
| void(const string&, const string&, const string&)> |
| ue_proxy_resolved_signal_; |
| |
| LibCrosProxy libcros_proxy_; |
| ChromeBrowserProxyResolver resolver_{&libcros_proxy_}; |
| }; |
| |
| |
| void ChromeBrowserProxyResolverTest::SendReplySignal( |
| const string& source_url, |
| const string& proxy_info, |
| const string& error_message) { |
| ASSERT_TRUE(ue_proxy_resolved_signal_.IsHandlerRegistered()); |
| ue_proxy_resolved_signal_.signal_callback().Run( |
| source_url, proxy_info, error_message); |
| } |
| |
| namespace { |
| void CheckResponseResolved(const deque<string>& proxies) { |
| EXPECT_EQ(2U, proxies.size()); |
| EXPECT_EQ("socks5://192.168.52.83:5555", proxies[0]); |
| EXPECT_EQ(kNoProxy, proxies[1]); |
| MessageLoop::current()->BreakLoop(); |
| } |
| |
| void CheckResponseNoReply(const deque<string>& proxies) { |
| EXPECT_EQ(1U, proxies.size()); |
| EXPECT_EQ(kNoProxy, proxies[0]); |
| MessageLoop::current()->BreakLoop(); |
| } |
| } // namespace |
| |
| // chrome_replies should be set to whether or not we fake a reply from |
| // chrome. If there's no reply, the resolver should time out. |
| // If chrome_alive is false, assume that sending to chrome fails. |
| void ChromeBrowserProxyResolverTest::RunTest(bool chrome_replies, |
| bool chrome_alive) { |
| char kUrl[] = "http://example.com/blah"; |
| char kProxyConfig[] = "SOCKS5 192.168.52.83:5555;DIRECT"; |
| |
| EXPECT_CALL(*service_interface_mock_, |
| ResolveNetworkProxy(StrEq(kUrl), |
| StrEq(kLibCrosProxyResolveSignalInterface), |
| StrEq(kLibCrosProxyResolveName), |
| _, |
| _)) |
| .WillOnce(Return(chrome_alive)); |
| |
| ProxiesResolvedFn get_proxies_response = base::Bind(&CheckResponseNoReply); |
| if (chrome_replies) { |
| get_proxies_response = base::Bind(&CheckResponseResolved); |
| MessageLoop::current()->PostDelayedTask( |
| FROM_HERE, |
| base::Bind(&ChromeBrowserProxyResolverTest::SendReplySignal, |
| base::Unretained(this), |
| kUrl, |
| kProxyConfig, |
| ""), |
| base::TimeDelta::FromSeconds(1)); |
| } |
| |
| EXPECT_NE(kProxyRequestIdNull, |
| resolver_.GetProxiesForUrl(kUrl, get_proxies_response)); |
| MessageLoop::current()->Run(); |
| } |
| |
| |
| TEST_F(ChromeBrowserProxyResolverTest, ParseTest) { |
| // Test ideas from |
| // http://src.chromium.org/svn/trunk/src/net/proxy/proxy_list_unittest.cc |
| vector<string> inputs = { |
| "PROXY foopy:10", |
| " DIRECT", // leading space. |
| "PROXY foopy1 ; proxy foopy2;\t DIRECT", |
| "proxy foopy1 ; SOCKS foopy2", |
| "DIRECT ; proxy foopy1 ; DIRECT ; SOCKS5 foopy2;DIRECT ", |
| "DIRECT ; proxy foopy1:80; DIRECT ; DIRECT", |
| "PROXY-foopy:10", |
| "PROXY", |
| "PROXY foopy1 ; JUNK ; JUNK ; SOCKS5 foopy2 ; ;", |
| "HTTP foopy1; SOCKS5 foopy2"}; |
| vector<deque<string>> outputs = { |
| {"http://foopy:10", kNoProxy}, |
| {kNoProxy}, |
| {"http://foopy1", "http://foopy2", kNoProxy}, |
| {"http://foopy1", "socks4://foopy2", kNoProxy}, |
| {kNoProxy, "http://foopy1", kNoProxy, "socks5://foopy2", kNoProxy}, |
| {kNoProxy, "http://foopy1:80", kNoProxy, kNoProxy}, |
| {kNoProxy}, |
| {kNoProxy}, |
| {"http://foopy1", "socks5://foopy2", kNoProxy}, |
| {"socks5://foopy2", kNoProxy}}; |
| ASSERT_EQ(inputs.size(), outputs.size()); |
| |
| for (size_t i = 0; i < inputs.size(); i++) { |
| deque<string> results = |
| ChromeBrowserProxyResolver::ParseProxyString(inputs[i]); |
| deque<string>& expected = outputs[i]; |
| EXPECT_EQ(results.size(), expected.size()) << "i = " << i; |
| if (expected.size() != results.size()) |
| continue; |
| for (size_t j = 0; j < expected.size(); j++) { |
| EXPECT_EQ(expected[j], results[j]) << "i = " << i; |
| } |
| } |
| } |
| |
| TEST_F(ChromeBrowserProxyResolverTest, SuccessTest) { |
| RunTest(true, true); |
| } |
| |
| TEST_F(ChromeBrowserProxyResolverTest, NoReplyTest) { |
| RunTest(false, true); |
| } |
| |
| TEST_F(ChromeBrowserProxyResolverTest, NoChromeTest) { |
| RunTest(false, false); |
| } |
| |
| TEST_F(ChromeBrowserProxyResolverTest, CancelCallbackTest) { |
| int called = 0; |
| auto callback = base::Bind( |
| [](int* called, const deque<string>& proxies) { (*called)++; }, &called); |
| |
| EXPECT_CALL(*service_interface_mock_, ResolveNetworkProxy(_, _, _, _, _)) |
| .Times(4) |
| .WillRepeatedly(Return(true)); |
| |
| EXPECT_NE(kProxyRequestIdNull, |
| resolver_.GetProxiesForUrl("http://urlA", callback)); |
| ProxyRequestId req_b = resolver_.GetProxiesForUrl("http://urlB", callback); |
| // Note that we add twice the same url. |
| ProxyRequestId req_c = resolver_.GetProxiesForUrl("http://urlC", callback); |
| EXPECT_NE(kProxyRequestIdNull, |
| resolver_.GetProxiesForUrl("http://urlC", callback)); |
| |
| EXPECT_EQ(0, called); |
| EXPECT_TRUE(resolver_.CancelProxyRequest(req_b)); |
| EXPECT_TRUE(resolver_.CancelProxyRequest(req_c)); |
| // Canceling the same request twice should fail even if there's another |
| // request for the same URL. |
| EXPECT_FALSE(resolver_.CancelProxyRequest(req_c)); |
| |
| loop_.Run(); |
| EXPECT_EQ(2, called); |
| } |
| |
| } // namespace chromeos_update_engine |