blob: 2820d29dcba3f6623919e8d79e4a22c7504a281b [file] [log] [blame]
// Copyright 2019 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 "patchpanel/shill_client.h"
#include <memory>
#include <set>
#include <string>
#include <utility>
#include <vector>
#include <chromeos/dbus/service_constants.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include "patchpanel/fake_shill_client.h"
namespace patchpanel {
class ShillClientTest : public testing::Test {
protected:
void SetUp() override {
helper_ = std::make_unique<FakeShillClientHelper>();
client_ = helper_->FakeClient();
client_->RegisterDefaultDeviceChangedHandler(base::Bind(
&ShillClientTest::DefaultDeviceChangedHandler, base::Unretained(this)));
client_->RegisterDevicesChangedHandler(base::Bind(
&ShillClientTest::DevicesChangedHandler, base::Unretained(this)));
client_->RegisterIPConfigsChangedHandler(base::Bind(
&ShillClientTest::IPConfigsChangedHandler, base::Unretained(this)));
default_ifname_.clear();
added_.clear();
removed_.clear();
}
void DefaultDeviceChangedHandler(const ShillClient::Device& new_device,
const ShillClient::Device& prev_device) {
default_ifname_ = new_device.ifname;
}
void DevicesChangedHandler(const std::set<std::string>& added,
const std::set<std::string>& removed) {
added_ = added;
removed_ = removed;
}
void IPConfigsChangedHandler(const std::string& device,
const ShillClient::IPConfig& ipconfig) {
ipconfig_change_calls_.emplace_back(std::make_pair(device, ipconfig));
}
protected:
std::string default_ifname_;
std::set<std::string> added_;
std::set<std::string> removed_;
std::vector<std::pair<std::string, ShillClient::IPConfig>>
ipconfig_change_calls_;
std::unique_ptr<FakeShillClient> client_;
std::unique_ptr<FakeShillClientHelper> helper_;
};
TEST_F(ShillClientTest, DevicesChangedHandlerCalledOnDevicesPropertyChange) {
std::vector<dbus::ObjectPath> devices = {dbus::ObjectPath("/device/eth0"),
dbus::ObjectPath("/device/wlan0")};
auto value = brillo::Any(devices);
client_->SetFakeDefaultDevice("eth0");
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
EXPECT_EQ(added_.size(), devices.size());
EXPECT_NE(added_.find("eth0"), added_.end());
EXPECT_NE(added_.find("wlan0"), added_.end());
EXPECT_EQ(removed_.size(), 0);
// Implies the default callback was run;
EXPECT_EQ(default_ifname_, "eth0");
EXPECT_NE(added_.find(default_ifname_), added_.end());
devices.pop_back();
devices.emplace_back(dbus::ObjectPath("/device/eth1"));
value = brillo::Any(devices);
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
EXPECT_EQ(added_.size(), 1);
EXPECT_EQ(*added_.begin(), "eth1");
EXPECT_EQ(removed_.size(), 1);
EXPECT_EQ(*removed_.begin(), "wlan0");
}
TEST_F(ShillClientTest, VerifyDevicesPrefixStripped) {
std::vector<dbus::ObjectPath> devices = {dbus::ObjectPath("/device/eth0")};
auto value = brillo::Any(devices);
client_->SetFakeDefaultDevice("eth0");
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
EXPECT_EQ(added_.size(), 1);
EXPECT_EQ(*added_.begin(), "eth0");
// Implies the default callback was run;
EXPECT_EQ(default_ifname_, "eth0");
}
TEST_F(ShillClientTest, DefaultDeviceChangedHandlerCalledOnNewDefaultDevice) {
client_->SetFakeDefaultDevice("eth0");
client_->NotifyManagerPropertyChange(shill::kDefaultServiceProperty,
brillo::Any() /* ignored */);
EXPECT_EQ(default_ifname_, "eth0");
client_->SetFakeDefaultDevice("wlan0");
client_->NotifyManagerPropertyChange(shill::kDefaultServiceProperty,
brillo::Any() /* ignored */);
EXPECT_EQ(default_ifname_, "wlan0");
}
TEST_F(ShillClientTest, DefaultDeviceChangedHandlerNotCalledForSameDefault) {
client_->SetFakeDefaultDevice("eth0");
client_->NotifyManagerPropertyChange(shill::kDefaultServiceProperty,
brillo::Any() /* ignored */);
EXPECT_EQ(default_ifname_, "eth0");
default_ifname_.clear();
client_->NotifyManagerPropertyChange(shill::kDefaultServiceProperty,
brillo::Any() /* ignored */);
// Implies the callback was not run the second time.
EXPECT_EQ(default_ifname_, "");
}
TEST_F(ShillClientTest, DefaultDeviceChanges) {
// One network device appears.
std::vector<dbus::ObjectPath> devices = {dbus::ObjectPath("/device/wlan0")};
auto value = brillo::Any(devices);
client_->SetFakeDefaultDevice("wlan0");
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
EXPECT_EQ(default_ifname_, "wlan0");
// A second device appears.
default_ifname_.clear();
devices = {dbus::ObjectPath("/device/eth0"),
dbus::ObjectPath("/device/wlan0")};
value = brillo::Any(devices);
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
EXPECT_EQ(default_ifname_, "");
// The second device becomes the default interface.
client_->SetFakeDefaultDevice("eth0");
client_->NotifyManagerPropertyChange(shill::kDefaultServiceProperty,
brillo::Any() /* ignored */);
EXPECT_EQ(default_ifname_, "eth0");
// The first device disappears.
devices = {dbus::ObjectPath("/device/eth0")};
value = brillo::Any(devices);
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
// The default device is still the same.
EXPECT_EQ(default_ifname_, "eth0");
// All devices have disappeared.
devices = {};
value = brillo::Any(devices);
client_->SetFakeDefaultDevice("");
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
EXPECT_EQ(default_ifname_, "");
}
TEST_F(ShillClientTest, ListenToDeviceChangeSignalOnNewDevices) {
// Adds a device.
std::vector<dbus::ObjectPath> devices = {dbus::ObjectPath("/device/wlan0")};
auto value = brillo::Any(devices);
EXPECT_CALL(*helper_->mock_proxy(),
DoConnectToSignal(shill::kFlimflamDeviceInterface,
shill::kMonitorPropertyChanged, _, _))
.Times(1);
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
// Adds another device. DoConnectToSignal() called only for the new added one.
devices = {dbus::ObjectPath("/device/wlan0"),
dbus::ObjectPath("/device/eth0")};
value = brillo::Any(devices);
EXPECT_CALL(*helper_->mock_proxy(),
DoConnectToSignal(shill::kFlimflamDeviceInterface,
shill::kMonitorPropertyChanged, _, _))
.Times(1);
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, value);
}
TEST_F(ShillClientTest, TriggerOnIPConfigsChangeHandlerOnce) {
// Adds a device.
std::vector<dbus::ObjectPath> devices = {dbus::ObjectPath("/device/wlan0")};
auto devices_value = brillo::Any(devices);
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, devices_value);
client_->NotifyDevicePropertyChange("wlan0", shill::kIPConfigsProperty,
brillo::Any());
ASSERT_EQ(ipconfig_change_calls_.size(), 1u);
EXPECT_EQ(ipconfig_change_calls_.back().first, "wlan0");
// Removes the device and adds it again.
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, brillo::Any());
client_->NotifyManagerPropertyChange(shill::kDevicesProperty, devices_value);
client_->NotifyDevicePropertyChange("wlan0", shill::kIPConfigsProperty,
brillo::Any());
ASSERT_EQ(ipconfig_change_calls_.size(), 2u);
EXPECT_EQ(ipconfig_change_calls_.back().first, "wlan0");
}
} // namespace patchpanel