blob: 6849b91e08ce351492107761a46be49ad4392ecb [file] [log] [blame]
// Copyright 2018 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "shill/ipconfig.h"
#include <sys/time.h>
#include <vector>
#include <base/bind.h>
#include <chromeos/dbus/service_constants.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include "shill/logging.h"
#include "shill/mock_adaptors.h"
#include "shill/mock_control.h"
#include "shill/mock_log.h"
#include "shill/net/mock_time.h"
#include "shill/static_ip_parameters.h"
using base::Bind;
using base::Unretained;
using testing::_;
using testing::DoAll;
using testing::EndsWith;
using testing::Mock;
using testing::Return;
using testing::SetArgPointee;
using testing::Test;
namespace shill {
namespace {
const char kDeviceName[] = "testdevice";
const uint32_t kTimeNow = 10;
} // namespace
class IPConfigTest : public Test {
public:
IPConfigTest() : ipconfig_(new IPConfig(&control_, kDeviceName)) {
ipconfig_->time_ = &time_;
}
void SetUp() override {
ScopeLogger::GetInstance()->EnableScopesByName("inet");
ScopeLogger::GetInstance()->set_verbose_level(3);
}
void TearDown() override {
ScopeLogger::GetInstance()->EnableScopesByName("-inet");
ScopeLogger::GetInstance()->set_verbose_level(0);
}
void DropRef(const IPConfigRefPtr& /*ipconfig*/,
bool /*new_lease_acquired*/) {
ipconfig_ = nullptr;
}
MOCK_METHOD(void, OnIPConfigUpdated, (const IPConfigRefPtr&, bool));
MOCK_METHOD(void, OnIPConfigFailed, (const IPConfigRefPtr&));
MOCK_METHOD(void, OnIPConfigRefreshed, (const IPConfigRefPtr&));
MOCK_METHOD(void, OnIPConfigExpired, (const IPConfigRefPtr&));
protected:
IPConfigMockAdaptor* GetAdaptor() {
return static_cast<IPConfigMockAdaptor*>(ipconfig_->adaptor_.get());
}
void UpdateProperties(const IPConfig::Properties& properties) {
ipconfig_->UpdateProperties(properties, true);
}
void NotifyFailure() { ipconfig_->NotifyFailure(); }
void NotifyExpiry() { ipconfig_->NotifyExpiry(); }
void ExpectPropertiesEqual(const IPConfig::Properties& properties) {
EXPECT_EQ(properties.address, ipconfig_->properties().address);
EXPECT_EQ(properties.subnet_prefix, ipconfig_->properties().subnet_prefix);
EXPECT_EQ(properties.broadcast_address,
ipconfig_->properties().broadcast_address);
EXPECT_EQ(properties.dns_servers.size(),
ipconfig_->properties().dns_servers.size());
if (properties.dns_servers.size() ==
ipconfig_->properties().dns_servers.size()) {
for (size_t i = 0; i < properties.dns_servers.size(); ++i) {
EXPECT_EQ(properties.dns_servers[i],
ipconfig_->properties().dns_servers[i]);
}
}
EXPECT_EQ(properties.domain_search.size(),
ipconfig_->properties().domain_search.size());
if (properties.domain_search.size() ==
ipconfig_->properties().domain_search.size()) {
for (size_t i = 0; i < properties.domain_search.size(); ++i) {
EXPECT_EQ(properties.domain_search[i],
ipconfig_->properties().domain_search[i]);
}
}
EXPECT_EQ(properties.gateway, ipconfig_->properties().gateway);
EXPECT_EQ(properties.blackhole_ipv6,
ipconfig_->properties().blackhole_ipv6);
EXPECT_EQ(properties.mtu, ipconfig_->properties().mtu);
}
MockControl control_;
MockTime time_;
IPConfigRefPtr ipconfig_;
};
TEST_F(IPConfigTest, DeviceName) {
EXPECT_EQ(kDeviceName, ipconfig_->device_name());
}
TEST_F(IPConfigTest, RequestIP) {
EXPECT_FALSE(ipconfig_->RequestIP());
}
TEST_F(IPConfigTest, RenewIP) {
EXPECT_FALSE(ipconfig_->RenewIP());
}
TEST_F(IPConfigTest, ReleaseIP) {
EXPECT_FALSE(ipconfig_->ReleaseIP(IPConfig::kReleaseReasonDisconnect));
}
TEST_F(IPConfigTest, SetBlackholedUids) {
std::vector<uint32_t> uids = {1000, 216};
std::vector<uint32_t> empty_uids = {};
// SetBlackholedUids returns true if the value changes
EXPECT_TRUE(ipconfig_->SetBlackholedUids(uids));
EXPECT_EQ(uids, ipconfig_->properties().blackholed_uids);
// SetBlackholeBrowserTraffic returns false if the value does not change
EXPECT_FALSE(ipconfig_->SetBlackholedUids(uids));
EXPECT_EQ(uids, ipconfig_->properties().blackholed_uids);
EXPECT_TRUE(ipconfig_->ClearBlackholedUids());
EXPECT_EQ(empty_uids, ipconfig_->properties().blackholed_uids);
EXPECT_FALSE(ipconfig_->ClearBlackholedUids());
EXPECT_EQ(empty_uids, ipconfig_->properties().blackholed_uids);
}
TEST_F(IPConfigTest, UpdateProperties) {
IPConfig::Properties properties;
properties.address = "1.2.3.4";
properties.subnet_prefix = 24;
properties.broadcast_address = "11.22.33.44";
properties.dns_servers = {"10.20.30.40", "20.30.40.50"};
properties.domain_name = "foo.org";
properties.domain_search = {"zoo.org", "zoo.com"};
properties.gateway = "5.6.7.8";
properties.blackhole_ipv6 = true;
properties.mtu = 700;
UpdateProperties(properties);
ExpectPropertiesEqual(properties);
// We should not reset on NotifyFailure.
NotifyFailure();
ExpectPropertiesEqual(properties);
// We should not reset on NotifyExpiry.
NotifyExpiry();
ExpectPropertiesEqual(properties);
// We should reset if ResetProperties is called.
ipconfig_->ResetProperties();
ExpectPropertiesEqual(IPConfig::Properties());
}
TEST_F(IPConfigTest, Callbacks) {
ipconfig_->RegisterUpdateCallback(
Bind(&IPConfigTest::OnIPConfigUpdated, Unretained(this)));
ipconfig_->RegisterFailureCallback(
Bind(&IPConfigTest::OnIPConfigFailed, Unretained(this)));
ipconfig_->RegisterRefreshCallback(
Bind(&IPConfigTest::OnIPConfigRefreshed, Unretained(this)));
ipconfig_->RegisterExpireCallback(
Bind(&IPConfigTest::OnIPConfigExpired, Unretained(this)));
EXPECT_CALL(*this, OnIPConfigUpdated(ipconfig_, true));
EXPECT_CALL(*this, OnIPConfigFailed(ipconfig_)).Times(0);
EXPECT_CALL(*this, OnIPConfigRefreshed(ipconfig_)).Times(0);
EXPECT_CALL(*this, OnIPConfigExpired(ipconfig_)).Times(0);
UpdateProperties(IPConfig::Properties());
Mock::VerifyAndClearExpectations(this);
EXPECT_CALL(*this, OnIPConfigUpdated(ipconfig_, true)).Times(0);
EXPECT_CALL(*this, OnIPConfigFailed(ipconfig_));
EXPECT_CALL(*this, OnIPConfigRefreshed(ipconfig_)).Times(0);
EXPECT_CALL(*this, OnIPConfigExpired(ipconfig_)).Times(0);
NotifyFailure();
Mock::VerifyAndClearExpectations(this);
EXPECT_CALL(*this, OnIPConfigUpdated(ipconfig_, true)).Times(0);
EXPECT_CALL(*this, OnIPConfigFailed(ipconfig_)).Times(0);
EXPECT_CALL(*this, OnIPConfigRefreshed(ipconfig_));
EXPECT_CALL(*this, OnIPConfigExpired(ipconfig_)).Times(0);
ipconfig_->Refresh();
Mock::VerifyAndClearExpectations(this);
EXPECT_CALL(*this, OnIPConfigUpdated(ipconfig_, true)).Times(0);
EXPECT_CALL(*this, OnIPConfigFailed(ipconfig_)).Times(0);
EXPECT_CALL(*this, OnIPConfigRefreshed(ipconfig_)).Times(0);
EXPECT_CALL(*this, OnIPConfigExpired(ipconfig_));
NotifyExpiry();
Mock::VerifyAndClearExpectations(this);
}
TEST_F(IPConfigTest, UpdatePropertiesWithDropRef) {
// The UpdateCallback should be able to drop a reference to the
// IPConfig object without crashing.
ipconfig_->RegisterUpdateCallback(
Bind(&IPConfigTest::DropRef, Unretained(this)));
UpdateProperties(IPConfig::Properties());
}
TEST_F(IPConfigTest, PropertyChanges) {
IPConfigMockAdaptor* adaptor = GetAdaptor();
StaticIPParameters static_ip_params;
EXPECT_CALL(*adaptor, EmitStringChanged(kAddressProperty, _));
EXPECT_CALL(*adaptor, EmitStringsChanged(kNameServersProperty, _));
ipconfig_->ApplyStaticIPParameters(&static_ip_params);
Mock::VerifyAndClearExpectations(adaptor);
EXPECT_CALL(*adaptor, EmitStringChanged(kAddressProperty, _));
EXPECT_CALL(*adaptor, EmitStringsChanged(kNameServersProperty, _));
ipconfig_->RestoreSavedIPParameters(&static_ip_params);
Mock::VerifyAndClearExpectations(adaptor);
IPConfig::Properties ip_properties;
EXPECT_CALL(*adaptor, EmitStringChanged(kAddressProperty, _));
EXPECT_CALL(*adaptor, EmitStringsChanged(kNameServersProperty, _));
UpdateProperties(ip_properties);
Mock::VerifyAndClearExpectations(adaptor);
// It is the callback's responsibility for resetting the IPConfig
// properties (via IPConfig::ResetProperties()). Since NotifyFailure
// by itself doesn't change any properties, it should not emit any
// property change events either.
EXPECT_CALL(*adaptor, EmitStringChanged(_, _)).Times(0);
EXPECT_CALL(*adaptor, EmitStringsChanged(_, _)).Times(0);
NotifyFailure();
Mock::VerifyAndClearExpectations(adaptor);
// Similarly, NotifyExpiry() should have no property change side effects.
EXPECT_CALL(*adaptor, EmitStringChanged(_, _)).Times(0);
EXPECT_CALL(*adaptor, EmitStringsChanged(_, _)).Times(0);
NotifyExpiry();
Mock::VerifyAndClearExpectations(adaptor);
EXPECT_CALL(*adaptor, EmitStringChanged(kAddressProperty, _));
EXPECT_CALL(*adaptor, EmitStringsChanged(kNameServersProperty, _));
ipconfig_->ResetProperties();
Mock::VerifyAndClearExpectations(adaptor);
}
TEST_F(IPConfigTest, UpdateLeaseExpirationTime) {
const struct timeval expected_time_now = {kTimeNow, 0};
uint32_t lease_duration = 1;
EXPECT_CALL(time_, GetTimeBoottime(_))
.WillOnce(DoAll(SetArgPointee<0>(expected_time_now), Return(0)));
ipconfig_->UpdateLeaseExpirationTime(lease_duration);
EXPECT_EQ(kTimeNow + lease_duration,
ipconfig_->current_lease_expiration_time_.tv_sec);
}
TEST_F(IPConfigTest, TimeToLeaseExpiry_NoDHCPLease) {
ScopedMockLog log;
uint32_t time_left = 0;
// |current_lease_expiration_time_| has not been set, so expect an error.
EXPECT_CALL(log, Log(_, _, EndsWith("No current DHCP lease")));
EXPECT_FALSE(ipconfig_->TimeToLeaseExpiry(&time_left));
EXPECT_EQ(0, time_left);
}
TEST_F(IPConfigTest, TimeToLeaseExpiry_CurrentLeaseExpired) {
ScopedMockLog log;
const struct timeval time_now = {kTimeNow, 0};
uint32_t time_left = 0;
// Set |current_lease_expiration_time_| so it is expired (i.e. earlier than
// current time).
ipconfig_->current_lease_expiration_time_ = {kTimeNow - 1, 0};
EXPECT_CALL(time_, GetTimeBoottime(_))
.WillOnce(DoAll(SetArgPointee<0>(time_now), Return(0)));
EXPECT_CALL(log,
Log(_, _, EndsWith("Current DHCP lease has already expired")));
EXPECT_FALSE(ipconfig_->TimeToLeaseExpiry(&time_left));
EXPECT_EQ(0, time_left);
}
TEST_F(IPConfigTest, TimeToLeaseExpiry_Success) {
const uint32_t expected_time_to_expiry = 10;
const struct timeval time_now = {kTimeNow, 0};
uint32_t time_left;
// Set |current_lease_expiration_time_| so it appears like we already
// have obtained a DHCP lease before.
ipconfig_->current_lease_expiration_time_ = {
kTimeNow + expected_time_to_expiry, 0};
EXPECT_CALL(time_, GetTimeBoottime(_))
.WillOnce(DoAll(SetArgPointee<0>(time_now), Return(0)));
EXPECT_TRUE(ipconfig_->TimeToLeaseExpiry(&time_left));
EXPECT_EQ(expected_time_to_expiry, time_left);
}
} // namespace shill