| // Copyright 2018 The ChromiumOS Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "shill/cellular/cellular.h" |
| |
| #include <linux/if.h> // NOLINT - Needs typedefs from sys/socket.h. |
| #include <linux/netlink.h> |
| #include <sys/socket.h> |
| |
| #include <map> |
| #include <memory> |
| #include <string> |
| #include <utility> |
| |
| #include <base/check.h> |
| #include <base/check_op.h> |
| #include <base/containers/contains.h> |
| #include <base/functional/bind.h> |
| #include <base/memory/scoped_refptr.h> |
| #include <base/strings/string_number_conversions.h> |
| #include <chromeos/dbus/service_constants.h> |
| #include <chromeos/dbus/shill/dbus-constants.h> |
| #include <gtest/gtest.h> |
| #include <net-base/ipv4_address.h> |
| #include <net-base/mock_process_manager.h> |
| #include <net-base/mock_rtnl_handler.h> |
| #include <net-base/network_config.h> |
| |
| extern "C" { |
| // A struct member in pppd.h has the name 'class'. |
| #define class class_num |
| // pppd.h defines a bool type. |
| #define bool pppd_bool_t |
| #include <pppd/pppd.h> |
| #undef bool |
| #undef class |
| } |
| |
| #include "shill/cellular/apn_list.h" |
| #include "shill/cellular/cellular_bearer.h" |
| #include "shill/cellular/cellular_capability_3gpp.h" |
| #include "shill/cellular/cellular_consts.h" |
| #include "shill/cellular/cellular_service.h" |
| #include "shill/cellular/cellular_service_provider.h" |
| #include "shill/cellular/mock_cellular_service.h" |
| #include "shill/cellular/mock_mm1_modem_location_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_modem3gpp_profile_manager_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_modem3gpp_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_signal_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_simple_proxy.h" |
| #include "shill/cellular/mock_mobile_operator_info.h" |
| #include "shill/cellular/mock_modem_info.h" |
| #include "shill/dbus/dbus_properties_proxy.h" |
| #include "shill/dbus/fake_properties_proxy.h" |
| #include "shill/dbus-constants.h" |
| #include "shill/device_id.h" |
| #include "shill/error.h" |
| #include "shill/manager.h" |
| #include "shill/metrics.h" |
| #include "shill/mock_adaptors.h" |
| #include "shill/mock_control.h" |
| #include "shill/mock_device_info.h" |
| #include "shill/mock_manager.h" |
| #include "shill/mock_metrics.h" |
| #include "shill/mock_profile.h" |
| #include "shill/mock_virtual_device.h" |
| #include "shill/network/mock_network.h" |
| #include "shill/ppp_daemon.h" |
| #include "shill/rpc_task.h" // for RpcTaskDelegate |
| #include "shill/service.h" |
| #include "shill/store/fake_store.h" |
| #include "shill/store/property_store_test.h" |
| #include "shill/test_event_dispatcher.h" |
| #include "shill/testing.h" |
| |
| using testing::_; |
| using testing::AnyNumber; |
| using testing::AtLeast; |
| using testing::DoAll; |
| using testing::Eq; |
| using testing::Field; |
| using testing::Invoke; |
| using testing::Mock; |
| using testing::NiceMock; |
| using testing::Optional; |
| using testing::Pointee; |
| using testing::Return; |
| using testing::ReturnRef; |
| using testing::SetArgPointee; |
| using testing::StrEq; |
| using testing::WithArg; |
| |
| namespace shill { |
| |
| namespace { |
| MATCHER_P(IsWeakPtrTo, address, "") { |
| return arg.get() == address; |
| } |
| MATCHER_P(KeyValueStoreHasApn, expected_apn, "") { |
| return arg.template Contains<std::string>(CellularBearer::kMMApnProperty) && |
| expected_apn == |
| arg.template Get<std::string>(CellularBearer::kMMApnProperty); |
| } |
| constexpr int kTestInterfaceIndex = 3; |
| constexpr char kTestInterfaceName[] = "wwan0"; |
| constexpr net_base::MacAddress kTestInterfaceAddress0( |
| 0x00, 0x01, 0x02, 0x03, 0x04, 0x05); |
| constexpr net_base::MacAddress kTestInterfaceAddress1( |
| 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff); |
| constexpr int kTestMultiplexedInterfaceIndex = 4; |
| constexpr int kTestMultiplexedInterfaceIndex2 = 5; |
| constexpr char kTestMultiplexedInterfaceName[] = "wwan0mux0"; |
| constexpr char kTestMultiplexedInterfaceName2[] = "wwan0mux1"; |
| constexpr char kDBusService[] = "org.freedesktop.ModemManager1"; |
| constexpr char kModemUid[] = "uid"; |
| constexpr char kIccid[] = "1234567890000"; |
| const RpcIdentifier kTestModemDBusPath( |
| "/org/freedesktop/ModemManager1/Modem/0"); |
| const RpcIdentifier kTestBearerDBusPath( |
| "/org/freedesktop/ModemManager1/Bearer/0"); |
| const RpcIdentifier kTestBearerDBusPath2( |
| "/org/freedesktop/ModemManager1/Bearer/1"); |
| } // namespace |
| |
| class CellularPropertyTest : public PropertyStoreTest { |
| public: |
| CellularPropertyTest() |
| : device_(new Cellular(manager(), |
| kTestInterfaceName, |
| kTestInterfaceAddress0, |
| kTestInterfaceIndex, |
| "", |
| RpcIdentifier(""))) {} |
| |
| ~CellularPropertyTest() { device_ = nullptr; } |
| |
| protected: |
| CellularRefPtr device_; |
| }; |
| |
| TEST_F(CellularPropertyTest, Contains) { |
| EXPECT_TRUE(device_->store().Contains(kNameProperty)); |
| EXPECT_FALSE(device_->store().Contains("")); |
| } |
| |
| TEST_F(CellularPropertyTest, SetProperty) { |
| { |
| Error error; |
| device_->mutable_store()->SetAnyProperty( |
| kCellularPolicyAllowRoamingProperty, false, &error); |
| EXPECT_TRUE(error.IsSuccess()); |
| } |
| // Ensure that attempting to write a R/O property returns InvalidArgs error. |
| { |
| Error error; |
| device_->mutable_store()->SetAnyProperty( |
| kAddressProperty, PropertyStoreTest::kStringV, &error); |
| EXPECT_TRUE(error.IsFailure()); |
| EXPECT_EQ(Error::kInvalidArguments, error.type()); |
| } |
| } |
| |
| class CellularTest : public testing::Test { |
| public: |
| CellularTest() |
| : control_interface_(this), |
| manager_(&control_interface_, &dispatcher_, &metrics_), |
| modem_info_(&control_interface_, &manager_), |
| mock_mobile_operator_info_(nullptr), |
| profile_(new NiceMock<MockProfile>(&manager_)) { |
| cellular_service_provider_.set_profile_for_testing(profile_); |
| } |
| |
| ~CellularTest() = default; |
| |
| void SetUp() override { |
| shill::ScopeLogger::GetInstance()->set_verbose_level(0); |
| shill::ScopeLogger::GetInstance()->EnableScopesByName("cellular"); |
| |
| EXPECT_CALL(manager_, modem_info()).WillRepeatedly(Return(&modem_info_)); |
| device_ = |
| new Cellular(&manager_, kTestInterfaceName, kTestInterfaceAddress0, |
| kTestInterfaceIndex, kDBusService, kTestModemDBusPath); |
| PopulateProxies(); |
| metrics_.RegisterDevice(device_->interface_index(), Technology::kCellular); |
| |
| static_cast<Device*>(device_.get())->rtnl_handler_ = &rtnl_handler_; |
| device_->process_manager_ = &process_manager_; |
| |
| EXPECT_CALL(manager_, DeregisterService(_)).Times(AnyNumber()); |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(_, _)) |
| .WillRepeatedly(Return(PendingActivationStore::kStateActivated)); |
| EXPECT_CALL(manager_, cellular_service_provider()) |
| .WillRepeatedly(Return(&cellular_service_provider_)); |
| EXPECT_CALL(*profile_, GetConstStorage()) |
| .WillRepeatedly(Return(&profile_storage_)); |
| EXPECT_CALL(*profile_, GetStorage()) |
| .WillRepeatedly(Return(&profile_storage_)); |
| } |
| |
| MockDeviceInfo* device_info() { return manager_.mock_device_info(); } |
| |
| void TearDown() override { |
| metrics_.DeregisterDevice(device_->interface_index()); |
| device_->set_state_for_testing(Cellular::State::kDisabled); |
| auto capability = GetCapability3gpp(); |
| if (capability) |
| capability->ReleaseProxies(); |
| // Break cycle between Cellular and CellularService. |
| device_->service_ = nullptr; |
| device_->SelectService(nullptr); |
| device_ = nullptr; |
| } |
| |
| void CreatePropertiesProxy() { |
| dbus_properties_proxy_ = |
| DBusPropertiesProxy::CreateDBusPropertiesProxyForTesting( |
| std::make_unique<FakePropertiesProxy>()); |
| FakePropertiesProxy* fake_properties = static_cast<FakePropertiesProxy*>( |
| dbus_properties_proxy_->GetDBusPropertiesProxyForTesting()); |
| // Ensure that GetAll calls to MM_DBUS_INTERFACE_MODEM and |
| // MM_DBUS_INTERFACE_MODEM_MODEM3GPP succeed and return a valid dictionary. |
| fake_properties->SetDictionaryForTesting(MM_DBUS_INTERFACE_MODEM, |
| brillo::VariantDictionary()); |
| fake_properties->SetDictionaryForTesting(MM_DBUS_INTERFACE_MODEM_MODEM3GPP, |
| brillo::VariantDictionary()); |
| // Set the Device property so that StartModem succeeds. |
| fake_properties->SetForTesting(modemmanager::kModemManager1ModemInterface, |
| MM_MODEM_PROPERTY_DEVICE, |
| brillo::Any(std::string(kModemUid))); |
| } |
| |
| void PopulateProxies() { |
| CreatePropertiesProxy(); |
| mm1_modem_location_proxy_.reset(new mm1::MockModemLocationProxy()); |
| mm1_modem_3gpp_proxy_.reset(new mm1::MockModemModem3gppProxy()); |
| mm1_modem_3gpp_profile_manager_proxy_.reset( |
| new mm1::MockModemModem3gppProfileManagerProxy()); |
| mm1_modem_proxy_.reset(new mm1::MockModemProxy()); |
| mm1_signal_proxy_.reset(new mm1::MockModemSignalProxy()); |
| mm1_simple_proxy_.reset(new mm1::MockModemSimpleProxy()); |
| } |
| |
| void SetMockMobileOperatorInfoObjects() { |
| mock_mobile_operator_info_ = |
| new NiceMock<MockMobileOperatorInfo>(&dispatcher_, "Test"); |
| // Takes ownership. |
| device_->set_mobile_operator_info_for_testing(mock_mobile_operator_info_); |
| } |
| |
| void InvokeEnable(bool enable, |
| ResultCallback callback, |
| base::TimeDelta timeout) { |
| std::move(callback).Run(Error(Error::kSuccess)); |
| } |
| void InvokeConnect(const KeyValueStore& props, |
| RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(Service::kStateAssociating, device_->service_->state()); |
| std::move(callback).Run(kTestBearerDBusPath, Error()); |
| } |
| void InvokeConnectFail(const KeyValueStore& props, |
| RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(Service::kStateAssociating, device_->service_->state()); |
| std::move(callback).Run(RpcIdentifier(), Error(Error::kNotOnHomeNetwork)); |
| } |
| void InvokeDisconnect(const RpcIdentifier& bearer, |
| ResultCallback callback, |
| base::TimeDelta timeout) { |
| if (!callback.is_null()) |
| std::move(callback).Run(Error()); |
| } |
| void InvokeDisconnectFail(const RpcIdentifier& bearer, |
| ResultCallback callback, |
| base::TimeDelta timeout) { |
| if (!callback.is_null()) |
| std::move(callback).Run(Error(Error::kOperationFailed)); |
| } |
| void InvokeList(ResultVariantDictionariesOnceCallback callback, |
| base::TimeDelta timeout) { |
| std::move(callback).Run(VariantDictionaries(), Error()); |
| } |
| |
| void ExpectDisconnectCapability3gpp() { |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| GetCapability3gpp()->modem_simple_proxy_.reset(mm1_simple_proxy_.release()); |
| } |
| |
| void VerifyDisconnect() { |
| EXPECT_EQ(Cellular::State::kRegistered, device_->state()); |
| } |
| |
| void StartPPP(int pid) { |
| EXPECT_CALL(process_manager_, StartProcess(_, _, _, _, _, _, _)) |
| .WillOnce(Return(pid)); |
| device_->StartPPP("fake_serial_device"); |
| EXPECT_FALSE(device_->selected_service()); |
| EXPECT_FALSE(device_->is_ppp_authenticating_); |
| EXPECT_NE(nullptr, device_->ppp_task_); |
| Mock::VerifyAndClearExpectations(&process_manager_); |
| } |
| |
| void FakeUpConnectedPPP() { |
| const char ifname[] = "fake-ppp-device"; |
| const int ifindex = 1; |
| auto mock_ppp_device = base::MakeRefCounted<MockVirtualDevice>( |
| &manager_, ifname, ifindex, Technology::kPPP); |
| device_->ppp_device_ = mock_ppp_device; |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| } |
| |
| void ExpectPPPStopped() { |
| auto mock_ppp_device = |
| static_cast<MockVirtualDevice*>(device_->ppp_device_.get()); |
| EXPECT_CALL(*mock_ppp_device, DropConnection()); |
| } |
| |
| void VerifyPPPStopped() { |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| EXPECT_FALSE(device_->ppp_device_); |
| } |
| |
| mm1::MockModemProxy* SetModemProxyExpectations() { |
| EXPECT_CALL(*mm1_modem_proxy_, set_state_changed_callback(_)) |
| .Times(AnyNumber()); |
| return mm1_modem_proxy_.get(); |
| } |
| |
| mm1::MockModemModem3gppProfileManagerProxy* |
| SetModem3gppProfileManagerProxyExpectations() { |
| EXPECT_CALL(*mm1_modem_3gpp_profile_manager_proxy_, SetUpdatedCallback(_)) |
| .Times(AnyNumber()); |
| return mm1_modem_3gpp_profile_manager_proxy_.get(); |
| } |
| |
| mm1::MockModemProxy* SetupOnAfterResume() { |
| EXPECT_CALL(manager_, UpdateEnabledTechnologies()).Times(AnyNumber()); |
| EXPECT_CALL(*static_cast<DeviceMockAdaptor*>(device_->adaptor()), |
| EmitBoolChanged(_, _)) |
| .Times(AnyNumber()); |
| return SetModemProxyExpectations(); |
| } |
| |
| void VerifyOperatorMap(const Stringmap& operator_map, |
| const std::string& code, |
| const std::string& name, |
| const std::string& country) { |
| Stringmap::const_iterator it; |
| Stringmap::const_iterator endit = operator_map.end(); |
| |
| it = operator_map.find(kOperatorCodeKey); |
| if (code == "") { |
| EXPECT_EQ(endit, it); |
| } else { |
| ASSERT_NE(endit, it); |
| EXPECT_EQ(code, it->second); |
| } |
| it = operator_map.find(kOperatorNameKey); |
| if (name == "") { |
| EXPECT_EQ(endit, it); |
| } else { |
| ASSERT_NE(endit, it); |
| EXPECT_EQ(name, it->second); |
| } |
| it = operator_map.find(kOperatorCountryKey); |
| if (country == "") { |
| EXPECT_EQ(endit, it); |
| } else { |
| ASSERT_NE(endit, it); |
| EXPECT_EQ(country, it->second); |
| } |
| } |
| |
| void CallStartModemCallback(const Error& error, const Error& expected_error) { |
| base::test::TestFuture<const Error&> future; |
| device_->StartModemCallback(future.GetCallback(), error); |
| EXPECT_EQ(future.Get<Error>().type(), expected_error.type()); |
| } |
| |
| void CallStopModemCallback(const Error& error) { |
| base::test::TestFuture<const Error&> future; |
| device_->StopModemCallback(future.GetCallback(), error); |
| EXPECT_EQ(future.Get<Error>().type(), error.type()); |
| } |
| |
| void CallSetPrimarySimProperties(const Cellular::SimProperties& properties) { |
| device_->SetPrimarySimProperties(properties); |
| } |
| |
| void CallSetSimSlotProperties( |
| const std::vector<Cellular::SimProperties>& properties, size_t primary) { |
| device_->SetSimSlotProperties(properties, static_cast<int>(primary)); |
| } |
| |
| void CallSetSimProperties( |
| const std::vector<Cellular::SimProperties>& properties, size_t primary) { |
| device_->SetSimProperties(properties, static_cast<int>(primary)); |
| } |
| |
| static net_base::NetworkConfig GetExpectedNetworkConfigFromPPPConfig( |
| std::map<std::string, std::string>& ppp_config) { |
| net_base::NetworkConfig network_config = |
| PPPDaemon::ParseNetworkConfig(ppp_config); |
| network_config.ipv6_blackhole_route = false; |
| return network_config; |
| } |
| |
| void SetStateDisconnected(Cellular::State state) { |
| CHECK(state != Cellular::State::kLinked && |
| state != Cellular::State::kConnected); |
| device_->set_state_for_testing(state); |
| } |
| |
| void SetStateConnected(Cellular::State state) { |
| CHECK(state == Cellular::State::kLinked || |
| state == Cellular::State::kConnected); |
| device_->set_state_for_testing(state); |
| } |
| |
| protected: |
| class TestControl : public MockControl { |
| public: |
| explicit TestControl(CellularTest* test) : test_(test) {} |
| |
| std::unique_ptr<DBusPropertiesProxy> CreateDBusPropertiesProxy( |
| const RpcIdentifier& path, const std::string& service) override { |
| std::unique_ptr<DBusPropertiesProxy> proxy = |
| std::move(test_->dbus_properties_proxy_); |
| |
| // Replace properties for subsequent requests. |
| test_->CreatePropertiesProxy(); |
| return proxy; |
| } |
| |
| std::unique_ptr<mm1::ModemLocationProxyInterface> |
| CreateMM1ModemLocationProxy(const RpcIdentifier& path, |
| const std::string& service) override { |
| if (!test_->mm1_modem_location_proxy_) { |
| test_->mm1_modem_location_proxy_.reset( |
| new mm1::MockModemLocationProxy()); |
| } |
| return std::move(test_->mm1_modem_location_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemModem3gppProxyInterface> |
| CreateMM1ModemModem3gppProxy(const RpcIdentifier& path, |
| const std::string& service) override { |
| if (!test_->mm1_modem_3gpp_proxy_) |
| test_->mm1_modem_3gpp_proxy_.reset(new mm1::MockModemModem3gppProxy()); |
| return std::move(test_->mm1_modem_3gpp_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemModem3gppProfileManagerProxyInterface> |
| CreateMM1ModemModem3gppProfileManagerProxy( |
| const RpcIdentifier& path, const std::string& service) override { |
| if (!test_->mm1_modem_3gpp_profile_manager_proxy_) |
| test_->mm1_modem_3gpp_profile_manager_proxy_.reset( |
| new mm1::MockModemModem3gppProfileManagerProxy()); |
| return std::move(test_->mm1_modem_3gpp_profile_manager_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemProxyInterface> CreateMM1ModemProxy( |
| const RpcIdentifier& path, const std::string& service) override { |
| if (!test_->mm1_modem_proxy_) |
| test_->mm1_modem_proxy_.reset(new mm1::MockModemProxy()); |
| if (!on_modem_proxy_created_callback_.is_null()) |
| on_modem_proxy_created_callback_.Run(test_->mm1_modem_proxy_.get()); |
| return std::move(test_->mm1_modem_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemSimpleProxyInterface> CreateMM1ModemSimpleProxy( |
| const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| if (!test_->mm1_simple_proxy_) |
| test_->mm1_simple_proxy_.reset(new mm1::MockModemSimpleProxy()); |
| return std::move(test_->mm1_simple_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemSignalProxyInterface> CreateMM1ModemSignalProxy( |
| const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| if (!test_->mm1_signal_proxy_) |
| test_->mm1_signal_proxy_.reset(new mm1::MockModemSignalProxy()); |
| return std::move(test_->mm1_signal_proxy_); |
| } |
| |
| void SetOnModemProxyCreatedCallback( |
| base::RepeatingCallback<void(mm1::MockModemProxy*)> |
| on_modem_proxy_created_callback) { |
| on_modem_proxy_created_callback_ = |
| std::move(on_modem_proxy_created_callback); |
| } |
| |
| private: |
| CellularTest* test_; |
| |
| // Callbacks to set expectations for newly-created proxies |
| base::RepeatingCallback<void(mm1::MockModemProxy*)> |
| on_modem_proxy_created_callback_; |
| }; |
| |
| void AllowCreateGsmCardProxyFromFactory() { |
| create_gsm_card_proxy_from_factory_ = true; |
| } |
| |
| CellularCapability3gpp* GetCapability3gpp() { |
| return device_->capability_for_testing(); |
| } |
| |
| // Different tests simulate a cellular service being set using a real /mock |
| // service. |
| CellularService* SetService() { |
| device_->service_ = new CellularService( |
| &manager_, device_->imsi(), device_->iccid(), device_->GetSimCardId()); |
| device_->service_->SetDevice(device_.get()); |
| storage_id_ = device_->service_->GetStorageIdentifier(); |
| profile_storage_.SetString(storage_id_, CellularService::kStorageType, |
| kTypeCellular); |
| profile_storage_.SetString(storage_id_, CellularService::kStorageIccid, |
| device_->iccid()); |
| profile_storage_.SetString(storage_id_, CellularService::kStorageImsi, |
| device_->imsi()); |
| return device_->service_.get(); |
| } |
| MockCellularService* SetMockService() { |
| device_->service_ = new NiceMock<MockCellularService>(&manager_, device_); |
| return static_cast<MockCellularService*>(device_->service_.get()); |
| } |
| |
| void SetCapability3gppActiveBearer(ApnList::ApnType apn_type, |
| std::unique_ptr<CellularBearer> bearer) { |
| GetCapability3gpp()->active_bearers_[apn_type] = std::move(bearer); |
| } |
| |
| void SetCapability3gppModemSimpleProxy() { |
| GetCapability3gpp()->modem_simple_proxy_ = std::move(mm1_simple_proxy_); |
| } |
| |
| void SetCapability3gppRegistrationState( |
| const MMModem3gppRegistrationState registration_state) { |
| GetCapability3gpp()->registration_state_ = registration_state; |
| } |
| |
| void Capability3gppCallOnProfilesChanged( |
| const CellularCapability3gpp::Profiles& profiles) { |
| GetCapability3gpp()->OnProfilesChanged(profiles); |
| } |
| |
| void InitCapability3gppProxies() { GetCapability3gpp()->InitProxies(); } |
| |
| CellularService* SetRegisteredWithService() { |
| device_->set_iccid_for_testing(kIccid); |
| device_->set_state_for_testing(Cellular::State::kRegistered); |
| device_->set_modem_state_for_testing(Cellular::kModemStateRegistered); |
| CellularService* service = SetService(); |
| cellular_service_provider_.LoadServicesForDevice(device_.get()); |
| return service; |
| } |
| |
| void SetInhibited(bool inhibited) { |
| device_->SetInhibited(inhibited, /*error=*/nullptr); |
| } |
| |
| void SetScanning(bool scanning) { device_->SetScanningProperty(scanning); } |
| |
| EventDispatcherForTest dispatcher_; |
| TestControl control_interface_; |
| NiceMock<MockManager> manager_; |
| NiceMock<MockMetrics> metrics_; |
| MockModemInfo modem_info_; |
| NiceMock<net_base::MockProcessManager> process_manager_; |
| NiceMock<net_base::MockRTNLHandler> rtnl_handler_; |
| |
| bool create_gsm_card_proxy_from_factory_; |
| std::unique_ptr<DBusPropertiesProxy> dbus_properties_proxy_; |
| std::unique_ptr<mm1::MockModemModem3gppProxy> mm1_modem_3gpp_proxy_; |
| std::unique_ptr<mm1::MockModemModem3gppProfileManagerProxy> |
| mm1_modem_3gpp_profile_manager_proxy_; |
| std::unique_ptr<mm1::MockModemLocationProxy> mm1_modem_location_proxy_; |
| std::unique_ptr<mm1::MockModemProxy> mm1_modem_proxy_; |
| std::unique_ptr<mm1::MockModemSignalProxy> mm1_signal_proxy_; |
| std::unique_ptr<mm1::MockModemSimpleProxy> mm1_simple_proxy_; |
| MockMobileOperatorInfo* mock_mobile_operator_info_; |
| CellularRefPtr device_; |
| MockNetwork* default_pdn_ = nullptr; // owned by |device_| |
| MockNetwork* tethering_pdn_ = nullptr; // owned by |device_| |
| CellularServiceProvider cellular_service_provider_{&manager_}; |
| std::string storage_id_; |
| FakeStore profile_storage_; |
| scoped_refptr<NiceMock<MockProfile>> profile_; |
| }; |
| |
| TEST_F(CellularTest, GetStorageIdentifier) { |
| EXPECT_EQ("device_wwan0", device_->GetStorageIdentifier()); |
| } |
| |
| TEST_F(CellularTest, HomeProviderServingOperator) { |
| // Must be std::string so that we can safely ReturnRef. |
| std::string kHomeProviderCode("10001"); |
| std::string kHomeProviderCountry("us"); |
| std::string kHomeProviderName("HomeProviderName"); |
| std::string kServingOperatorCode("10002"); |
| std::string kServingOperatorCountry("ca"); |
| std::string kServingOperatorName("ServingOperatorName"); |
| |
| // Test that the the home provider information is correctly updated under |
| // different scenarios w.r.t. information about the mobile network operators. |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| Stringmap home_provider; |
| Stringmap serving_operator; |
| |
| InitCapability3gppProxies(); |
| |
| // (1) Neither home provider nor serving operator known. |
| EXPECT_CALL(*mock_mobile_operator_info_, IsHomeOperatorKnown()) |
| .WillRepeatedly(Return(false)); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| IsServingMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(false)); |
| |
| device_->CreateServices(); |
| |
| home_provider = device_->home_provider(); |
| VerifyOperatorMap(home_provider, "", "", ""); |
| serving_operator = device_->service_->serving_operator(); |
| VerifyOperatorMap(serving_operator, "", "", ""); |
| Mock::VerifyAndClearExpectations(mock_mobile_operator_info_); |
| device_->DestroyAllServices(); |
| |
| // (2) serving operator known. |
| // When home provider is not known, serving operator proxies in. |
| EXPECT_CALL(*mock_mobile_operator_info_, IsHomeOperatorKnown()) |
| .WillRepeatedly(Return(false)); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| IsServingMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_mobile_operator_info_, serving_mccmnc()) |
| .WillRepeatedly(ReturnRef(kServingOperatorCode)); |
| EXPECT_CALL(*mock_mobile_operator_info_, serving_operator_name()) |
| .WillRepeatedly(ReturnRef(kServingOperatorName)); |
| EXPECT_CALL(*mock_mobile_operator_info_, serving_country()) |
| .WillRepeatedly(ReturnRef(kServingOperatorCountry)); |
| |
| device_->CreateServices(); |
| |
| home_provider = device_->home_provider(); |
| VerifyOperatorMap(home_provider, kServingOperatorCode, kServingOperatorName, |
| kServingOperatorCountry); |
| serving_operator = device_->service_->serving_operator(); |
| VerifyOperatorMap(serving_operator, kServingOperatorCode, |
| kServingOperatorName, kServingOperatorCountry); |
| Mock::VerifyAndClearExpectations(mock_mobile_operator_info_); |
| device_->DestroyAllServices(); |
| |
| // (3) home provider known. |
| // When serving operator is not known, home provider proxies in. |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| IsServingMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(false)); |
| EXPECT_CALL(*mock_mobile_operator_info_, IsHomeOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_mobile_operator_info_, mccmnc()) |
| .WillRepeatedly(ReturnRef(kHomeProviderCode)); |
| EXPECT_CALL(*mock_mobile_operator_info_, operator_name()) |
| .WillRepeatedly(ReturnRef(kHomeProviderName)); |
| EXPECT_CALL(*mock_mobile_operator_info_, country()) |
| .WillRepeatedly(ReturnRef(kHomeProviderCountry)); |
| |
| device_->CreateServices(); |
| |
| home_provider = device_->home_provider(); |
| VerifyOperatorMap(home_provider, kHomeProviderCode, kHomeProviderName, |
| kHomeProviderCountry); |
| serving_operator = device_->service_->serving_operator(); |
| VerifyOperatorMap(serving_operator, kHomeProviderCode, kHomeProviderName, |
| kHomeProviderCountry); |
| Mock::VerifyAndClearExpectations(mock_mobile_operator_info_); |
| device_->DestroyAllServices(); |
| |
| // (4) Serving operator known, home provider known. |
| EXPECT_CALL(*mock_mobile_operator_info_, IsHomeOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_mobile_operator_info_, mccmnc()) |
| .WillRepeatedly(ReturnRef(kHomeProviderCode)); |
| EXPECT_CALL(*mock_mobile_operator_info_, operator_name()) |
| .WillRepeatedly(ReturnRef(kHomeProviderName)); |
| EXPECT_CALL(*mock_mobile_operator_info_, country()) |
| .WillRepeatedly(ReturnRef(kHomeProviderCountry)); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| IsServingMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_mobile_operator_info_, serving_mccmnc()) |
| .WillRepeatedly(ReturnRef(kServingOperatorCode)); |
| EXPECT_CALL(*mock_mobile_operator_info_, serving_operator_name()) |
| .WillRepeatedly(ReturnRef(kServingOperatorName)); |
| EXPECT_CALL(*mock_mobile_operator_info_, serving_country()) |
| .WillRepeatedly(ReturnRef(kServingOperatorCountry)); |
| |
| device_->CreateServices(); |
| |
| home_provider = device_->home_provider(); |
| VerifyOperatorMap(home_provider, kHomeProviderCode, kHomeProviderName, |
| kHomeProviderCountry); |
| serving_operator = device_->service_->serving_operator(); |
| VerifyOperatorMap(serving_operator, kServingOperatorCode, |
| kServingOperatorName, kServingOperatorCountry); |
| } |
| |
| TEST_F(CellularTest, SetPrimarySimProperties) { |
| // The default storage identifier should always be cellular_{iccid} |
| Cellular::SimProperties sim_properties; |
| sim_properties.eid = "test_eid"; |
| sim_properties.iccid = "test_iccid"; |
| sim_properties.imsi = "test_imsi"; |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kEidProperty, sim_properties.eid)) |
| .Times(1); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kIccidProperty, sim_properties.iccid)) |
| .Times(1); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kImsiProperty, sim_properties.imsi)) |
| .Times(1); |
| CallSetPrimarySimProperties(sim_properties); |
| EXPECT_EQ("test_eid", device_->eid()); |
| EXPECT_EQ("test_iccid", device_->iccid()); |
| EXPECT_EQ("test_imsi", device_->imsi()); |
| } |
| |
| TEST_F(CellularTest, SetSimSlotProperties) { |
| std::vector<Cellular::SimProperties> slot_properties = { |
| {0, "iccid1", "eid1", "operator_id1", "spn1", "imsi1"}, |
| {1, "iccid2", "eid2", "operator_id2", "spn2", "imsi2"}, |
| }; |
| KeyValueStore expected1, expected2; |
| expected1.Set(kSIMSlotInfoEID, slot_properties[0].eid); |
| expected1.Set(kSIMSlotInfoICCID, slot_properties[0].iccid); |
| expected1.Set(kSIMSlotInfoPrimary, false); |
| expected2.Set(kSIMSlotInfoEID, slot_properties[1].eid); |
| expected2.Set(kSIMSlotInfoICCID, slot_properties[1].iccid); |
| expected2.Set(kSIMSlotInfoPrimary, true); |
| |
| KeyValueStores expected; |
| expected.push_back(expected1); |
| expected.push_back(expected2); |
| EXPECT_CALL(*static_cast<DeviceMockAdaptor*>(device_->adaptor()), |
| EmitKeyValueStoresChanged(kSIMSlotInfoProperty, expected)) |
| .Times(1); |
| CallSetSimSlotProperties(slot_properties, 1u); |
| |
| // Set the primary slot to 0 and ensure that a SimSlots properties change is |
| // emitted. |
| expected1.Set(kSIMSlotInfoPrimary, true); |
| expected2.Set(kSIMSlotInfoPrimary, false); |
| expected.clear(); |
| expected.push_back(expected1); |
| expected.push_back(expected2); |
| EXPECT_CALL(*static_cast<DeviceMockAdaptor*>(device_->adaptor()), |
| EmitKeyValueStoresChanged(kSIMSlotInfoProperty, expected)) |
| .Times(1); |
| CallSetSimSlotProperties(slot_properties, 0u); |
| } |
| |
| TEST_F(CellularTest, StorageIdentifier) { |
| // The default storage identifier should always be cellular_{iccid} |
| InitCapability3gppProxies(); |
| Cellular::SimProperties sim_properties; |
| sim_properties.iccid = "test_iccid"; |
| sim_properties.imsi = "test_imsi"; |
| CallSetPrimarySimProperties(sim_properties); |
| device_->CreateServices(); |
| EXPECT_EQ("cellular_test_iccid", device_->service()->GetStorageIdentifier()); |
| device_->DestroyAllServices(); |
| } |
| |
| TEST_F(CellularTest, Connect) { |
| Error error; |
| device_->set_state_for_testing(Cellular::State::kModemStarted); |
| SetService(); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Error::kAlreadyConnected, error.type()); |
| error.Populate(Error::kSuccess); |
| |
| error.Reset(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Error::kAlreadyConnected, error.type()); |
| |
| error.Reset(); |
| device_->set_state_for_testing(Cellular::State::kModemStarted); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Error::kNotRegistered, error.type()); |
| |
| error.Reset(); |
| device_->set_state_for_testing(Cellular::State::kDisabled); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Error::kOperationFailed, error.type()); |
| |
| error.Reset(); |
| device_->set_state_for_testing(Cellular::State::kRegistered); |
| device_->service_->allow_roaming_ = false; |
| device_->service_->roaming_state_ = kRoamingStateRoaming; |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Error::kNotOnHomeNetwork, error.type()); |
| |
| // Check that connect fails if policy restricts roaming |
| error.Reset(); |
| device_->service_->allow_roaming_ = true; |
| device_->policy_allow_roaming_ = false; |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Error::kNotOnHomeNetwork, error.type()); |
| device_->policy_allow_roaming_ = true; |
| |
| // Common state for the successful connection attempts |
| device_->set_skip_establish_link_for_testing(true); |
| error.Populate(Error::kSuccess); |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Connect(_, _, CellularCapability3gpp::kTimeoutConnect)) |
| .Times(3) |
| .WillRepeatedly(Invoke(this, &CellularTest::InvokeConnect)); |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Connection at home network |
| device_->service_->roaming_state_ = kRoamingStateHome; |
| device_->set_state_for_testing(Cellular::State::kRegistered); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_TRUE(error.IsSuccess()); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kConnected, device_->state()); |
| |
| // Connection at roaming network |
| device_->service_->allow_roaming_ = true; |
| device_->service_->roaming_state_ = kRoamingStateRoaming; |
| device_->set_state_for_testing(Cellular::State::kRegistered); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_TRUE(error.IsSuccess()); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kConnected, device_->state()); |
| |
| // Check that provider_requires_roaming_ will override all other roaming |
| // settings |
| device_->service_->allow_roaming_ = false; |
| device_->policy_allow_roaming_ = false; |
| device_->provider_requires_roaming_ = true; |
| device_->service_->roaming_state_ = kRoamingStateRoaming; |
| device_->set_state_for_testing(Cellular::State::kRegistered); |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_TRUE(error.IsSuccess()); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kConnected, device_->state()); |
| } |
| |
| TEST_F(CellularTest, SimSlotSwitch) { |
| // Only provide a SIM in the second slot. Setup capability with all sim |
| // properties. |
| std::vector<Cellular::SimProperties> slot_properties = { |
| {0, "", "eid1", "", "", ""}, |
| {1, "unknown-iccid", "", "", "", ""}, |
| }; |
| base::flat_map<RpcIdentifier, Cellular::SimProperties> sim_properties; |
| sim_properties[RpcIdentifier("sim_path1")] = slot_properties[0]; |
| sim_properties[RpcIdentifier("sim_path2")] = slot_properties[1]; |
| GetCapability3gpp()->set_sim_properties_for_testing(sim_properties); |
| |
| // Simulate creation of capability and enabling the modem. |
| mm1::MockModemProxy* mm1_modem_proxy = SetModemProxyExpectations(); |
| EXPECT_CALL(*mm1_modem_proxy, SetPrimarySimSlot(2u, _, _)); |
| EXPECT_CALL(*mm1_modem_proxy, Enable(true, _, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeEnable)); |
| InitCapability3gppProxies(); |
| device_->SetEnabled(true); |
| device_->set_state_for_testing(Cellular::State::kModemStarted); |
| CallSetSimProperties(slot_properties, 0u); |
| |
| // Call Connect on secondary slot |
| Error error; |
| device_->Connect( |
| cellular_service_provider_.FindService("unknown-iccid").get(), &error); |
| EXPECT_TRUE(error.IsSuccess()); |
| dispatcher_.DispatchPendingEvents(); |
| |
| // Simulate MM state changes that occur when a slot switch occurs |
| device_->OnModemStateChanged(Cellular::kModemStateDisabling); |
| dispatcher_.DispatchPendingEvents(); |
| device_->OnModemStateChanged(Cellular::kModemStateDisabled); |
| dispatcher_.DispatchPendingEvents(); |
| device_->OnModemDestroyed(); |
| // Check that existing services aren't destroyed even though the modem DBus |
| // object is |
| EXPECT_TRUE(cellular_service_provider_.FindService("unknown-iccid")); |
| |
| // Simulate MM changes that occur when a new MM DBus object appears after a |
| // slot switch |
| device_->UpdateModemProperties(kTestModemDBusPath, kTestInterfaceAddress1); |
| device_->OnModemStateChanged(Cellular::kModemStateDisabled); |
| slot_properties[1].iccid = "8900000000000000000", |
| GetCapability3gpp()->set_sim_properties_for_testing(sim_properties); |
| CallSetSimProperties(slot_properties, 1u); |
| device_->set_state_for_testing(Cellular::State::kModemStarted); |
| device_->OnModemStateChanged(Cellular::kModemStateEnabling); |
| dispatcher_.DispatchPendingEvents(); |
| device_->OnModemStateChanged(Cellular::kModemStateEnabled); |
| dispatcher_.DispatchPendingEvents(); |
| device_->OnModemStateChanged(Cellular::kModemStateRegistered); |
| dispatcher_.DispatchPendingEvents(); |
| |
| // Cellular should call Connect once MM's 3GPP interface updates it's |
| // registration state |
| SetCapability3gppRegistrationState(MM_MODEM_3GPP_REGISTRATION_STATE_HOME); |
| PopulateProxies(); |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)); |
| SetCapability3gppModemSimpleProxy(); |
| device_->HandleNewRegistrationState(); |
| constexpr base::TimeDelta kTestTimeout = |
| Cellular::kPendingConnectDelay + base::Seconds(10); |
| dispatcher_.task_environment().FastForwardBy(kTestTimeout); |
| } |
| |
| TEST_F(CellularTest, Disconnect) { |
| Error error; |
| SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kRegistered); |
| device_->Disconnect(&error, "in test"); |
| EXPECT_EQ(Error::kNotConnected, error.type()); |
| error.Reset(); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| EXPECT_CALL(*default_pdn_, Stop()); |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Disconnect(_, _, CellularCapability3gpp::kTimeoutDisconnect)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| SetCapability3gppModemSimpleProxy(); |
| device_->Disconnect(&error, "in test"); |
| EXPECT_TRUE(error.IsSuccess()); |
| EXPECT_EQ(Cellular::State::kRegistered, device_->state()); |
| EXPECT_EQ(nullptr, device_->default_pdn_for_testing()); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, DisconnectFailure) { |
| SetRegisteredWithService(); |
| // Test the case where the underlying modem state is set |
| // to disconnecting, but shill thinks it's still connected |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Disconnect(_, _, CellularCapability3gpp::kTimeoutDisconnect)) |
| .Times(2) |
| .WillRepeatedly(Invoke(this, &CellularTest::InvokeDisconnectFail)); |
| SetCapability3gppModemSimpleProxy(); |
| device_->set_modem_state_for_testing(Cellular::kModemStateDisconnecting); |
| Error error; |
| device_->Disconnect(&error, "in test"); |
| EXPECT_EQ(Cellular::State::kConnected, device_->state()); |
| EXPECT_NE(nullptr, device_->default_pdn_for_testing()); |
| |
| device_->set_modem_state_for_testing(Cellular::kModemStateConnected); |
| device_->Disconnect(&error, "in test"); |
| EXPECT_EQ(Cellular::State::kRegistered, device_->state()); |
| EXPECT_EQ(nullptr, device_->default_pdn_for_testing()); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, ConnectFailure) { |
| SetRegisteredWithService(); |
| ASSERT_EQ(Service::kStateIdle, device_->service_->state()); |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Connect(_, _, CellularCapability3gpp::kTimeoutConnect)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeConnectFail)); |
| SetCapability3gppModemSimpleProxy(); |
| Error error; |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_EQ(Service::kStateFailure, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, ConnectWhileInhibited) { |
| SetRegisteredWithService(); |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)).Times(0); |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Connect while inhibited should fail. |
| SetInhibited(true); |
| Error error; |
| device_->Connect(device_->service().get(), &error); |
| EXPECT_FALSE(error.IsSuccess()); |
| EXPECT_EQ(Error::kWrongState, error.type()); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, PendingConnect) { |
| CellularService* service = SetRegisteredWithService(); |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillRepeatedly(Invoke(this, &CellularTest::InvokeConnect)); |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Connect while scanning should set a pending connect. |
| SetScanning(true); |
| Error error; |
| service->Connect(&error, "test"); |
| EXPECT_TRUE(error.IsSuccess()); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_NE(device_->state(), Cellular::State::kConnected); |
| EXPECT_EQ(device_->connect_pending_iccid(), service->iccid()); |
| |
| // Setting scanning to false should connect to the pending iccid. |
| SetScanning(false); |
| // Fast forward the task environment by the pending connect delay plus |
| // time to complete the connect. |
| constexpr base::TimeDelta kTestTimeout = |
| Cellular::kPendingConnectDelay + base::Seconds(10); |
| dispatcher_.task_environment().FastForwardBy(kTestTimeout); |
| EXPECT_EQ(device_->state(), Cellular::State::kConnected); |
| EXPECT_TRUE(device_->connect_pending_iccid().empty()); |
| } |
| |
| TEST_F(CellularTest, PendingDisconnect) { |
| CellularService* service = SetRegisteredWithService(); |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillRepeatedly(Invoke(this, &CellularTest::InvokeConnect)); |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Connect while scanning should set a pending connect. |
| SetScanning(true); |
| Error error; |
| service->Connect(&error, "test"); |
| EXPECT_TRUE(error.IsSuccess()); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_NE(device_->state(), Cellular::State::kConnected); |
| EXPECT_EQ(device_->connect_pending_iccid(), service->iccid()); |
| |
| // Disconnecting from the service should cancel the pending connect. |
| service->Disconnect(&error, "test"); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_TRUE(device_->connect_pending_iccid().empty()); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| // TODO(b/232177767): Add a test to verify that Cellular start the Network with |
| // the correct options. |
| |
| TEST_F(CellularTest, ModemStateChangeValidConnected) { |
| device_->set_state_for_testing(Cellular::State::kEnabled); |
| device_->set_skip_establish_link_for_testing(true); |
| device_->set_modem_state_for_testing(Cellular::kModemStateConnecting); |
| SetService(); |
| device_->OnModemStateChanged(Cellular::kModemStateConnected); |
| // A change of the modem state only won't make the shill cellular object |
| // transition to the connected state. The first transition to Connected |
| // will exclusively happen once the connection attempt is finished and |
| // reported as successful. |
| EXPECT_NE(Cellular::State::kConnected, device_->state()); |
| } |
| |
| TEST_F(CellularTest, ModemStateChangeLostRegistration) { |
| CellularCapability3gpp* capability = GetCapability3gpp(); |
| capability->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_HOME; |
| EXPECT_TRUE(capability->IsRegistered()); |
| device_->set_modem_state_for_testing(Cellular::kModemStateRegistered); |
| device_->OnModemStateChanged(Cellular::kModemStateEnabled); |
| EXPECT_FALSE(capability->IsRegistered()); |
| } |
| |
| TEST_F(CellularTest, StartModemCallback) { |
| device_->set_state_for_testing(Cellular::State::kEnabled); |
| CallStartModemCallback(Error(Error::kSuccess), Error(Error::kSuccess)); |
| EXPECT_EQ(device_->state(), Cellular::State::kModemStarted); |
| } |
| |
| TEST_F(CellularTest, StartModemCallbackFail) { |
| device_->set_state_for_testing(Cellular::State::kEnabled); |
| CallStartModemCallback(Error(Error::kOperationFailed), |
| Error(Error::kOperationFailed)); |
| EXPECT_EQ(device_->state(), Cellular::State::kEnabled); |
| } |
| |
| TEST_F(CellularTest, StartModemCallbackFailWrongState) { |
| device_->set_state_for_testing(Cellular::State::kEnabled); |
| // Wrong state error gets ignored. |
| CallStartModemCallback(Error(Error::kWrongState), Error(Error::kSuccess)); |
| EXPECT_EQ(device_->state(), Cellular::State::kEnabled); |
| } |
| |
| TEST_F(CellularTest, StopModemCallback) { |
| SetMockService(); |
| CallStopModemCallback(Error(Error::kSuccess)); |
| EXPECT_EQ(device_->state(), Cellular::State::kDisabled); |
| } |
| |
| TEST_F(CellularTest, StopModemCallbackFail) { |
| SetMockService(); |
| CallStopModemCallback(Error(Error::kOperationFailed)); |
| EXPECT_EQ(device_->state(), Cellular::State::kDisabled); |
| } |
| |
| TEST_F(CellularTest, SetPolicyAllowRoaming) { |
| EXPECT_TRUE(device_->policy_allow_roaming_); |
| EXPECT_CALL(manager_, UpdateDevice(_)); |
| Error error; |
| device_->SetPolicyAllowRoaming(false, &error); |
| EXPECT_TRUE(error.IsSuccess()); |
| error.Reset(); |
| EXPECT_FALSE(device_->GetPolicyAllowRoaming(&error)); |
| EXPECT_TRUE(error.IsSuccess()); |
| } |
| |
| TEST_F(CellularTest, SetInhibited) { |
| PopulateProxies(); |
| |
| // Invoke Cellular::StartModemCallback() to simulate the modem starting, which |
| // is required before SetInhibit can succeed. |
| CallStartModemCallback(Error(Error::kSuccess), Error(Error::kSuccess)); |
| |
| EXPECT_FALSE(device_->inhibited()); |
| SetInhibited(true); |
| EXPECT_TRUE(device_->inhibited()); |
| } |
| |
| class TestRpcTaskDelegate : public RpcTaskDelegate, |
| public base::SupportsWeakPtr<TestRpcTaskDelegate> { |
| public: |
| virtual void GetLogin(std::string* user, std::string* password) {} |
| virtual void Notify(const std::string& reason, |
| const std::map<std::string, std::string>& dict) {} |
| }; |
| |
| TEST_F(CellularTest, StartPPP) { |
| const int kPID = 234; |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| StartPPP(kPID); |
| } |
| |
| TEST_F(CellularTest, StartPPPAlreadyStarted) { |
| const int kPID = 234; |
| StartPPP(kPID); |
| |
| const int kPID2 = 235; |
| StartPPP(kPID2); |
| } |
| |
| TEST_F(CellularTest, StartPPPAfterEthernetUp) { |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| CellularService* service(SetService()); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->SelectService(service); |
| |
| const int kPID = 234; |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| StartPPP(kPID); |
| EXPECT_EQ(Cellular::State::kLinked, device_->state()); |
| } |
| |
| TEST_F(CellularTest, GetLogin) { |
| // Doesn't crash when there is no service. |
| std::string username_to_pppd; |
| std::string password_to_pppd; |
| EXPECT_FALSE(device_->service()); |
| device_->GetLogin(&username_to_pppd, &password_to_pppd); |
| |
| // Provides expected username and password in normal case. |
| const char kFakeUsername[] = "fake-user"; |
| const char kFakePassword[] = "fake-password"; |
| CellularService& service(*SetService()); |
| service.ppp_username_ = kFakeUsername; |
| service.ppp_password_ = kFakePassword; |
| device_->GetLogin(&username_to_pppd, &password_to_pppd); |
| } |
| |
| TEST_F(CellularTest, Notify) { |
| // Common setup. |
| const int kPID = 91; |
| SetMockService(); |
| StartPPP(kPID); |
| |
| const std::map<std::string, std::string> kEmptyArgs; |
| device_->Notify(kPPPReasonAuthenticating, kEmptyArgs); |
| EXPECT_TRUE(device_->is_ppp_authenticating_); |
| device_->Notify(kPPPReasonAuthenticated, kEmptyArgs); |
| EXPECT_FALSE(device_->is_ppp_authenticating_); |
| |
| // Normal connect. |
| const std::string ifname1 = "fake-device"; |
| const int ifindex1 = 1; |
| auto ppp_device1 = |
| new MockVirtualDevice(&manager_, ifname1, ifindex1, Technology::kPPP); |
| std::map<std::string, std::string> ppp_config; |
| ppp_config[kPPPInterfaceName] = ifname1; |
| EXPECT_CALL(*device_info(), GetIndex(ifname1)).WillOnce(Return(ifindex1)); |
| EXPECT_CALL(*device_info(), CreatePPPDevice(_, StrEq(ifname1), ifindex1)) |
| .WillOnce(Return(ppp_device1)); |
| EXPECT_CALL(*device_info(), |
| RegisterDevice(static_cast<DeviceRefPtr>(ppp_device1))); |
| EXPECT_CALL(*ppp_device1, SetEnabled(true)); |
| EXPECT_CALL(*ppp_device1, |
| SelectService(static_cast<ServiceRefPtr>(device_->service_))); |
| EXPECT_CALL(*ppp_device1, |
| UpdateNetworkConfig( |
| Pointee(GetExpectedNetworkConfigFromPPPConfig(ppp_config)))); |
| device_->Notify(kPPPReasonConnect, ppp_config); |
| Mock::VerifyAndClearExpectations(device_info()); |
| Mock::VerifyAndClearExpectations(ppp_device1); |
| |
| // Re-connect on same network device: if pppd sends us multiple connect |
| // events, we behave rationally. |
| EXPECT_CALL(*device_info(), GetIndex(ifname1)).WillOnce(Return(ifindex1)); |
| EXPECT_CALL(*device_info(), CreatePPPDevice(_, _, _)).Times(0); |
| EXPECT_CALL(*device_info(), RegisterDevice(_)).Times(0); |
| EXPECT_CALL(*ppp_device1, SetEnabled(true)); |
| EXPECT_CALL(*ppp_device1, |
| SelectService(static_cast<ServiceRefPtr>(device_->service_))); |
| EXPECT_CALL(*ppp_device1, |
| UpdateNetworkConfig( |
| Pointee(GetExpectedNetworkConfigFromPPPConfig(ppp_config)))); |
| device_->Notify(kPPPReasonConnect, ppp_config); |
| Mock::VerifyAndClearExpectations(device_info()); |
| Mock::VerifyAndClearExpectations(ppp_device1); |
| |
| // Re-connect on new network device: if we still have the PPPDevice |
| // from a prior connect, this new connect should DTRT. This is |
| // probably an unlikely case. |
| const std::string ifname2 = "fake-device2"; |
| const int ifindex2 = 2; |
| auto ppp_device2 = |
| new MockVirtualDevice(&manager_, ifname2, ifindex2, Technology::kPPP); |
| std::map<std::string, std::string> ppp_config2; |
| ppp_config2[kPPPInterfaceName] = ifname2; |
| EXPECT_CALL(*device_info(), GetIndex(ifname2)).WillOnce(Return(ifindex2)); |
| EXPECT_CALL(*device_info(), CreatePPPDevice(_, StrEq(ifname2), ifindex2)) |
| .WillOnce(Return(ppp_device2)); |
| EXPECT_CALL(*device_info(), |
| RegisterDevice(static_cast<DeviceRefPtr>(ppp_device2))); |
| EXPECT_CALL(*ppp_device1, SelectService(ServiceRefPtr(nullptr))); |
| EXPECT_CALL(*ppp_device2, SetEnabled(true)); |
| EXPECT_CALL(*ppp_device2, |
| SelectService(static_cast<ServiceRefPtr>(device_->service_))); |
| EXPECT_CALL(*ppp_device2, |
| UpdateNetworkConfig( |
| Pointee(GetExpectedNetworkConfigFromPPPConfig(ppp_config2)))); |
| device_->Notify(kPPPReasonConnect, ppp_config2); |
| Mock::VerifyAndClearExpectations(device_info()); |
| Mock::VerifyAndClearExpectations(ppp_device1); |
| Mock::VerifyAndClearExpectations(ppp_device2); |
| |
| // Disconnect should report no failure, since we had a |
| // Notify(kPPPReasonAuthenticated, ...) and got no error from pppd. |
| EXPECT_CALL(*ppp_device2, SetServiceFailure(Service::kFailureNone)); |
| device_->OnPPPDied(kPID, EXIT_OK); |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| |
| // |Cellular::ppp_task_| is destroyed on the task loop. Must dispatch once to |
| // cleanup. |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularTest, PPPConnectionFailedBeforeAuth) { |
| // Test that we properly set Service state in the case where pppd |
| // disconnects before authenticating (as opposed to the Notify test, |
| // where pppd disconnects after connecting). |
| const int kPID = 52; |
| const std::map<std::string, std::string> kEmptyArgs; |
| MockCellularService* service = SetMockService(); |
| StartPPP(kPID); |
| |
| ExpectDisconnectCapability3gpp(); |
| EXPECT_CALL(*service, SetFailure(Service::kFailureUnknown)); |
| device_->OnPPPDied(kPID, EXIT_FATAL_ERROR); |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| VerifyDisconnect(); |
| |
| // |Cellular::ppp_task_| is destroyed on the task loop. Must dispatch once to |
| // cleanup. |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularTest, PPPConnectionFailedDuringAuth) { |
| // Test that we properly set Service state in the case where pppd |
| // disconnects during authentication (as opposed to the Notify test, |
| // where pppd disconnects after connecting). |
| const int kPID = 52; |
| const std::map<std::string, std::string> kEmptyArgs; |
| MockCellularService* service = SetMockService(); |
| StartPPP(kPID); |
| |
| ExpectDisconnectCapability3gpp(); |
| // Even if pppd gives a generic error, if we know that the failure occurred |
| // during authentication, we will consider it an auth error. |
| EXPECT_CALL(*service, SetFailure(Service::kFailurePPPAuth)); |
| device_->Notify(kPPPReasonAuthenticating, kEmptyArgs); |
| device_->OnPPPDied(kPID, EXIT_FATAL_ERROR); |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| VerifyDisconnect(); |
| |
| // |Cellular::ppp_task_| is destroyed on the task loop. Must dispatch once to |
| // cleanup. |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularTest, PPPConnectionFailedAfterAuth) { |
| // Test that we properly set Service state in the case where pppd |
| // disconnects after authenticating, but before connecting (as |
| // opposed to the Notify test, where pppd disconnects after |
| // connecting). |
| const int kPID = 52; |
| const std::map<std::string, std::string> kEmptyArgs; |
| MockCellularService* service = SetMockService(); |
| StartPPP(kPID); |
| |
| EXPECT_CALL(*service, SetFailure(Service::kFailureUnknown)); |
| ExpectDisconnectCapability3gpp(); |
| device_->Notify(kPPPReasonAuthenticating, kEmptyArgs); |
| device_->Notify(kPPPReasonAuthenticated, kEmptyArgs); |
| device_->OnPPPDied(kPID, EXIT_FATAL_ERROR); |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| VerifyDisconnect(); |
| |
| // |Cellular::ppp_task_| is destroyed on the task loop. Must dispatch once to |
| // cleanup. |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularTest, PPPConnectionFailedAfterConnect) { |
| // Test that we properly set Service state in the case where pppd fails after |
| // connecting (as opposed to the Notify test, where pppd disconnects normally |
| // after connecting). |
| const int kPID = 52; |
| const std::map<std::string, std::string> kEmptyArgs; |
| MockCellularService* service = SetMockService(); |
| StartPPP(kPID); |
| |
| const std::string ifname = "ppp0"; |
| const int ifindex = 1; |
| auto ppp_device = |
| new MockVirtualDevice(&manager_, ifname, ifindex, Technology::kPPP); |
| std::map<std::string, std::string> ppp_config; |
| ppp_config[kPPPInterfaceName] = ifname; |
| EXPECT_CALL(*device_info(), GetIndex("ppp0")).WillOnce(Return(ifindex)); |
| EXPECT_CALL(*device_info(), CreatePPPDevice(_, StrEq(ifname), ifindex)) |
| .WillOnce(Return(ppp_device)); |
| EXPECT_CALL(*device_info(), |
| RegisterDevice(static_cast<DeviceRefPtr>(ppp_device))); |
| EXPECT_CALL(*ppp_device, SetEnabled(true)); |
| EXPECT_CALL(*ppp_device, SelectService(static_cast<ServiceRefPtr>(service))); |
| EXPECT_CALL(*ppp_device, UpdateNetworkConfig(_)); |
| EXPECT_CALL(*ppp_device, SetServiceFailure(Service::kFailureUnknown)); |
| ExpectDisconnectCapability3gpp(); |
| device_->Notify(kPPPReasonAuthenticating, ppp_config); |
| device_->Notify(kPPPReasonAuthenticated, ppp_config); |
| device_->Notify(kPPPReasonConnect, ppp_config); |
| device_->OnPPPDied(kPID, EXIT_FATAL_ERROR); |
| EXPECT_EQ(nullptr, device_->ppp_task_); |
| VerifyDisconnect(); |
| |
| // |Cellular::ppp_task_| is destroyed on the task loop. Must dispatch once to |
| // cleanup. |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularTest, OnPPPDied) { |
| const int kPID = 1234; |
| const int kExitStatus = 5; |
| ExpectDisconnectCapability3gpp(); |
| device_->OnPPPDied(kPID, kExitStatus); |
| VerifyDisconnect(); |
| } |
| |
| TEST_F(CellularTest, OnPPPDiedCleanupDevice) { |
| // Test that OnPPPDied causes the ppp_device_ reference to be dropped. |
| const int kPID = 123; |
| const int kExitStatus = 5; |
| StartPPP(kPID); |
| FakeUpConnectedPPP(); |
| ExpectDisconnectCapability3gpp(); |
| device_->OnPPPDied(kPID, kExitStatus); |
| VerifyPPPStopped(); |
| |
| // |Cellular::ppp_task_| is destroyed on the task loop. Must dispatch once to |
| // cleanup. |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularTest, DropConnection) { |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| EXPECT_CALL(*default_pdn_, Stop()); |
| device_->DropConnection(); |
| EXPECT_EQ(nullptr, device_->default_pdn_for_testing()); |
| } |
| |
| TEST_F(CellularTest, DropConnectionPPP) { |
| scoped_refptr<MockVirtualDevice> ppp_device( |
| new MockVirtualDevice(&manager_, "ppp0", 123, Technology::kPPP)); |
| // Calling device_->DropConnection() explicitly will trigger |
| // DestroyCapability() which also triggers a (redundant and harmless) |
| // ppp_device->DropConnection() call. |
| EXPECT_CALL(*ppp_device, DropConnection()).Times(AtLeast(1)); |
| device_->ppp_device_ = ppp_device; |
| device_->DropConnection(); |
| } |
| |
| TEST_F(CellularTest, ChangeServiceState) { |
| MockCellularService* service(SetMockService()); |
| EXPECT_CALL(*service, SetState(_)); |
| EXPECT_CALL(*service, SetFailure(_)); |
| EXPECT_CALL(*service, SetFailureSilent(_)); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Without PPP, these should be handled by our selected_service(). |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->SelectService(service); |
| device_->SetServiceState(Service::kStateConfiguring); |
| device_->SetServiceFailure(Service::kFailurePPPAuth); |
| device_->SetServiceFailureSilent(Service::kFailureUnknown); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, ChangeServiceStatePPP) { |
| MockCellularService* service(SetMockService()); |
| scoped_refptr<MockVirtualDevice> ppp_device( |
| new MockVirtualDevice(&manager_, "ppp0", 123, Technology::kPPP)); |
| EXPECT_CALL(*ppp_device, SetServiceState(_)); |
| EXPECT_CALL(*ppp_device, SetServiceFailure(_)); |
| EXPECT_CALL(*ppp_device, SetServiceFailureSilent(_)); |
| EXPECT_CALL(*service, SetState(_)).Times(0); |
| EXPECT_CALL(*service, SetFailure(_)).Times(0); |
| EXPECT_CALL(*service, SetFailureSilent(_)).Times(0); |
| device_->ppp_device_ = ppp_device; |
| |
| // With PPP, these should all be punted over to the |ppp_device|. |
| // Note in particular that Cellular does not manipulate |service| in |
| // this case. |
| device_->SetServiceState(Service::kStateConfiguring); |
| device_->SetServiceFailure(Service::kFailurePPPAuth); |
| device_->SetServiceFailureSilent(Service::kFailureUnknown); |
| } |
| |
| TEST_F(CellularTest, StopPPPOnDisconnect) { |
| const int kPID = 123; |
| Error error; |
| StartPPP(kPID); |
| FakeUpConnectedPPP(); |
| ExpectPPPStopped(); |
| device_->Disconnect(&error, "in test"); |
| VerifyPPPStopped(); |
| } |
| |
| TEST_F(CellularTest, StopPPPOnSuspend) { |
| const int kPID = 123; |
| StartPPP(kPID); |
| FakeUpConnectedPPP(); |
| ExpectPPPStopped(); |
| device_->OnBeforeSuspend(base::DoNothing()); |
| VerifyPPPStopped(); |
| } |
| |
| TEST_F(CellularTest, OnAfterResumeDisabledWantDisabled) { |
| // The Device was disabled prior to resume, and the profile settings |
| // indicate that the device should be disabled. We should leave |
| // things alone. |
| |
| // Initial state. |
| mm1::MockModemProxy* mm1_modem_proxy = SetupOnAfterResume(); |
| Error error; |
| SetEnabledSync(device_.get(), false, true, &error); |
| EXPECT_FALSE(device_->enabled_pending()); |
| EXPECT_FALSE(device_->enabled_persistent()); |
| EXPECT_EQ(Cellular::State::kDisabled, device_->state()); |
| |
| // Resume, while device is disabled. |
| EXPECT_CALL(*mm1_modem_proxy, Enable(_, _, _)).Times(0); |
| device_->OnAfterResume(); |
| EXPECT_FALSE(device_->enabled_pending()); |
| EXPECT_FALSE(device_->enabled_persistent()); |
| EXPECT_EQ(Cellular::State::kDisabled, device_->state()); |
| } |
| |
| TEST_F(CellularTest, OnAfterResumeDisabledWantEnabled) { |
| // This is the ideal case. The disable process completed before |
| // going into suspend. |
| mm1::MockModemProxy* mm1_modem_proxy = SetupOnAfterResume(); |
| mm1::MockModemModem3gppProfileManagerProxy* |
| mm1_modem_3gpp_profile_manager_proxy = |
| SetModem3gppProfileManagerProxyExpectations(); |
| EXPECT_FALSE(device_->enabled_pending()); |
| EXPECT_TRUE(device_->enabled_persistent()); |
| EXPECT_EQ(Cellular::State::kDisabled, device_->state()); |
| |
| // Resume. |
| ResultCallback modem_proxy_enable_callback; |
| EXPECT_CALL(*mm1_modem_proxy, Enable(true, _, _)) |
| .WillOnce(WithArg<1>([&modem_proxy_enable_callback](ResultCallback cb) { |
| modem_proxy_enable_callback = std::move(cb); |
| })); |
| device_->OnAfterResume(); |
| |
| // Complete enable. |
| EXPECT_CALL(*mm1_modem_3gpp_profile_manager_proxy, List(_, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeList)); |
| Error error; |
| ASSERT_TRUE(error.IsSuccess()); |
| std::move(modem_proxy_enable_callback).Run(error); |
| EXPECT_TRUE(device_->enabled_pending()); |
| EXPECT_TRUE(device_->enabled_persistent()); |
| EXPECT_EQ(Cellular::State::kModemStarted, device_->state()); |
| } |
| |
| TEST_F(CellularTest, EstablishLinkFailureNoBearer) { |
| // Link establishment without active bearer set will fail and request |
| // disconnection |
| SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Disconnect(_, _, CellularCapability3gpp::kTimeoutDisconnect)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| SetCapability3gppModemSimpleProxy(); |
| device_->EstablishLink(); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kRegistered, device_->state()); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, EstablishLinkPPP) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kPPP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| |
| const int kPID = 123; |
| EXPECT_CALL(process_manager_, StartProcess(_, _, _, _, _, _, _)) |
| .WillOnce(Return(kPID)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| device_->EstablishLink(); |
| EXPECT_FALSE(device_->selected_service()); |
| EXPECT_FALSE(device_->is_ppp_authenticating_); |
| EXPECT_EQ(nullptr, device_->default_pdn_for_testing()); |
| EXPECT_NE(nullptr, device_->ppp_task_); |
| } |
| |
| TEST_F(CellularTest, EstablishLinkDHCP) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(device_->link_name())) |
| .WillOnce(Return(device_->interface_index())); |
| EXPECT_CALL(rtnl_handler_, RequestDump(net_base::RTNLHandler::kRequestLink)) |
| .Times(1); |
| // Associating because the internal RTNL handler handles the interface up |
| // logic, and at this point that is not yet known. |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| device_->EstablishLink(); |
| EXPECT_FALSE(device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, EstablishLinkMultiplexDHCP) { |
| // Bearer will report an interface different to the one in the |
| // device, allow it as it may be a multiplexed interface |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| RpcIdentifier(""), ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(kTestMultiplexedInterfaceName)) |
| .WillOnce(Return(kTestMultiplexedInterfaceIndex)); |
| |
| // Associating because the internal RTNL handler handles the interface up |
| // logic, and at this point that is not yet known. |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| device_->EstablishLink(); |
| EXPECT_FALSE(device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpDHCP) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestInterfaceName)); |
| |
| device_->DefaultLinkUp(); |
| |
| EXPECT_EQ(service, device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpDHCPL850) { |
| auto l850DeviceId = std::make_unique<DeviceId>( |
| cellular::kL850GLBusType, cellular::kL850GLVid, cellular::kL850GLPid); |
| device_->SetDeviceId(std::move(l850DeviceId)); |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)).Times(0); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))) |
| .Times(0); |
| |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Disconnect(_, _, CellularCapability3gpp::kTimeoutDisconnect)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| SetCapability3gppModemSimpleProxy(); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestInterfaceName)) |
| .Times(0); |
| |
| device_->DefaultLinkUp(); |
| |
| EXPECT_FALSE(device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpMultiplexDHCP) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestMultiplexedInterfaceName)); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex, |
| kTestMultiplexedInterfaceName, |
| Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))); |
| |
| device_->DefaultLinkUp(); |
| EXPECT_EQ(service, device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpConfigureFailure) { |
| // No IPv4 or IPv6 settings in bearer. |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)).Times(0); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))) |
| .Times(0); |
| |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Disconnect(_, _, CellularCapability3gpp::kTimeoutDisconnect)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| SetCapability3gppModemSimpleProxy(); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestInterfaceName)) |
| .Times(0); |
| |
| device_->DefaultLinkUp(); |
| |
| EXPECT_FALSE(device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, EstablishLinkStatic) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kStatic); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(device_->link_name())) |
| .WillOnce(Return(device_->interface_index())); |
| EXPECT_CALL(rtnl_handler_, RequestDump(net_base::RTNLHandler::kRequestLink)) |
| .Times(1); |
| // Associating because the internal RTNL handler handles the interface up |
| // logic, and at this point that is not yet known. |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| device_->EstablishLink(); |
| EXPECT_FALSE(device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, EstablishLinkMultiplexStatic) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kStatic); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(kTestMultiplexedInterfaceName)) |
| .WillOnce(Return(kTestMultiplexedInterfaceIndex)); |
| EXPECT_CALL(rtnl_handler_, RequestDump(net_base::RTNLHandler::kRequestLink)) |
| .Times(1); |
| // Associating because the internal RTNL handler handles the interface up |
| // logic, and at this point that is not yet known. |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| device_->EstablishLink(); |
| EXPECT_FALSE(device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpStatic) { |
| net_base::NetworkConfig network_config; |
| network_config.ipv4_address = |
| net_base::IPv4CIDR::CreateFromCIDRString("10.0.0.1/16"); |
| network_config.ipv4_gateway = |
| net_base::IPv4Address::CreateFromString("10.0.0.254"); |
| network_config.dns_servers = std::vector<net_base::IPAddress>{ |
| *net_base::IPAddress::CreateFromString("10.0.0.2"), |
| *net_base::IPAddress::CreateFromString("8.8.4.4"), |
| *net_base::IPAddress::CreateFromString("8.8.8.8")}; |
| |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kStatic); |
| bearer->set_ipv4_config_for_testing( |
| std::make_unique<net_base::NetworkConfig>(network_config)); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestInterfaceName)); |
| |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| |
| EXPECT_CALL(*default_pdn_, |
| set_link_protocol_network_config(Pointee(Eq(network_config)))); |
| |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Eq(std::nullopt)))); |
| |
| device_->DefaultLinkUp(); |
| EXPECT_EQ(service, device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpMultiplexStatic) { |
| net_base::NetworkConfig network_config; |
| network_config.ipv4_address = |
| net_base::IPv4CIDR::CreateFromCIDRString("10.0.0.1/16"); |
| network_config.ipv4_gateway = |
| net_base::IPv4Address::CreateFromString("10.0.0.254"); |
| network_config.dns_servers = std::vector<net_base::IPAddress>{ |
| *net_base::IPAddress::CreateFromString("10.0.0.2"), |
| *net_base::IPAddress::CreateFromString("8.8.4.4"), |
| *net_base::IPAddress::CreateFromString("8.8.8.8")}; |
| |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kStatic); |
| bearer->set_ipv4_config_for_testing( |
| std::make_unique<net_base::NetworkConfig>(network_config)); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| MockCellularService* service = SetMockService(); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateUnknown)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestMultiplexedInterfaceName)); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex, |
| kTestMultiplexedInterfaceName, |
| Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| |
| EXPECT_CALL(*default_pdn_, |
| set_link_protocol_network_config(Pointee(Eq(network_config)))); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Eq(std::nullopt)))); |
| |
| device_->DefaultLinkUp(); |
| EXPECT_EQ(service, device_->selected_service()); |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkAlreadyUp) { |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| |
| MockCellularService* service = SetMockService(); |
| EXPECT_CALL(*service, SetState(_)).Times(0); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| EXPECT_CALL(*default_pdn_, Start(_)).Times(0); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, EmitStringChanged(_, _)).Times(0); |
| |
| device_->DefaultLinkUp(); |
| |
| Mock::VerifyAndClearExpectations(service); // before Cellular dtor |
| } |
| |
| TEST_F(CellularTest, DefaultLinkInitiallyDown) { |
| SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUnknown); |
| |
| EXPECT_CALL(rtnl_handler_, SetInterfaceFlags(default_pdn_->interface_index(), |
| IFF_UP, IFF_UP)); |
| |
| device_->DefaultLinkDown(); |
| |
| EXPECT_EQ(Cellular::State::kConnected, device_->state()); |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpToDown) { |
| SetRegisteredWithService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| EXPECT_CALL(rtnl_handler_, SetInterfaceFlags(_, _, _)).Times(0); |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Disconnect(_, _, CellularCapability3gpp::kTimeoutDisconnect)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| SetCapability3gppModemSimpleProxy(); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")); |
| |
| device_->DefaultLinkDown(); |
| dispatcher_.DispatchPendingEvents(); |
| |
| EXPECT_EQ(Cellular::State::kRegistered, device_->state()); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, DefaultLinkUpToDownAlreadyDisconnecting) { |
| SetRegisteredWithService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Expect one single proxy->Disconnect() call, the one explicitly triggered by |
| // the device->Disconnect(). There must be no additional call due to the link |
| // down event. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)).Times(1); |
| SetCapability3gppModemSimpleProxy(); |
| |
| device_->Disconnect(nullptr, "in test"); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")) |
| .Times(0); |
| EXPECT_CALL(rtnl_handler_, SetInterfaceFlags(_, _, _)).Times(0); |
| device_->DefaultLinkDown(); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Service::kStateIdle, device_->service_->state()); |
| } |
| |
| TEST_F(CellularTest, DefaultLinkAlreadyDown) { |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kDown); |
| |
| EXPECT_CALL(rtnl_handler_, SetInterfaceFlags(_, _, _)).Times(0); |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)).Times(0); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| device_->DefaultLinkDown(); |
| } |
| |
| TEST_F(CellularTest, DefaultLinkDeleted) { |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConfiguring); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->SelectService(service); |
| |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")); |
| |
| device_->DefaultLinkDeleted(); |
| |
| EXPECT_FALSE(device_->selected_service()); |
| } |
| |
| TEST_F(CellularTest, UpdateGeolocationObjects) { |
| static const Cellular::LocationInfo kGoodLocations[] = { |
| {"310", "410", "DE7E", "4985F6"}, |
| {"001", "010", "O100", "googol"}, |
| {"foo", "bar", "bazz", "quuux"}}; |
| static const Cellular::LocationInfo kBadLocations[] = {{"wat", "", "", ""}, |
| {"", "", "", ""}}; |
| |
| std::vector<GeolocationInfo> objects; |
| |
| for (const auto& location : kGoodLocations) { |
| std::string raw_location = location.mcc + "," + location.mnc + "," + |
| location.lac + "," + location.ci; |
| Error error; |
| |
| GeolocationInfo expected_info; |
| expected_info[kGeoMobileCountryCodeProperty] = location.mcc; |
| expected_info[kGeoMobileNetworkCodeProperty] = location.mnc; |
| expected_info[kGeoLocationAreaCodeProperty] = location.lac; |
| expected_info[kGeoCellIdProperty] = location.ci; |
| |
| device_->GetLocationCallback(raw_location, error); |
| device_->UpdateGeolocationObjects(&objects); |
| |
| ASSERT_EQ(objects.size(), 1); |
| EXPECT_EQ(expected_info, objects[0]); |
| } |
| |
| for (const auto& location : kBadLocations) { |
| std::string raw_location = location.mcc + "," + location.mnc + "," + |
| location.lac + "," + location.ci; |
| Error error; |
| GeolocationInfo empty_info; |
| |
| device_->GetLocationCallback(raw_location, error); |
| device_->UpdateGeolocationObjects(&objects); |
| |
| ASSERT_EQ(objects.size(), 1); |
| EXPECT_EQ(empty_info, objects[0]); |
| } |
| } |
| |
| // Helper class because gmock doesn't play nicely with unique_ptr |
| class FakeMobileOperatorInfo : public NiceMock<MockMobileOperatorInfo> { |
| public: |
| FakeMobileOperatorInfo(EventDispatcher* dispatcher, |
| std::vector<MobileAPN> apn_list) |
| : NiceMock<MockMobileOperatorInfo>(dispatcher, "Fake"), |
| apn_list_(std::move(apn_list)) {} |
| |
| const std::vector<MobileAPN>& apn_list() const override { return apn_list_; } |
| |
| private: |
| std::vector<MobileAPN> apn_list_; |
| }; |
| |
| TEST_F(CellularTest, SimpleApnList) { |
| constexpr char kApn[] = "apn"; |
| constexpr char kUsername[] = "foo"; |
| constexpr char kPassword[] = "bar"; |
| |
| std::vector<MobileAPN> apn_list; |
| MobileAPN mobile_apn; |
| mobile_apn.apn = kApn; |
| mobile_apn.username = kUsername; |
| mobile_apn.password = kPassword; |
| apn_list.emplace_back(std::move(mobile_apn)); |
| |
| FakeMobileOperatorInfo* info = |
| new FakeMobileOperatorInfo(&dispatcher_, std::move(apn_list)); |
| // Pass ownership of |info| |
| device_->set_mobile_operator_info_for_testing(info); |
| |
| device_->UpdateHomeProvider(); |
| auto apn_list_prop = device_->apn_list(); |
| CHECK_EQ(1U, apn_list_prop.size()); |
| CHECK_EQ(kApn, apn_list_prop[0][kApnProperty]); |
| CHECK_EQ(kUsername, apn_list_prop[0][kApnUsernameProperty]); |
| CHECK_EQ(kPassword, apn_list_prop[0][kApnPasswordProperty]); |
| } |
| |
| TEST_F(CellularTest, ProfilesApnList) { |
| constexpr char kApn1[] = "ota.apn"; |
| brillo::VariantDictionary profile; |
| profile[CellularBearer::kMMApnProperty] = std::string(kApn1); |
| profile[CellularBearer::kMMApnTypeProperty] = |
| uint32_t(MM_BEARER_APN_TYPE_DEFAULT); |
| Capability3gppCallOnProfilesChanged({profile}); |
| |
| constexpr char kApn2[] = "normal.apn"; |
| std::vector<MobileAPN> apn_list; |
| MobileAPN mobile_apn; |
| mobile_apn.apn = kApn2; |
| apn_list.emplace_back(std::move(mobile_apn)); |
| FakeMobileOperatorInfo* info = |
| new FakeMobileOperatorInfo(&dispatcher_, std::move(apn_list)); |
| // Pass ownership of |info| |
| device_->set_mobile_operator_info_for_testing(info); |
| |
| device_->UpdateHomeProvider(); |
| auto apn_list_prop = device_->apn_list(); |
| CHECK_EQ(2U, apn_list_prop.size()); |
| // Profile APNs are likely deployed by the network. They should be tried |
| // first, so they should be higher in the list. |
| CHECK_EQ(kApn1, apn_list_prop[0][kApnProperty]); |
| CHECK_EQ(kApn2, apn_list_prop[1][kApnProperty]); |
| } |
| |
| TEST_F(CellularTest, MergeProfileAndOperatorApn) { |
| constexpr char kApn[] = "normal.apn"; |
| constexpr char kApnName[] = "Normal APN"; |
| brillo::VariantDictionary profile; |
| profile[CellularBearer::kMMApnProperty] = std::string(kApn); |
| profile[CellularBearer::kMMApnTypeProperty] = |
| uint32_t(MM_BEARER_APN_TYPE_DEFAULT); |
| Capability3gppCallOnProfilesChanged({profile}); |
| |
| std::vector<MobileAPN> apn_list; |
| MobileAPN mobile_apn; |
| mobile_apn.apn = kApn; |
| mobile_apn.operator_name_list.push_back({kApnName, ""}); |
| apn_list.emplace_back(std::move(mobile_apn)); |
| FakeMobileOperatorInfo* info = |
| new FakeMobileOperatorInfo(&dispatcher_, std::move(apn_list)); |
| // Pass ownership of |info| |
| device_->set_mobile_operator_info_for_testing(info); |
| |
| device_->UpdateHomeProvider(); |
| auto apn_list_prop = device_->apn_list(); |
| CHECK_EQ(1U, apn_list_prop.size()); |
| CHECK_EQ(kApn, apn_list_prop[0][kApnProperty]); |
| CHECK_EQ(kApnName, apn_list_prop[0][kApnNameProperty]); |
| } |
| |
| TEST_F(CellularTest, DontMergeProfileAndOperatorApn) { |
| constexpr char kApn[] = "normal.apn"; |
| constexpr char kUsernameFromProfile[] = "user1"; |
| brillo::VariantDictionary profile; |
| profile[CellularBearer::kMMApnProperty] = std::string(kApn); |
| profile[CellularBearer::kMMUserProperty] = std::string(kUsernameFromProfile); |
| profile[CellularBearer::kMMApnTypeProperty] = |
| uint32_t(MM_BEARER_APN_TYPE_DEFAULT); |
| Capability3gppCallOnProfilesChanged({profile}); |
| |
| constexpr char kUsernameFromOperator[] = "user2"; |
| std::vector<MobileAPN> apn_list; |
| MobileAPN mobile_apn; |
| mobile_apn.apn = kApn; |
| mobile_apn.username = kUsernameFromOperator; |
| apn_list.emplace_back(std::move(mobile_apn)); |
| FakeMobileOperatorInfo* info = |
| new FakeMobileOperatorInfo(&dispatcher_, std::move(apn_list)); |
| // Pass ownership of |info| |
| device_->set_mobile_operator_info_for_testing(info); |
| |
| device_->UpdateHomeProvider(); |
| auto apn_list_prop = device_->apn_list(); |
| CHECK_EQ(2U, apn_list_prop.size()); |
| // As before, profile APNs come first. |
| CHECK_EQ(kApn, apn_list_prop[0][kApnProperty]); |
| CHECK_EQ(kUsernameFromProfile, apn_list_prop[0][kApnUsernameProperty]); |
| CHECK_EQ(kApn, apn_list_prop[1][kApnProperty]); |
| CHECK_EQ(kUsernameFromOperator, apn_list_prop[1][kApnUsernameProperty]); |
| } |
| |
| TEST_F(CellularTest, RequiredApnExists) { |
| // The default and attach try list always have an additional empty APN |
| // fallback added automatically |
| Stringmap default_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kDefault); |
| Stringmap attach_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kAttach); |
| |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn1"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn2"; |
| apn2[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| EXPECT_FALSE(device_->RequiredApnExists(ApnList::ApnType::kAttach)); |
| EXPECT_FALSE(device_->RequiredApnExists(ApnList::ApnType::kDefault)); |
| |
| // Required APNs are only meant for MODB APNs |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn1[kApnSourceProperty] = kApnSourceModem; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn2[kApnSourceProperty] = kApnSourceUi; |
| apn_list.clear(); |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| EXPECT_FALSE(device_->RequiredApnExists(ApnList::ApnType::kAttach)); |
| EXPECT_FALSE(device_->RequiredApnExists(ApnList::ApnType::kDefault)); |
| |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.clear(); |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| EXPECT_FALSE(device_->RequiredApnExists(ApnList::ApnType::kAttach)); |
| EXPECT_TRUE(device_->RequiredApnExists(ApnList::ApnType::kDefault)); |
| |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.clear(); |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| EXPECT_TRUE(device_->RequiredApnExists(ApnList::ApnType::kAttach)); |
| EXPECT_TRUE(device_->RequiredApnExists(ApnList::ApnType::kDefault)); |
| } |
| |
| TEST_F(CellularTest, BuildApnTryListSetApn) { |
| // The default and attach try list always have an additional empty APN |
| // fallback added automatically |
| Stringmap default_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kDefault); |
| Stringmap attach_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kAttach); |
| |
| Stringmaps apn_list; |
| Stringmap apn_modb, apn_modem; |
| apn_modb[kApnProperty] = "apn_modb"; |
| apn_modb[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn_modb[kApnSourceProperty] = kApnSourceMoDb; |
| apn_modb[kApnIsRequiredByCarrierSpecProperty] = |
| kApnIsRequiredByCarrierSpecFalse; |
| apn_modem[kApnProperty] = "apn_modem"; |
| apn_modem[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA}); |
| apn_modem[kApnSourceProperty] = kApnSourceModem; |
| apn_list.push_back(apn_modb); |
| apn_list.push_back(apn_modem); |
| device_->SetApnList(apn_list); |
| |
| ASSERT_EQ(device_->BuildTetheringApnTryList().size(), 0); |
| std::deque<Stringmap> default_apn_try_list = |
| device_->BuildDefaultApnTryList(); |
| std::deque<Stringmap> attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[2], apn_modem); |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| // Modem APNs go first |
| EXPECT_EQ(default_apn_try_list[0], apn_modem); |
| EXPECT_EQ(default_apn_try_list[1], apn_modb); |
| EXPECT_EQ(default_apn_try_list[2], default_empty_apn); |
| |
| // Add a custom APN |
| CellularService* service = SetService(); |
| Stringmap custom_apn; |
| custom_apn[kApnProperty] = "custom_apn"; |
| custom_apn[kApnSourceProperty] = kApnSourceUi; |
| custom_apn[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| service->set_apn_info_for_testing(custom_apn); |
| ASSERT_EQ(device_->BuildTetheringApnTryList().size(), 0); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[2], apn_modem); |
| ASSERT_EQ(default_apn_try_list.size(), 4); |
| EXPECT_EQ(default_apn_try_list[0], custom_apn); |
| EXPECT_EQ(default_apn_try_list[1], apn_modem); |
| EXPECT_EQ(default_apn_try_list[2], apn_modb); |
| EXPECT_EQ(default_apn_try_list[3], default_empty_apn); |
| |
| // Set the last good APN to an APN not in the current list |
| Stringmap last_good_apn; |
| last_good_apn[kApnProperty] = "last_good_apn"; |
| last_good_apn[kApnSourceProperty] = kApnSourceUi; |
| last_good_apn[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| service->SetLastGoodApn(last_good_apn); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[2], apn_modem); |
| ASSERT_EQ(default_apn_try_list.size(), 5); |
| EXPECT_EQ(default_apn_try_list[0], custom_apn); |
| EXPECT_EQ(default_apn_try_list[1], apn_modem); |
| EXPECT_EQ(default_apn_try_list[2], apn_modb); |
| EXPECT_EQ(default_apn_try_list[3], default_empty_apn); |
| EXPECT_EQ(default_apn_try_list[4], last_good_apn); |
| |
| // Set the last good APN to an existing APN |
| service->SetLastGoodApn(apn_modem); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| ASSERT_EQ(default_apn_try_list.size(), 4); |
| EXPECT_EQ(default_apn_try_list[0], custom_apn); |
| EXPECT_EQ(default_apn_try_list[1], |
| apn_modem); // MODB sorted based on last_good_apn |
| EXPECT_EQ(default_apn_try_list[2], apn_modb); |
| EXPECT_EQ(default_apn_try_list[3], default_empty_apn); |
| |
| // Set the last good APN to the empty fallback APN |
| service->SetLastGoodApn(default_empty_apn); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| ASSERT_EQ(default_apn_try_list.size(), 4); |
| EXPECT_EQ(default_apn_try_list[0], custom_apn); |
| EXPECT_EQ(default_apn_try_list[1], apn_modem); |
| EXPECT_EQ(default_apn_try_list[2], apn_modb); |
| EXPECT_EQ(default_apn_try_list[3], default_empty_apn); |
| |
| // Set the custom APN to an existing APN |
| service->set_apn_info_for_testing(apn_modb); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| EXPECT_EQ(default_apn_try_list[0], apn_modb); |
| EXPECT_EQ(default_apn_try_list[1], apn_modem); |
| EXPECT_EQ(default_apn_try_list[2], default_empty_apn); |
| |
| // Add a custom IA APN |
| Stringmap custom_apn2; |
| custom_apn2[kApnProperty] = "custom_apn2"; |
| custom_apn2[kApnSourceProperty] = kApnSourceUi; |
| custom_apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeIA}); |
| service->set_apn_info_for_testing(custom_apn2); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 1); |
| EXPECT_EQ(attach_apn_try_list[0], custom_apn2); |
| |
| // Verify that a required MODB APN blocks any Custom APNs. |
| Stringmap apn_required( |
| {{kApnProperty, "apn_required"}, |
| {kApnTypesProperty, ApnList::JoinApnTypes({kApnTypeIA})}, |
| {kApnSourceProperty, kApnSourceMoDb}, |
| {kApnIsRequiredByCarrierSpecProperty, kApnIsRequiredByCarrierSpecTrue}}); |
| Stringmap apn4({{kApnProperty, "apn4"}, |
| {kApnTypesProperty, |
| ApnList::JoinApnTypes({kApnTypeIA, kApnTypeDefault})}, |
| {kApnSourceProperty, kApnSourceMoDb}}); |
| apn_list.push_back(apn_required); |
| apn_list.push_back(apn4); |
| device_->SetApnList(apn_list); |
| service->SetLastGoodApn(Stringmap()); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 4); |
| // Modem APNs are not excluded |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], apn_required); |
| EXPECT_EQ(attach_apn_try_list[2], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[3], apn_modem); |
| ASSERT_EQ(default_apn_try_list.size(), 4); |
| EXPECT_EQ(default_apn_try_list[0], apn_modem); |
| EXPECT_EQ(default_apn_try_list[1], apn_modb); |
| EXPECT_EQ(default_apn_try_list[2], apn4); |
| EXPECT_EQ(default_apn_try_list[3], default_empty_apn); |
| } |
| |
| TEST_F(CellularTest, BuildApnTryListSetCustomApnList) { |
| // The default and attach try list always have an additional empty APN |
| // fallback added automatically |
| Stringmap default_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kDefault); |
| Stringmap attach_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kAttach); |
| |
| Stringmaps apn_list; |
| Stringmap apn_modb, apn_modem; |
| apn_modb[kApnProperty] = "apn_modb"; |
| apn_modb[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn_modb[kApnSourceProperty] = kApnSourceMoDb; |
| apn_modem[kApnProperty] = "apn_modem"; |
| apn_modem[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA}); |
| apn_modem[kApnSourceProperty] = kApnSourceModem; |
| apn_list.push_back(apn_modb); |
| apn_list.push_back(apn_modem); |
| device_->SetApnList(apn_list); |
| |
| // Without any custom APNs, Build*ApnTryList should return APNs from modb and |
| // modem(apn_list), as well as the empty APN fallback |
| std::deque<Stringmap> default_apn_try_list = |
| device_->BuildDefaultApnTryList(); |
| std::deque<Stringmap> attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(device_->BuildTetheringApnTryList().size(), 0); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[2], apn_modem); |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| EXPECT_EQ(default_apn_try_list[0], apn_modem); |
| EXPECT_EQ(default_apn_try_list[1], apn_modb); |
| EXPECT_EQ(default_apn_try_list[2], default_empty_apn); |
| |
| // Check that when an empty CustomApnList is used, the APNs from the modb and |
| // modem are included, as well as the empty APN fallback |
| CellularService* service = SetService(); |
| service->set_custom_apn_list_for_testing(Stringmaps()); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[2], apn_modem); |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| EXPECT_EQ(default_apn_try_list[0], apn_modem); |
| EXPECT_EQ(default_apn_try_list[1], apn_modb); |
| EXPECT_EQ(default_apn_try_list[2], default_empty_apn); |
| |
| // Set CustomApnList |
| Stringmap apnP({{kApnProperty, "apnP"}, |
| {kApnTypesProperty, ApnList::JoinApnTypes({kApnTypeIA})}, |
| {kApnSourceProperty, kApnSourceUi}}); |
| Stringmap apnQ({{kApnProperty, "apnQ"}, |
| {kApnTypesProperty, ApnList::JoinApnTypes({kApnTypeDefault})}, |
| {kApnSourceProperty, kApnSourceUi}}); |
| Stringmap apnR({{kApnProperty, "apnR"}, |
| {kApnTypesProperty, |
| ApnList::JoinApnTypes({kApnTypeIA, kApnTypeDefault})}, |
| {kApnSourceProperty, kApnSourceUi}}); |
| Stringmap apnS({{kApnProperty, "apnS"}, |
| {kApnTypesProperty, ApnList::JoinApnTypes({kApnTypeDefault})}, |
| {kApnSourceProperty, kApnSourceAdmin}}); |
| Stringmaps custom_list = {apnP, apnQ, apnR, apnS}; |
| service->set_custom_apn_list_for_testing(custom_list); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apnP); |
| EXPECT_EQ(attach_apn_try_list[1], apnR); |
| EXPECT_EQ(attach_apn_try_list[2], apnP); |
| EXPECT_EQ(default_apn_try_list[0], apnQ); |
| EXPECT_EQ(default_apn_try_list[1], apnR); |
| EXPECT_EQ(default_apn_try_list[2], apnS); |
| |
| // Verify that a required MODB APN blocks any Custom APNs. |
| Stringmap apn_required( |
| {{kApnProperty, "apn_required"}, |
| {kApnTypesProperty, ApnList::JoinApnTypes({kApnTypeIA})}, |
| {kApnSourceProperty, kApnSourceMoDb}, |
| {kApnIsRequiredByCarrierSpecProperty, kApnIsRequiredByCarrierSpecTrue}}); |
| apn_list.push_back(apn_required); |
| device_->SetApnList(apn_list); |
| service->SetLastGoodApn(Stringmap()); |
| default_apn_try_list = device_->BuildDefaultApnTryList(); |
| attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 4); |
| // Modem APNs are not excluded |
| EXPECT_EQ(attach_apn_try_list[0], apn_modem); |
| EXPECT_EQ(attach_apn_try_list[1], apn_required); |
| EXPECT_EQ(attach_apn_try_list[2], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[3], apn_modem); |
| // CustomApnList has exclusive priority if no required APNs exist. |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| EXPECT_EQ(default_apn_try_list[0], apnQ); |
| EXPECT_EQ(default_apn_try_list[1], apnR); |
| EXPECT_EQ(default_apn_try_list[2], apnS); |
| } |
| |
| TEST_F(CellularTest, BuildApnTryListWithInvalid) { |
| // The default and attach try list always have an additional empty APN |
| // fallback added automatically |
| Stringmap default_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kDefault); |
| Stringmap attach_empty_apn = |
| Cellular::BuildFallbackEmptyApn(ApnList::ApnType::kAttach); |
| |
| Stringmaps apn_list; |
| Stringmap apn1, apn2, apn3; |
| // Valid default APN |
| apn1[kApnProperty] = "apn1"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.push_back(apn1); |
| // Valid default+initial APN |
| apn2[kApnProperty] = "apn2"; |
| apn2[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.push_back(apn2); |
| // Invalid APN entry without kApnProperty |
| apn3[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA}); |
| apn3[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.push_back(apn3); |
| device_->SetApnList(apn_list); |
| |
| std::deque<Stringmap> default_apn_try_list = |
| device_->BuildDefaultApnTryList(); |
| std::deque<Stringmap> attach_apn_try_list = device_->BuildAttachApnTryList(); |
| ASSERT_EQ(attach_apn_try_list.size(), 3); |
| EXPECT_EQ(attach_apn_try_list[0], apn2); |
| EXPECT_EQ(attach_apn_try_list[1], attach_empty_apn); |
| EXPECT_EQ(attach_apn_try_list[2], apn2); |
| ASSERT_EQ(default_apn_try_list.size(), 3); |
| EXPECT_EQ(default_apn_try_list[0], apn1); |
| EXPECT_EQ(default_apn_try_list[1], apn2); |
| EXPECT_EQ(default_apn_try_list[2], default_empty_apn); |
| } |
| |
| TEST_F(CellularTest, BuildTetheringApnTryList) { |
| ASSERT_EQ(device_->BuildTetheringApnTryList().size(), 0); |
| Stringmaps apn_list; |
| Stringmap apn_modb, apn_modem; |
| apn_modb[kApnProperty] = "apn_modb"; |
| apn_modb[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun}); |
| apn_modb[kApnSourceProperty] = kApnSourceMoDb; |
| apn_modb[kApnIsRequiredByCarrierSpecProperty] = |
| kApnIsRequiredByCarrierSpecFalse; |
| apn_modem[kApnProperty] = "apn_modem"; |
| apn_modem[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA, kApnTypeDun}); |
| apn_modem[kApnSourceProperty] = kApnSourceModem; |
| apn_list.push_back(apn_modb); |
| apn_list.push_back(apn_modem); |
| device_->SetApnList(apn_list); |
| |
| std::deque<Stringmap> dun_apn_try_list = device_->BuildTetheringApnTryList(); |
| ASSERT_EQ(dun_apn_try_list.size(), 2); |
| EXPECT_EQ(dun_apn_try_list[0], apn_modem); |
| EXPECT_EQ(dun_apn_try_list[1], apn_modb); |
| |
| // Set CustomApnList |
| Stringmap apnP({{kApnProperty, "apnP"}, |
| {kApnTypesProperty, |
| ApnList::JoinApnTypes({kApnTypeIA, kApnTypeDefault})}, |
| {kApnSourceProperty, kApnSourceUi}}); |
| Stringmap apnQ({{kApnProperty, "apnQ"}, |
| {kApnTypesProperty, |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun})}, |
| {kApnSourceProperty, kApnSourceUi}}); |
| Stringmap apnR({{kApnProperty, "apnR"}, |
| {kApnTypesProperty, |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun})}, |
| {kApnSourceProperty, kApnSourceUi}}); |
| Stringmaps custom_list = {apnP, apnQ, apnR}; |
| CellularService* service = SetService(); |
| service->set_custom_apn_list_for_testing(custom_list); |
| |
| // When using a custom DUN APN, modb and modem apns are not included. |
| dun_apn_try_list = device_->BuildTetheringApnTryList(); |
| ASSERT_EQ(dun_apn_try_list.size(), 2); |
| EXPECT_EQ(dun_apn_try_list[0], apnQ); |
| EXPECT_EQ(dun_apn_try_list[1], apnR); |
| |
| // Set the modb apn as required. |
| apn_modb[kApnIsRequiredByCarrierSpecProperty] = |
| kApnIsRequiredByCarrierSpecTrue; |
| apn_list.clear(); |
| apn_list.push_back(apn_modb); |
| apn_list.push_back(apn_modem); |
| device_->SetApnList(apn_list); |
| |
| dun_apn_try_list = device_->BuildTetheringApnTryList(); |
| ASSERT_EQ(dun_apn_try_list.size(), 2); |
| EXPECT_EQ(dun_apn_try_list[0], apn_modem); |
| EXPECT_EQ(dun_apn_try_list[1], apn_modb); |
| } |
| |
| TEST_F(CellularTest, CompareApns) { |
| Stringmap apn1, apn2; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| apn1[kApnNameProperty] = "apn_name1"; |
| apn2[kApnNameProperty] = "apn_name2"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| |
| apn1[kApnSourceProperty] = "test_source"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| EXPECT_TRUE(device_->CompareApns(apn2, apn1)); |
| |
| apn2[cellular::kApnVersionProperty] = "test_version"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| EXPECT_TRUE(device_->CompareApns(apn2, apn1)); |
| |
| apn1[kApnUsernameProperty] = "username"; |
| EXPECT_FALSE(device_->CompareApns(apn1, apn2)); |
| EXPECT_FALSE(device_->CompareApns(apn2, apn1)); |
| |
| apn2[kApnUsernameProperty] = "username_two"; |
| EXPECT_FALSE(device_->CompareApns(apn1, apn2)); |
| EXPECT_FALSE(device_->CompareApns(apn2, apn1)); |
| |
| apn2[kApnUsernameProperty] = "username"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| EXPECT_TRUE(device_->CompareApns(apn2, apn1)); |
| |
| apn2[kApnLanguageProperty] = "language"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| EXPECT_TRUE(device_->CompareApns(apn2, apn1)); |
| |
| apn2[cellular::kApnVersionProperty] = "version"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| EXPECT_TRUE(device_->CompareApns(apn2, apn1)); |
| |
| apn1[kApnProperty] = "apn.test"; |
| EXPECT_FALSE(device_->CompareApns(apn1, apn2)); |
| EXPECT_FALSE(device_->CompareApns(apn2, apn1)); |
| |
| apn2[kApnProperty] = "apn.test"; |
| EXPECT_TRUE(device_->CompareApns(apn1, apn2)); |
| EXPECT_TRUE(device_->CompareApns(apn2, apn1)); |
| } |
| |
| TEST_F(CellularTest, CompareApnsFromStorage) { |
| // Store the last good APN and retrieve it again. Verify that CompareApns |
| // matches the loaded value. |
| Stringmap last_good_apn; |
| last_good_apn[kApnProperty] = "apn.com"; |
| last_good_apn[kApnNameProperty] = "Last Good APN"; |
| last_good_apn[kApnSourceProperty] = kApnSourceUi; |
| last_good_apn[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeIA, kApnTypeDun}); |
| last_good_apn[kApnUsernameProperty] = "username"; |
| last_good_apn[kApnPasswordProperty] = "password"; |
| last_good_apn[kApnAuthenticationProperty] = "auth"; |
| last_good_apn[kApnIpTypeProperty] = "iptype"; |
| last_good_apn[kApnIsRequiredByCarrierSpecProperty] = "true"; |
| last_good_apn[cellular::kApnVersionProperty] = |
| base::NumberToString(cellular::kCurrentApnCacheVersion); |
| last_good_apn[kApnAttachProperty] = "attach"; |
| last_good_apn[kApnLocalizedNameProperty] = "localized"; |
| |
| device_->set_iccid_for_testing(kIccid); |
| CellularService* service = SetService(); |
| service->SetLastGoodApn(last_good_apn); |
| ASSERT_TRUE(service->Save(&profile_storage_)); |
| service->ClearLastGoodApn(); |
| ASSERT_TRUE(service->Load(&profile_storage_)); |
| ASSERT_NE(service->GetLastGoodApn(), nullptr); |
| EXPECT_TRUE(device_->CompareApns(last_good_apn, *service->GetLastGoodApn())); |
| } |
| |
| TEST_F(CellularTest, CompareApnsFromApnList) { |
| std::vector<MobileAPN> apn_list; |
| MobileAPN mobile_apn; |
| mobile_apn.apn = "apn"; |
| mobile_apn.username = "username"; |
| mobile_apn.password = "password"; |
| mobile_apn.operator_name_list.push_back({"name", ""}); |
| mobile_apn.authentication = kApnAuthenticationChap; |
| mobile_apn.apn_types = {"DEFAULT", "IA", "DUN"}; |
| mobile_apn.ip_type = kApnIpTypeV4V6; |
| mobile_apn.is_required_by_carrier_spec = true; |
| apn_list.emplace_back(std::move(mobile_apn)); |
| FakeMobileOperatorInfo* info = |
| new FakeMobileOperatorInfo(&dispatcher_, std::move(apn_list)); |
| // Pass ownership of |info| |
| device_->set_mobile_operator_info_for_testing(info); |
| device_->UpdateHomeProvider(); |
| auto apn_list_prop = device_->apn_list(); |
| ASSERT_EQ(apn_list_prop.size(), 1); |
| |
| // Save value to storage and check for equality. |
| device_->set_iccid_for_testing(kIccid); |
| CellularService* service = SetService(); |
| service->SetLastGoodApn(apn_list_prop[0]); |
| ASSERT_TRUE(service->Save(&profile_storage_)); |
| service->ClearLastGoodApn(); |
| ASSERT_TRUE(service->Load(&profile_storage_)); |
| ASSERT_NE(service->GetLastGoodApn(), nullptr); |
| EXPECT_TRUE( |
| device_->CompareApns(apn_list_prop[0], *service->GetLastGoodApn())); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_NoService) { |
| device_->SetServiceForTesting(nullptr); |
| device_->SetSelectedServiceForTesting(nullptr); |
| ASSERT_EQ(device_->service(), nullptr); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| base::test::TestFuture<TetheringManager::CellularUpstreamEvent> future_event; |
| device_->AcquireTetheringNetwork( |
| TetheringManager::UpdateTimeoutCallback(), future.GetCallback(), |
| future_event.GetRepeatingCallback(), false /*experimental_tethering*/); |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_NoModem) { |
| SetRegisteredWithService(); |
| ASSERT_NE(device_->service(), nullptr); |
| device_->DestroyCapability(); |
| ASSERT_EQ(GetCapability3gpp(), nullptr); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| base::test::TestFuture<TetheringManager::CellularUpstreamEvent> future_event; |
| device_->AcquireTetheringNetwork( |
| TetheringManager::UpdateTimeoutCallback(), future.GetCallback(), |
| future_event.GetRepeatingCallback(), false /*experimental_tethering*/); |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_Inhibited) { |
| SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| SetInhibited(true); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| base::test::TestFuture<TetheringManager::CellularUpstreamEvent> future_event; |
| device_->AcquireTetheringNetwork( |
| TetheringManager::UpdateTimeoutCallback(), future.GetCallback(), |
| future_event.GetRepeatingCallback(), false /*experimental_tethering*/); |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_Disconnected) { |
| SetRegisteredWithService(); |
| EXPECT_EQ(Cellular::State::kRegistered, device_->state()); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| base::test::TestFuture<TetheringManager::CellularUpstreamEvent> future_event; |
| device_->AcquireTetheringNetwork( |
| TetheringManager::UpdateTimeoutCallback(), future.GetCallback(), |
| future_event.GetRepeatingCallback(), false /*experimental_tethering*/); |
| |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_DisallowedByOperator) { |
| SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(false)); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| base::test::TestFuture<TetheringManager::CellularUpstreamEvent> future_event; |
| device_->AcquireTetheringNetwork( |
| TetheringManager::UpdateTimeoutCallback(), future.GetCallback(), |
| future_event.GetRepeatingCallback(), false /*experimental_tethering*/); |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_ReuseDefault) { |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| device_->ReuseDefaultPdnForTethering(future.GetCallback()); |
| EXPECT_EQ(future.Get<Network*>(), default_pdn_); |
| EXPECT_TRUE(future.Get<Error>().IsSuccess()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_OperationType_Reuse_DefaultNoDun) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // No DUN APN in the list |
| Stringmap apn; |
| apn[kApnProperty] = "apn"; |
| apn[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn[kApnSourceProperty] = kApnSourceMoDb; |
| service->SetLastGoodApn(apn); |
| |
| Stringmaps apn_list; |
| apn_list.push_back(apn); |
| device_->SetApnList(apn_list); |
| |
| // Test only the operation type selection |
| EXPECT_EQ(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kReuseDefaultPdn); |
| } |
| |
| TEST_F(CellularTest, |
| AcquireTetheringNetwork_OperationType_Reuse_DefaultIsAlsoDun) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // Last good APN is also in the APN list |
| Stringmaps apn_list; |
| Stringmap apn; |
| apn[kApnProperty] = "apn"; |
| apn[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun}); |
| apn[kApnSourceProperty] = kApnSourceMoDb; |
| apn_list.push_back(apn); |
| device_->SetApnList(apn_list); |
| service->SetLastGoodApn(apn); |
| |
| // Test only the operation type selection |
| EXPECT_EQ(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kReuseDefaultPdn); |
| } |
| |
| TEST_F(CellularTest, |
| AcquireTetheringNetwork_OperationType_Reuse_DefaultIsAlsoDunRequired) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // Last good APN is also in the APN list, and is flagged as required. |
| Stringmaps apn_list; |
| Stringmap apn; |
| apn[kApnProperty] = "apn"; |
| apn[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun}); |
| apn[kApnSourceProperty] = kApnSourceMoDb; |
| apn[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn); |
| device_->SetApnList(apn_list); |
| service->SetLastGoodApn(apn); |
| |
| // Test only the operation type selection |
| EXPECT_EQ(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kReuseDefaultPdn); |
| } |
| |
| TEST_F(CellularTest, |
| AcquireTetheringNetwork_OperationType_NoReuse_DefaultIsAlsoDun) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // Last good APN is NOT in the APN list (e.g. coming from UI), and there is |
| // another APN flagged as required, so the network cannot be reused. |
| Stringmap apn1; |
| apn1[kApnProperty] = "apn1"; |
| apn1[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun}); |
| apn1[kApnSourceProperty] = kApnSourceUi; |
| service->SetLastGoodApn(apn1); |
| Stringmaps apn_list; |
| Stringmap apn2; |
| apn2[kApnProperty] = "apn2"; |
| apn2[kApnTypesProperty] = |
| ApnList::JoinApnTypes({kApnTypeDefault, kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Test only the operation type selection. It MUST NOT BE kReuseDefaultPdn. |
| EXPECT_NE(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kReuseDefaultPdn); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_DunAsDefault) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Will request stop of the current default network. |
| EXPECT_CALL(*default_pdn_, Stop()); |
| |
| // Will request disconnection of all bearers via capability. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| |
| // Primary multiplexed interface name will be cleared up. |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")); |
| |
| // Will request reconnection with the DUN APN. |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillOnce( |
| Invoke([](const KeyValueStore& props, RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(props.Get<uint32_t>(CellularBearer::kMMApnTypeProperty), |
| MM_BEARER_APN_TYPE_TETHERING); |
| EXPECT_EQ(props.Get<std::string>(CellularBearer::kMMApnProperty), |
| "apn-dun"); |
| std::move(callback).Run(kTestBearerDBusPath, Error()); |
| })); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| // 1st step: run ConnectTetheringAsDefaultPdn() until the new PDN |
| // connection is connected and we're requested to EstablishLink(). |
| device_->set_skip_establish_link_for_testing(true); |
| |
| // Metrics for the DUN APN should be reported. |
| EXPECT_CALL(metrics_, |
| NotifyCellularConnectionResult( |
| Error::kSuccess, |
| Metrics::DetailedCellularConnectionResult::APNType::kDUN)); |
| |
| // Portal detection is supported. |
| ON_CALL(*service, GetNetworkValidationMode) |
| .WillByDefault(Return(NetworkMonitor::ValidationMode::kFullValidation)); |
| |
| // Last good APN should NOT be updated because we're connecting a DUN APN. |
| EXPECT_CALL(*service, SetLastGoodApn(_)).Times(0); |
| |
| // Service state should transition to Associating. |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateConnected)); |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| EXPECT_CALL(*service, SetFailure(_)).Times(0); |
| EXPECT_CALL(*service, SetFailureSilent(_)).Times(0); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| device_->ConnectTetheringAsDefaultPdn(future.GetCallback()); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Operation doesn't finish yet. |
| EXPECT_FALSE(future.IsReady()); |
| |
| // Setup new bearer with the connected DUN APN. |
| auto bearer2 = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer2->set_apn_type_for_testing(ApnList::ApnType::kDun); |
| bearer2->set_data_interface_for_testing(kTestInterfaceName); |
| bearer2->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDun, std::move(bearer2)); |
| |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| // Setup new network with the connected DUN APN. |
| auto network2 = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network2.get(); |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDun); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network2), |
| Cellular::LinkState::kDown); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))); |
| |
| // Service would get the attached network updated, and state transitions to |
| // Configuring. |
| EXPECT_CALL(*service, AttachNetwork(IsWeakPtrTo(default_pdn_))); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateAssociating)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)); |
| |
| // The multiplexed interface property name will be repopulated. |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestInterfaceName)); |
| |
| // 2nd step: simulate receiving a default link UP event via rtnl |
| device_->DefaultLinkUp(); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Operation doesn't finish yet. |
| EXPECT_FALSE(future.IsReady()); |
| |
| // State check will be called in two different contexts. The first time will |
| // control the transition to Connected state (so must be not connected first). |
| // The second time controls whether portal detection should be started (so |
| // must be connected). |
| { |
| ::testing::InSequence s; |
| EXPECT_CALL(*service, state()) |
| .WillRepeatedly(Return(Service::kStateConfiguring)); |
| // Service will transition to Connected. |
| EXPECT_CALL(*service, SetState(Service::kStateConnected)); |
| |
| EXPECT_CALL(*service, state()) |
| .WillRepeatedly(Return(Service::kStateConnected)); |
| } |
| |
| // 3rd step: Network reports connection updated |
| device_->OnConnectionUpdated(kTestInterfaceIndex); |
| |
| // Operation should have finished already. |
| EXPECT_TRUE(future.IsReady()); |
| EXPECT_EQ(future.Get<Network*>(), default_pdn_); |
| EXPECT_TRUE(future.Get<Error>().IsSuccess()); |
| |
| Mock::VerifyAndClearExpectations(service); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_DunAsDefaultFailedBearerConnect) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Will request stop of the current default network. |
| EXPECT_CALL(*default_pdn_, Stop()); |
| |
| // Will request disconnection of all bearers via capability, TWICE because |
| // the new connection with DUN APN will also fail. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)) |
| .WillRepeatedly(Invoke(this, &CellularTest::InvokeDisconnect)); |
| |
| // Primary multiplexed interface name will be cleared up. |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")); |
| |
| // Will request reconnection with the DUN APN |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(KeyValueStoreHasApn("apn-dun"), _, _)) |
| .WillOnce( |
| Invoke([](const KeyValueStore& props, RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| // Connect operation with DUN FAILS. |
| std::move(callback).Run(kTestBearerDBusPath, |
| Error(Error::kOperationFailed)); |
| })); |
| |
| // The recovery logic after the failure will reconnect with DEFAULT APN |
| EXPECT_CALL(*mm1_simple_proxy_, |
| Connect(KeyValueStoreHasApn("apn-default"), _, _)) |
| .WillOnce(Invoke([](const KeyValueStore& props, |
| RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| // Connect operation with DUN FAILS. |
| std::move(callback).Run(kTestBearerDBusPath, Error(Error::kSuccess)); |
| })); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Metrics for the DUN APN should be reported. |
| EXPECT_CALL(metrics_, |
| NotifyCellularConnectionResult( |
| Error::kOperationFailed, |
| Metrics::DetailedCellularConnectionResult::APNType::kDUN)); |
| |
| // The recovery logic after the failure will report successful metrics. |
| EXPECT_CALL( |
| metrics_, |
| NotifyCellularConnectionResult( |
| Error::kSuccess, |
| Metrics::DetailedCellularConnectionResult::APNType::kDefault)); |
| |
| // Service is connected. |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateConnected)); |
| |
| // Operation finishes with an error. |
| base::test::TestFuture<Network*, const Error&> future; |
| device_->ConnectTetheringAsDefaultPdn(future.GetCallback()); |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| |
| Mock::VerifyAndClearExpectations(service); |
| } |
| |
| TEST_F(CellularTest, |
| AcquireTetheringNetwork_OperationType_DunAsDefault_OperatorRequired) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // Device supports multiplexing more than one PDN. |
| device_->SetMaxActiveMultiplexedBearers(4); |
| |
| // Operator requires DUN as DEFAULT |
| EXPECT_CALL(*mock_mobile_operator_info_, use_dun_apn_as_default()) |
| .WillRepeatedly(Return(true)); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // DEFAULT is connected. |
| service->SetLastGoodApn(apn1); |
| |
| // Test only the operation type selection. |
| EXPECT_EQ(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kConnectDunAsDefaultPdn); |
| } |
| |
| TEST_F(CellularTest, |
| AcquireTetheringNetwork_OperationType_DunAsDefault_NoMultiplex) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // Operator does not require DUN as DEFAULT. |
| EXPECT_CALL(*mock_mobile_operator_info_, use_dun_apn_as_default()) |
| .WillRepeatedly(Return(false)); |
| |
| // Device doesn't support multiplexing more than one PDN. |
| device_->SetMaxActiveMultiplexedBearers(1); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // DEFAULT is connected. |
| service->SetLastGoodApn(apn1); |
| |
| // Test only the operation type selection. |
| EXPECT_EQ(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kConnectDunAsDefaultPdn); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_DunMultiplexed) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestMultiplexedInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex, |
| kTestMultiplexedInterfaceName, |
| Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Setup bearer with the connected DEFAULT APN. |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Will NOT request stop of the current default network. |
| EXPECT_CALL(*default_pdn_, Stop()).Times(0); |
| |
| // Will NOT request disconnection of all bearers via capability. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)).Times(0); |
| |
| // Primary multiplexed interface name will NOT be changed in any way. |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| // Will request a new connection with the DUN APN. |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillOnce( |
| Invoke([](const KeyValueStore& props, RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(props.Get<uint32_t>(CellularBearer::kMMApnTypeProperty), |
| MM_BEARER_APN_TYPE_TETHERING); |
| EXPECT_EQ(props.Get<std::string>(CellularBearer::kMMApnProperty), |
| "apn-dun"); |
| std::move(callback).Run(kTestBearerDBusPath2, Error()); |
| })); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| // 1st step: run ConnectMultiplexedTetheringPdn() until the new PDN |
| // connection is connected and we're requested to |
| // EstablishMultiplexedTetheringLink(). |
| device_->set_skip_establish_link_for_testing(true); |
| |
| // Metrics for the DUN APN should be reported. |
| EXPECT_CALL(metrics_, |
| NotifyCellularConnectionResult( |
| Error::kSuccess, |
| Metrics::DetailedCellularConnectionResult::APNType::kDUN)); |
| |
| // Portal detection is supported. |
| ON_CALL(*service, GetNetworkValidationMode) |
| .WillByDefault(Return(NetworkMonitor::ValidationMode::kFullValidation)); |
| |
| // Last good APN should NOT be updated because we're connecting a DUN APN. |
| EXPECT_CALL(*service, SetLastGoodApn(_)).Times(0); |
| |
| // Service state should NOT be updated in any way. |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateConnected)); |
| EXPECT_CALL(*service, SetState(_)).Times(0); |
| EXPECT_CALL(*service, SetFailure(_)).Times(0); |
| EXPECT_CALL(*service, SetFailureSilent(_)).Times(0); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| device_->ConnectMultiplexedTetheringPdn(future.GetCallback()); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Operation doesn't finish yet. |
| EXPECT_FALSE(future.IsReady()); |
| |
| // Setup new bearer with the connected DUN APN. |
| auto bearer2 = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath2, ""); |
| bearer2->set_apn_type_for_testing(ApnList::ApnType::kDun); |
| bearer2->set_data_interface_for_testing(kTestMultiplexedInterfaceName2); |
| bearer2->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDun, std::move(bearer2)); |
| |
| // Setup new network with the connected DUN APN. |
| auto network2 = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex2, |
| kTestMultiplexedInterfaceName2, |
| Technology::kCellular); |
| tethering_pdn_ = network2.get(); |
| device_->SetMultiplexedTetheringPdnForTesting( |
| kTestBearerDBusPath2, std::move(network2), Cellular::LinkState::kDown); |
| EXPECT_CALL(*tethering_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))); |
| |
| // Service state should NOT be updated in any way. |
| EXPECT_CALL(*service, AttachNetwork(_)).Times(0); |
| EXPECT_CALL(*service, SetState(_)).Times(0); |
| |
| // The multiplexed interface property name will NOT be repopulated. |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, _)) |
| .Times(0); |
| |
| // 2nd step: simulate receiving a link UP event via rtnl |
| device_->MultiplexedTetheringLinkUp(); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Operation doesn't finish yet. |
| EXPECT_FALSE(future.IsReady()); |
| |
| // 3rd step: Network reports connection updated |
| device_->OnConnectionUpdated(kTestMultiplexedInterfaceIndex2); |
| |
| // Operation should have finished already. |
| EXPECT_TRUE(future.IsReady()); |
| EXPECT_EQ(future.Get<Network*>(), tethering_pdn_); |
| EXPECT_TRUE(future.Get<Error>().IsSuccess()); |
| |
| Mock::VerifyAndClearExpectations(service); |
| Mock::VerifyAndClearExpectations(default_pdn_); |
| } |
| |
| TEST_F(CellularTest, |
| AcquireTetheringNetwork_DunMultiplexed_AbortOnDisconnected) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestMultiplexedInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_modem_state_for_testing( |
| Cellular::ModemState::kModemStateConnected); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex, |
| kTestMultiplexedInterfaceName, |
| Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Setup bearer with the connected DEFAULT APN. |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Will request a new connection with the DUN APN, but the request will be |
| // aborted via the disconnect call, so we mimic that with an error. |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillOnce(Invoke([this](const KeyValueStore& props, |
| RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(props.Get<uint32_t>(CellularBearer::kMMApnTypeProperty), |
| MM_BEARER_APN_TYPE_TETHERING); |
| EXPECT_EQ(props.Get<std::string>(CellularBearer::kMMApnProperty), |
| "apn-dun"); |
| // We suddenly are reported a disconnection |
| device_->OnModemStateChanged(Cellular::kModemStateRegistered); |
| // Fail to simulate the abort. |
| std::move(callback).Run(RpcIdentifier(""), |
| Error(Error::kOperationFailed)); |
| })); |
| |
| // Must request disconnection of all bearers via capability when the abort |
| // is being processed |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| base::test::TestFuture<Network*, const Error&> future; |
| device_->ConnectMultiplexedTetheringPdn(future.GetCallback()); |
| |
| // Operation should have finished already. |
| EXPECT_TRUE(future.IsReady()); |
| EXPECT_EQ(future.Get<Network*>(), nullptr); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| } |
| |
| TEST_F(CellularTest, AcquireTetheringNetwork_OperationType_DunMultiplexed) { |
| CellularService* service = SetRegisteredWithService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| SetMockMobileOperatorInfoObjects(); |
| CHECK(mock_mobile_operator_info_); |
| EXPECT_CALL(*mock_mobile_operator_info_, |
| tethering_allowed(false /*allow_untested_carriers*/)) |
| .WillRepeatedly(Return(true)); |
| |
| // Operator does not require DUN as DEFAULT. |
| EXPECT_CALL(*mock_mobile_operator_info_, use_dun_apn_as_default()) |
| .WillRepeatedly(Return(false)); |
| |
| // Device supports multiplexing more than one PDN. |
| device_->SetMaxActiveMultiplexedBearers(2); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // DEFAULT is connected. |
| service->SetLastGoodApn(apn1); |
| |
| // Test only the operation type selection. |
| EXPECT_EQ(device_->GetTetheringOperationType(false /*experimental_tethering*/, |
| nullptr), |
| Cellular::TetheringOperationType::kConnectDunMultiplexed); |
| } |
| |
| TEST_F(CellularTest, ReleaseTetheringNetwork_NoOp) { |
| SetRegisteredWithService(); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| base::test::TestFuture<const Error&> future; |
| device_->ReleaseTetheringNetwork(default_pdn_, future.GetCallback()); |
| EXPECT_TRUE(future.Get<Error>().IsSuccess()); |
| } |
| |
| TEST_F(CellularTest, ReleaseTetheringNetwork_DunAsDefault) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| // Currently connected PDN is of type DUN. |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDun); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Will request stop of the current DUN as DEFAULT network. |
| EXPECT_CALL(*default_pdn_, Stop()); |
| |
| // Will request disconnection of all bearers via capability. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| |
| // Primary multiplexed interface name will be cleared up. |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")); |
| |
| // Will request reconnection with the DEFAULT APN. |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillOnce( |
| Invoke([](const KeyValueStore& props, RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(props.Get<uint32_t>(CellularBearer::kMMApnTypeProperty), |
| MM_BEARER_APN_TYPE_DEFAULT); |
| EXPECT_EQ(props.Get<std::string>(CellularBearer::kMMApnProperty), |
| "apn-default"); |
| // When reconnecting back DEFAULT service should still be connected. |
| std::move(callback).Run(kTestBearerDBusPath, Error()); |
| })); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| // 1st step: run ReleaseTetheringNetwork() until the new PDN connection is |
| // connected and we're requested to EstablishLink(). |
| device_->set_skip_establish_link_for_testing(true); |
| |
| // Metrics for the DEFAULT APN should be reported. |
| EXPECT_CALL( |
| metrics_, |
| NotifyCellularConnectionResult( |
| Error::kSuccess, |
| Metrics::DetailedCellularConnectionResult::APNType::kDefault)); |
| |
| // Portal detection is supported. |
| ON_CALL(*service, GetNetworkValidationMode) |
| .WillByDefault(Return(NetworkMonitor::ValidationMode::kFullValidation)); |
| |
| // Last good APN should be updated because we're connecting a DEFAULT APN. |
| EXPECT_CALL(*service, SetLastGoodApn(_)); |
| |
| // Service state should transition to Associating. |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateConnected)); |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| EXPECT_CALL(*service, SetFailure(_)).Times(0); |
| EXPECT_CALL(*service, SetFailureSilent(_)).Times(0); |
| |
| base::test::TestFuture<const Error&> future; |
| device_->ReleaseTetheringNetwork(default_pdn_, future.GetCallback()); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Setup new bearer with the connected default APN. |
| auto bearer2 = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer2->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer2->set_data_interface_for_testing(kTestInterfaceName); |
| bearer2->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer2)); |
| |
| device_->set_state_for_testing(Cellular::State::kConnected); |
| |
| // Setup new network with the connected default APN. |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network2 = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network2.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network2), |
| Cellular::LinkState::kDown); |
| EXPECT_CALL(*default_pdn_, |
| Start(Field(&Network::StartOptions::dhcp, Optional(_)))); |
| |
| // Service would get the attached network updated, and state transitions to |
| // Configuring. |
| EXPECT_CALL(*service, AttachNetwork(IsWeakPtrTo(default_pdn_))); |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateAssociating)); |
| EXPECT_CALL(*service, SetState(Service::kStateConfiguring)); |
| |
| // The multiplexed interface property name will be repopulated. |
| EXPECT_CALL(*adaptor, EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, |
| kTestInterfaceName)); |
| |
| // 2nd step: simulate receiving a default link UP event via rtnl |
| device_->DefaultLinkUp(); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Operation doesn't finish yet. |
| EXPECT_FALSE(future.IsReady()); |
| |
| // State check will be called in two different contexts. The first time will |
| // control the transition to Connected state (so must be not connected first). |
| // The second time controls whether portal detection should be started (so |
| // must be connected). |
| { |
| ::testing::InSequence s; |
| EXPECT_CALL(*service, state()) |
| .WillRepeatedly(Return(Service::kStateConfiguring)); |
| // Service will transition to Connected. |
| EXPECT_CALL(*service, SetState(Service::kStateConnected)); |
| |
| EXPECT_CALL(*service, state()) |
| .WillRepeatedly(Return(Service::kStateConnected)); |
| } |
| |
| // 3rd step: Network reports connection updated |
| device_->OnConnectionUpdated(kTestInterfaceIndex); |
| |
| // Operation should have finished already. |
| EXPECT_TRUE(future.IsReady()); |
| EXPECT_TRUE(future.Get<Error>().IsSuccess()); |
| |
| Mock::VerifyAndClearExpectations(service); |
| } |
| |
| TEST_F(CellularTest, ReleaseTetheringNetwork_DunAsDefaultFailedConnect) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| // Currently connected PDN is of type DUN. |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDun); |
| auto network = std::make_unique<MockNetwork>( |
| kTestInterfaceIndex, kTestInterfaceName, Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| // Separate DEFAULT and DUN APNs. |
| Stringmaps apn_list; |
| Stringmap apn1, apn2; |
| apn1[kApnProperty] = "apn-default"; |
| apn1[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDefault}); |
| apn1[kApnSourceProperty] = kApnSourceMoDb; |
| apn1[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecFalse; |
| apn2[kApnProperty] = "apn-dun"; |
| apn2[kApnTypesProperty] = ApnList::JoinApnTypes({kApnTypeDun}); |
| apn2[kApnSourceProperty] = kApnSourceMoDb; |
| apn2[kApnIsRequiredByCarrierSpecProperty] = kApnIsRequiredByCarrierSpecTrue; |
| apn_list.push_back(apn1); |
| apn_list.push_back(apn2); |
| device_->SetApnList(apn_list); |
| |
| // Will request stop of the current DUN as DEFAULT network. |
| EXPECT_CALL(*default_pdn_, Stop()); |
| |
| // Will request disconnection of all bearers via capability. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(_, _, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| |
| // Primary multiplexed interface name will be cleared up. |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")); |
| |
| // Will request reconnection with the DEFAULT APN. |
| EXPECT_CALL(*service, SetState(Service::kStateAssociating)); |
| EXPECT_CALL(*mm1_simple_proxy_, Connect(_, _, _)) |
| .WillOnce( |
| Invoke([](const KeyValueStore& props, RpcIdentifierCallback callback, |
| base::TimeDelta timeout) { |
| EXPECT_EQ(props.Get<uint32_t>(CellularBearer::kMMApnTypeProperty), |
| MM_BEARER_APN_TYPE_DEFAULT); |
| EXPECT_EQ(props.Get<std::string>(CellularBearer::kMMApnProperty), |
| "apn-default"); |
| // This operation WILL fail in this test. |
| std::move(callback).Run(RpcIdentifier(), |
| Error(Error::kOperationNotAllowed)); |
| })); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Metrics for the DEFAULT APN should be reported. |
| EXPECT_CALL( |
| metrics_, |
| NotifyCellularConnectionResult( |
| Error::kOperationNotAllowed, |
| Metrics::DetailedCellularConnectionResult::APNType::kDefault)); |
| |
| // Service will be reported a failure. |
| EXPECT_CALL(*service, SetFailure(_)).Times(1); |
| |
| base::test::TestFuture<const Error&> future; |
| device_->ReleaseTetheringNetwork(default_pdn_, future.GetCallback()); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Device must be disconnected after the failure. |
| VerifyDisconnect(); |
| |
| // Operation should have finished already. |
| EXPECT_TRUE(future.IsReady()); |
| EXPECT_TRUE(future.Get<Error>().IsFailure()); |
| |
| Mock::VerifyAndClearExpectations(service); |
| } |
| |
| TEST_F(CellularTest, ReleaseTetheringNetwork_DunMultiplexed) { |
| MockCellularService* service = SetMockService(); |
| device_->SetPrimaryMultiplexedInterface(kTestMultiplexedInterfaceName); |
| device_->SetServiceState(Service::kStateConnected); |
| device_->set_state_for_testing(Cellular::State::kLinked); |
| device_->set_selected_service_for_testing(service); |
| ASSERT_NE(device_->service(), nullptr); |
| |
| // Default PDN. |
| device_->set_default_pdn_apn_type_for_testing(ApnList::ApnType::kDefault); |
| auto network = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex, |
| kTestMultiplexedInterfaceName, |
| Technology::kCellular); |
| default_pdn_ = network.get(); |
| device_->SetDefaultPdnForTesting(kTestBearerDBusPath, std::move(network), |
| Cellular::LinkState::kUp); |
| |
| auto bearer = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath, ""); |
| bearer->set_apn_type_for_testing(ApnList::ApnType::kDefault); |
| bearer->set_data_interface_for_testing(kTestMultiplexedInterfaceName); |
| bearer->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDefault, std::move(bearer)); |
| |
| // DUN PDN. |
| auto network2 = std::make_unique<MockNetwork>(kTestMultiplexedInterfaceIndex2, |
| kTestMultiplexedInterfaceName2, |
| Technology::kCellular); |
| tethering_pdn_ = network2.get(); |
| device_->SetMultiplexedTetheringPdnForTesting( |
| kTestBearerDBusPath2, std::move(network2), Cellular::LinkState::kUp); |
| |
| auto bearer2 = std::make_unique<CellularBearer>(&control_interface_, |
| kTestBearerDBusPath2, ""); |
| bearer2->set_apn_type_for_testing(ApnList::ApnType::kDun); |
| bearer2->set_data_interface_for_testing(kTestMultiplexedInterfaceName2); |
| bearer2->set_ipv4_config_method_for_testing( |
| CellularBearer::IPConfigMethod::kDHCP); |
| SetCapability3gppActiveBearer(ApnList::ApnType::kDun, std::move(bearer2)); |
| |
| // Will NOT request stop of the current DEFAULT network. |
| EXPECT_CALL(*default_pdn_, Stop()).Times(0); |
| |
| // Will request stop of the current DUN network. |
| EXPECT_CALL(*tethering_pdn_, Stop()); |
| |
| // Will request disconnection of the DUN bearer via capability. |
| EXPECT_CALL(*mm1_simple_proxy_, Disconnect(kTestBearerDBusPath2, _, _)) |
| .WillOnce(Invoke(this, &CellularTest::InvokeDisconnect)); |
| |
| // Primary multiplexed interface name will NOT be cleared up. |
| auto* adaptor = static_cast<DeviceMockAdaptor*>(device_->adaptor()); |
| EXPECT_CALL(*adaptor, |
| EmitStringChanged(kPrimaryMultiplexedInterfaceProperty, "")) |
| .Times(0); |
| |
| SetCapability3gppModemSimpleProxy(); |
| |
| // Service state should NOT change in any way. |
| ON_CALL(*service, state()).WillByDefault(Return(Service::kStateConnected)); |
| EXPECT_CALL(*service, SetState(_)).Times(0); |
| EXPECT_CALL(*service, SetFailure(_)).Times(0); |
| EXPECT_CALL(*service, SetFailureSilent(_)).Times(0); |
| |
| base::test::TestFuture<const Error&> future; |
| device_->ReleaseTetheringNetwork(tethering_pdn_, future.GetCallback()); |
| Mock::VerifyAndClearExpectations(adaptor); |
| |
| // Operation should have finished already. |
| EXPECT_TRUE(future.IsReady()); |
| EXPECT_TRUE(future.Get<Error>().IsSuccess()); |
| |
| Mock::VerifyAndClearExpectations(service); |
| Mock::VerifyAndClearExpectations(default_pdn_); |
| } |
| |
| TEST_F(CellularTest, ShouldBringNetworkInterfaceDownAfterDisabled) { |
| auto device_id = std::make_unique<DeviceId>( |
| cellular::kFM101BusType, cellular::kFM101Vid, cellular::kFM101Pid); |
| device_->SetDeviceId(std::move(device_id)); |
| EXPECT_FALSE(device_->ShouldBringNetworkInterfaceDownAfterDisabled()); |
| |
| device_id = std::make_unique<DeviceId>( |
| cellular::kL850GLBusType, cellular::kL850GLVid, cellular::kL850GLPid); |
| device_->SetDeviceId(std::move(device_id)); |
| EXPECT_TRUE(device_->ShouldBringNetworkInterfaceDownAfterDisabled()); |
| } |
| |
| TEST_F(CellularTest, IsModemUnknown) { |
| EXPECT_FALSE(device_->IsModemL850GL()); |
| EXPECT_FALSE(device_->IsModemFM101()); |
| EXPECT_FALSE(device_->IsModemFM350()); |
| } |
| |
| TEST_F(CellularTest, IsModemOther) { |
| auto device_id = |
| std::make_unique<DeviceId>(DeviceId::BusType::kUsb, 0x0001, 0x0002); |
| device_->SetDeviceId(std::move(device_id)); |
| EXPECT_FALSE(device_->IsModemL850GL()); |
| EXPECT_FALSE(device_->IsModemFM101()); |
| EXPECT_FALSE(device_->IsModemFM350()); |
| } |
| |
| TEST_F(CellularTest, IsModemL850GL) { |
| auto device_id = std::make_unique<DeviceId>( |
| cellular::kL850GLBusType, cellular::kL850GLVid, cellular::kL850GLPid); |
| device_->SetDeviceId(std::move(device_id)); |
| EXPECT_TRUE(device_->IsModemL850GL()); |
| EXPECT_FALSE(device_->IsModemFM101()); |
| EXPECT_FALSE(device_->IsModemFM350()); |
| } |
| |
| TEST_F(CellularTest, IsModemFM101) { |
| auto device_id = std::make_unique<DeviceId>( |
| cellular::kFM101BusType, cellular::kFM101Vid, cellular::kFM101Pid); |
| device_->SetDeviceId(std::move(device_id)); |
| EXPECT_TRUE(device_->IsModemFM101()); |
| EXPECT_FALSE(device_->IsModemL850GL()); |
| EXPECT_FALSE(device_->IsModemFM350()); |
| } |
| |
| TEST_F(CellularTest, IsModemFM350) { |
| auto device_id = std::make_unique<DeviceId>( |
| cellular::kFM350BusType, cellular::kFM350Vid, cellular::kFM350Pid); |
| device_->SetDeviceId(std::move(device_id)); |
| EXPECT_TRUE(device_->IsModemFM350()); |
| EXPECT_FALSE(device_->IsModemFM101()); |
| EXPECT_FALSE(device_->IsModemL850GL()); |
| } |
| |
| TEST_F(CellularTest, FirmwareSupportsTetheringUnknownModem) { |
| device_->SetFirmwareRevision("1.2.3.4.5"); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| } |
| |
| TEST_F(CellularTest, FirmwareSupportsTetheringFM350) { |
| auto device_id = std::make_unique<DeviceId>( |
| cellular::kFM350BusType, cellular::kFM350Vid, cellular::kFM350Pid); |
| device_->SetDeviceId(std::move(device_id)); |
| // Unsupported FWs |
| const std::string mr1Generic = "81600.0000.00.29.19.16"; |
| const std::string mr1Carrier = "81600.0000.00.29.19.16_GC\r\nC82"; |
| |
| device_->SetFirmwareRevision(mr1Generic); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr1Carrier); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| |
| // Supported FWs |
| const std::string mr2_21Generic = "81600.0000.00.29.21.21"; |
| const std::string mr2_21Carrier = "81600.0000.00.29.21.21_GC\r\nD66"; |
| const std::string mr2_24Generic = "81600.0000.00.29.21.24"; |
| const std::string mr2_24Carrier = "81600.0000.00.29.21.24_GC\r\nD66"; |
| const std::string mr4Generic = "81600.0000.00.29.23.06"; |
| const std::string mr4Carrier = "81600.0000.00.29.23.06_GC\r\nD66"; |
| |
| device_->SetFirmwareRevision(mr2_21Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr2_21Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr2_24Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr2_24Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr4Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr4Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| } |
| |
| TEST_F(CellularTest, FirmwareSupportsTetheringL850GL) { |
| auto device_id = std::make_unique<DeviceId>( |
| cellular::kL850GLBusType, cellular::kL850GLVid, cellular::kL850GLPid); |
| device_->SetDeviceId(std::move(device_id)); |
| // Unsupported FWs |
| const std::string mr2Generic = "18500.5001.00.02.24.09"; |
| const std::string mr2Carrier = "18500.5001.30.02.24.09"; |
| const std::string mr3Generic = "18500.5001.00.03.25.18"; |
| const std::string mr3Carrier = "18500.5001.02.03.25.18"; |
| const std::string mr4_01Generic = "18500.5001.00.04.26.01"; |
| const std::string mr4_01Carrier = "18500.5001.20.04.26.01"; |
| const std::string mr4_06Generic = "18500.5001.00.04.26.06"; |
| const std::string mr4_06Carrier = "18500.5001.07.04.26.06"; |
| |
| device_->SetFirmwareRevision(mr2Generic); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr2Carrier); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr3Generic); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr3Carrier); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr4_01Generic); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr4_01Carrier); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr4_06Generic); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr4_06Carrier); |
| EXPECT_FALSE(device_->FirmwareSupportsTethering()); |
| |
| // Supported FWs |
| const std::string mr5_12Generic = "18500.5001.00.05.27.12"; |
| const std::string mr5_12Carrier = "18500.5001.30.05.27.12"; |
| const std::string mr5_16Generic = "18500.5001.00.05.27.16"; |
| const std::string mr5_16Carrier = "18500.5001.30.05.27.16"; |
| const std::string mr5_17Generic = "18500.5001.00.05.27.17"; |
| const std::string mr5_17Carrier = "18500.5001.30.05.27.17"; |
| const std::string mr6Generic = "18500.5001.00.06.28.01"; |
| const std::string mr6Carrier = "18500.5001.30.06.28.01"; |
| const std::string mr8Generic = "18500.5001.00.07.29.08"; |
| const std::string mr8Carrier = "18500.5001.30.07.29.08"; |
| |
| device_->SetFirmwareRevision(mr5_12Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr5_12Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr5_16Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr5_16Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr5_17Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr5_17Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr6Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr6Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr8Generic); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| device_->SetFirmwareRevision(mr8Carrier); |
| EXPECT_TRUE(device_->FirmwareSupportsTethering()); |
| } |
| |
| } // namespace shill |