blob: 90129e675c8fddec9fa87c5c3e21e0b2a2253401 [file] [log] [blame]
// Copyright 2015 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "services/proxy_resolver/host_resolver_mojo.h"
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include "base/functional/bind.h"
#include "base/test/task_environment.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "net/base/ip_address.h"
#include "net/base/net_errors.h"
#include "net/base/network_anonymization_key.h"
#include "net/base/request_priority.h"
#include "net/base/test_completion_callback.h"
#include "net/log/net_log_with_source.h"
#include "net/test/event_waiter.h"
#include "net/test/gtest_util.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "url/gurl.h"
#include "url/origin.h"
using net::test::IsError;
using net::test::IsOk;
namespace proxy_resolver {
namespace {
void Fail(int result) {
FAIL() << "Unexpected callback called with error " << result;
}
class MockMojoHostResolverRequest {
public:
MockMojoHostResolverRequest(
mojo::PendingRemote<mojom::HostResolverRequestClient> client,
base::OnceClosure error_callback)
: client_(std::move(client)), error_callback_(std::move(error_callback)) {
client_.set_disconnect_handler(base::BindOnce(
&MockMojoHostResolverRequest::OnDisconnect, base::Unretained(this)));
}
void OnDisconnect() { std::move(error_callback_).Run(); }
private:
mojo::Remote<mojom::HostResolverRequestClient> client_;
base::OnceClosure error_callback_;
};
struct HostResolverAction {
enum Action {
COMPLETE,
DROP,
RETAIN,
};
static HostResolverAction ReturnError(net::Error error) {
HostResolverAction result;
result.error = error;
return result;
}
static HostResolverAction ReturnResult(
std::vector<net::IPAddress> addresses) {
HostResolverAction result;
result.addresses = std::move(addresses);
return result;
}
static HostResolverAction DropRequest() {
HostResolverAction result;
result.action = DROP;
return result;
}
static HostResolverAction RetainRequest() {
HostResolverAction result;
result.action = RETAIN;
return result;
}
Action action = COMPLETE;
std::vector<net::IPAddress> addresses;
net::Error error = net::OK;
};
class MockMojoHostResolver : public HostResolverMojo::Impl {
public:
// Information logged from a call to ResolveDns().
struct RequestInfo {
std::string hostname;
net::NetworkAnonymizationKey network_anonymization_key;
bool operator==(const RequestInfo& other) const {
return hostname == other.hostname &&
network_anonymization_key == other.network_anonymization_key;
}
};
explicit MockMojoHostResolver(
base::RepeatingClosure request_connection_error_callback)
: request_connection_error_callback_(
std::move(request_connection_error_callback)) {}
~MockMojoHostResolver() override {
EXPECT_EQ(results_returned_, actions_.size());
}
void AddAction(HostResolverAction action) {
actions_.push_back(std::move(action));
}
const std::vector<RequestInfo>& request_info() const { return request_info_; }
void ResolveDns(
const std::string& hostname,
net::ProxyResolveDnsOperation operation,
const net::NetworkAnonymizationKey& network_anonymization_key,
mojo::PendingRemote<mojom::HostResolverRequestClient> client) override {
request_info_.push_back(RequestInfo{hostname, network_anonymization_key});
ASSERT_LE(results_returned_, actions_.size());
switch (actions_[results_returned_].action) {
case HostResolverAction::COMPLETE:
mojo::Remote<mojom::HostResolverRequestClient>(std::move(client))
->ReportResult(actions_[results_returned_].error,
actions_[results_returned_].addresses);
break;
case HostResolverAction::RETAIN:
requests_.push_back(std::make_unique<MockMojoHostResolverRequest>(
std::move(client),
base::BindOnce(request_connection_error_callback_)));
break;
case HostResolverAction::DROP:
break;
}
results_returned_++;
}
private:
std::vector<HostResolverAction> actions_;
size_t results_returned_ = 0;
std::vector<RequestInfo> request_info_;
base::RepeatingClosure request_connection_error_callback_;
std::vector<std::unique_ptr<MockMojoHostResolverRequest>> requests_;
};
} // namespace
class HostResolverMojoTest : public testing::Test {
protected:
enum class ConnectionErrorSource {
REQUEST,
};
using Waiter = net::EventWaiter<ConnectionErrorSource>;
HostResolverMojoTest()
: mock_resolver_(base::BindRepeating(&Waiter::NotifyEvent,
base::Unretained(&waiter_),
ConnectionErrorSource::REQUEST)),
resolver_(&mock_resolver_) {}
int Resolve(const std::string& hostname,
const net::NetworkAnonymizationKey& network_anonymization_key,
std::vector<net::IPAddress>* out_addresses) {
std::unique_ptr<ProxyHostResolver::Request> request =
resolver_.CreateRequest(hostname,
net::ProxyResolveDnsOperation::DNS_RESOLVE_EX,
network_anonymization_key);
net::TestCompletionCallback callback;
int result = callback.GetResult(request->Start(callback.callback()));
*out_addresses = request->GetResults();
return result;
}
base::test::TaskEnvironment task_environment_;
Waiter waiter_;
MockMojoHostResolver mock_resolver_;
HostResolverMojo resolver_;
};
TEST_F(HostResolverMojoTest, Basic) {
const net::SchemefulSite kSite =
net::SchemefulSite(GURL("https://not-example.com/"));
const auto kNetworkAnonymizationKey =
net::NetworkAnonymizationKey::CreateSameSite(kSite);
std::vector<net::IPAddress> addresses;
net::IPAddress address(1, 2, 3, 4);
addresses.push_back(address);
addresses.push_back(ConvertIPv4ToIPv4MappedIPv6(address));
mock_resolver_.AddAction(HostResolverAction::ReturnResult(addresses));
std::vector<net::IPAddress> result;
EXPECT_THAT(Resolve("example.com", kNetworkAnonymizationKey, &result),
IsOk());
EXPECT_EQ(addresses, result);
ASSERT_EQ(1u, mock_resolver_.request_info().size());
EXPECT_EQ("example.com", mock_resolver_.request_info()[0].hostname);
EXPECT_EQ(kNetworkAnonymizationKey,
mock_resolver_.request_info()[0].network_anonymization_key);
}
TEST_F(HostResolverMojoTest, ResolveCachedResult) {
std::vector<net::IPAddress> addresses;
net::IPAddress address(1, 2, 3, 4);
addresses.push_back(address);
addresses.push_back(ConvertIPv4ToIPv4MappedIPv6(address));
mock_resolver_.AddAction(HostResolverAction::ReturnResult(addresses));
// Load results into cache.
std::vector<net::IPAddress> result;
ASSERT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsOk());
ASSERT_EQ(1u, mock_resolver_.request_info().size());
// Expect results from cache.
result.clear();
EXPECT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsOk());
EXPECT_EQ(addresses, result);
EXPECT_EQ(1u, mock_resolver_.request_info().size());
}
// Make sure the cache indexes entries by NetworkAnonymizationKey.
TEST_F(HostResolverMojoTest, ResolveCachedResultWithNetworkAnonymizationKey) {
const net::SchemefulSite kSite =
net::SchemefulSite(GURL("https://not-example.com/"));
const auto kNetworkAnonymizationKey =
net::NetworkAnonymizationKey::CreateSameSite(kSite);
std::vector<net::IPAddress> addresses1;
net::IPAddress address1(1, 2, 3, 4);
addresses1.push_back(address1);
addresses1.push_back(ConvertIPv4ToIPv4MappedIPv6(address1));
mock_resolver_.AddAction(HostResolverAction::ReturnResult(addresses1));
// Load results into cache using kNetworkAnonymizationKey.
std::vector<net::IPAddress> result;
ASSERT_THAT(Resolve("example.com", kNetworkAnonymizationKey, &result),
IsOk());
ASSERT_EQ(1u, mock_resolver_.request_info().size());
// Expect results from cache when using kNetworkAnonymizationKey.
result.clear();
EXPECT_THAT(Resolve("example.com", kNetworkAnonymizationKey, &result),
IsOk());
EXPECT_EQ(addresses1, result);
EXPECT_EQ(1u, mock_resolver_.request_info().size());
// A request with an empty NetworkAnonymizationKey should not use results
// cached using kNetworkAnonymizationKey.
std::vector<net::IPAddress> addresses2;
net::IPAddress address2(2, 3, 5, 8);
addresses2.push_back(address2);
addresses2.push_back(ConvertIPv4ToIPv4MappedIPv6(address2));
mock_resolver_.AddAction(HostResolverAction::ReturnResult(addresses2));
result.clear();
EXPECT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsOk());
EXPECT_EQ(addresses2, result);
EXPECT_EQ(2u, mock_resolver_.request_info().size());
// Using the empty NetworkAnonymizationKey again should result in the second
// cached address list.
result.clear();
EXPECT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsOk());
EXPECT_EQ(addresses2, result);
EXPECT_EQ(2u, mock_resolver_.request_info().size());
// Using kNetworkAnonymizationKey again should result in the first cached
// address list.
result.clear();
EXPECT_THAT(Resolve("example.com", kNetworkAnonymizationKey, &result),
IsOk());
EXPECT_EQ(addresses1, result);
EXPECT_EQ(2u, mock_resolver_.request_info().size());
}
TEST_F(HostResolverMojoTest, Multiple) {
std::vector<net::IPAddress> addresses;
addresses.emplace_back(1, 2, 3, 4);
mock_resolver_.AddAction(HostResolverAction::ReturnResult(addresses));
mock_resolver_.AddAction(
HostResolverAction::ReturnError(net::ERR_NAME_NOT_RESOLVED));
std::unique_ptr<ProxyHostResolver::Request> request1 =
resolver_.CreateRequest("example.com",
net::ProxyResolveDnsOperation::DNS_RESOLVE_EX,
net::NetworkAnonymizationKey());
std::unique_ptr<ProxyHostResolver::Request> request2 =
resolver_.CreateRequest("example.org",
net::ProxyResolveDnsOperation::DNS_RESOLVE_EX,
net::NetworkAnonymizationKey());
net::TestCompletionCallback callback1;
net::TestCompletionCallback callback2;
ASSERT_EQ(net::ERR_IO_PENDING, request1->Start(callback1.callback()));
ASSERT_EQ(net::ERR_IO_PENDING, request2->Start(callback2.callback()));
EXPECT_THAT(callback1.GetResult(net::ERR_IO_PENDING), IsOk());
EXPECT_THAT(callback2.GetResult(net::ERR_IO_PENDING),
IsError(net::ERR_NAME_NOT_RESOLVED));
EXPECT_EQ(addresses, request1->GetResults());
ASSERT_EQ(0u, request2->GetResults().size());
EXPECT_THAT(mock_resolver_.request_info(),
testing::ElementsAre(
MockMojoHostResolver::RequestInfo{
"example.com", net::NetworkAnonymizationKey()},
MockMojoHostResolver::RequestInfo{
"example.org", net::NetworkAnonymizationKey()}));
}
TEST_F(HostResolverMojoTest, Error) {
mock_resolver_.AddAction(
HostResolverAction::ReturnError(net::ERR_NAME_NOT_RESOLVED));
std::vector<net::IPAddress> result;
EXPECT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsError(net::ERR_NAME_NOT_RESOLVED));
EXPECT_TRUE(result.empty());
ASSERT_EQ(1u, mock_resolver_.request_info().size());
EXPECT_EQ("example.com", mock_resolver_.request_info()[0].hostname);
}
TEST_F(HostResolverMojoTest, EmptyResult) {
mock_resolver_.AddAction(HostResolverAction::ReturnError(net::OK));
std::vector<net::IPAddress> result;
EXPECT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsOk());
EXPECT_TRUE(result.empty());
ASSERT_EQ(1u, mock_resolver_.request_info().size());
}
TEST_F(HostResolverMojoTest, Cancel) {
mock_resolver_.AddAction(HostResolverAction::RetainRequest());
std::unique_ptr<ProxyHostResolver::Request> request = resolver_.CreateRequest(
"example.com", net::ProxyResolveDnsOperation::DNS_RESOLVE_EX,
net::NetworkAnonymizationKey());
request->Start(base::BindOnce(&Fail));
request.reset();
waiter_.WaitForEvent(ConnectionErrorSource::REQUEST);
ASSERT_EQ(1u, mock_resolver_.request_info().size());
EXPECT_EQ("example.com", mock_resolver_.request_info()[0].hostname);
}
TEST_F(HostResolverMojoTest, ImplDropsClientConnection) {
mock_resolver_.AddAction(HostResolverAction::DropRequest());
std::vector<net::IPAddress> result;
EXPECT_THAT(Resolve("example.com", net::NetworkAnonymizationKey(), &result),
IsError(net::ERR_FAILED));
EXPECT_TRUE(result.empty());
ASSERT_EQ(1u, mock_resolver_.request_info().size());
EXPECT_EQ("example.com", mock_resolver_.request_info()[0].hostname);
}
} // namespace proxy_resolver