//
// 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