blob: 10dda96f341e021339bc6cc257bfdf5839fd43c5 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/base/network_change_notifier_linux.h"
#include "base/bind.h"
#include "base/message_loop_proxy.h"
#include "base/synchronization/waitable_event.h"
#include "dbus/mock_bus.h"
#include "dbus/mock_object_proxy.h"
#include "dbus/message.h"
#include "dbus/object_path.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace net {
using testing::_;
using testing::DoAll;
using testing::InvokeWithoutArgs;
using testing::Return;
using testing::SaveArg;
class NetworkChangeNotifierLinuxTest : public testing::Test {
protected:
// A subset of the NetworkManager-defined constants used in
// the tests below. See network_change_notifier_linux.cc
// for the full list.
enum {
NM_STATE_DISCONNECTED = 20,
NM_STATE_DISCONNECTING = 30,
NM_STATE_CONNECTED_SITE = 70,
NM_STATE_CONNECTED_GLOBAL = 70
};
NetworkChangeNotifierLinuxTest()
: initialized_(false, false) {}
virtual void SetUp() {
dbus::Bus::Options options;
options.bus_type = dbus::Bus::SYSTEM;
mock_bus_ = new dbus::MockBus(options);
mock_object_proxy_ = new dbus::MockObjectProxy(
mock_bus_.get(),
"service_name",
dbus::ObjectPath("service_path"));
EXPECT_CALL(*mock_bus_, GetObjectProxyWithOptions(_, _, _))
.WillOnce(Return(mock_object_proxy_.get()));
EXPECT_CALL(*mock_object_proxy_, CallMethod(_, _, _))
.WillOnce(SaveArg<2>(&response_callback_));
EXPECT_CALL(*mock_object_proxy_, ConnectToSignal(_, _, _, _))
.WillOnce(
DoAll(
SaveArg<2>(&signal_callback_),
InvokeWithoutArgs(
this,
&NetworkChangeNotifierLinuxTest::Initialize)));
notifier_.reset(NetworkChangeNotifierLinux::CreateForTest(mock_bus_.get()));
initialized_.Wait();
}
void Initialize() {
notifier_thread_proxy_ = base::MessageLoopProxy::current();
initialized_.Signal();
}
void RunOnNotifierThread(const base::Closure& callback) {
base::WaitableEvent event(false, false);
notifier_thread_proxy_->PostTask(FROM_HERE, base::Bind(
&RunOnNotifierThreadHelper, callback, &event));
event.Wait();
// Run any tasks queued on the main thread, e.g. by
// ObserverListThreadSafe.
MessageLoop::current()->RunAllPending();
}
void SendResponse(uint32 state) {
scoped_ptr<dbus::Response> response(dbus::Response::CreateEmpty());
dbus::MessageWriter writer(response.get());
writer.AppendVariantOfUint32(state);
RunOnNotifierThread(base::Bind(response_callback_, response.get()));
}
void SendSignal(uint32 state) {
dbus::Signal signal("org.freedesktop.NetworkManager", "StateChanged");
dbus::MessageWriter writer(&signal);
writer.AppendUint32(state);
RunOnNotifierThread(base::Bind(signal_callback_, &signal));
}
dbus::ObjectProxy::ResponseCallback response_callback_;
dbus::ObjectProxy::SignalCallback signal_callback_;
// Allows creating a new NetworkChangeNotifier. Must be created before
// |notifier_| and destroyed after it to avoid DCHECK failures.
NetworkChangeNotifier::DisableForTest disable_for_test_;
scoped_ptr<NetworkChangeNotifier> notifier_;
private:
static void RunOnNotifierThreadHelper(const base::Closure& callback,
base::WaitableEvent* event) {
callback.Run();
event->Signal();
}
base::WaitableEvent initialized_;
// Valid only after initialized_ is signaled.
scoped_refptr<base::MessageLoopProxy> notifier_thread_proxy_;
scoped_refptr<dbus::MockBus> mock_bus_;
scoped_refptr<dbus::MockObjectProxy> mock_object_proxy_;
};
namespace {
class OfflineObserver : public NetworkChangeNotifier::ConnectionTypeObserver {
public:
OfflineObserver()
: notification_count(0),
last_online_value(true) {
NetworkChangeNotifier::AddConnectionTypeObserver(this);
}
~OfflineObserver() {
NetworkChangeNotifier::RemoveConnectionTypeObserver(this);
}
virtual void OnConnectionTypeChanged(
NetworkChangeNotifier::ConnectionType type) OVERRIDE {
notification_count++;
last_online_value = type != NetworkChangeNotifier::CONNECTION_NONE;
}
int notification_count;
bool last_online_value;
};
TEST_F(NetworkChangeNotifierLinuxTest, Offline) {
SendResponse(NM_STATE_DISCONNECTED);
EXPECT_EQ(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
}
TEST_F(NetworkChangeNotifierLinuxTest, Online) {
SendResponse(NM_STATE_CONNECTED_GLOBAL);
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());}
TEST_F(NetworkChangeNotifierLinuxTest, OfflineThenOnline) {
OfflineObserver observer;
SendResponse(NM_STATE_DISCONNECTED);
EXPECT_EQ(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(0, observer.notification_count);
SendSignal(NM_STATE_CONNECTED_GLOBAL);
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(1, observer.notification_count);
EXPECT_TRUE(observer.last_online_value);
}
TEST_F(NetworkChangeNotifierLinuxTest, MultipleStateChanges) {
OfflineObserver observer;
SendResponse(NM_STATE_CONNECTED_GLOBAL);
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(0, observer.notification_count);
SendSignal(NM_STATE_DISCONNECTED);
EXPECT_EQ(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(1, observer.notification_count);
EXPECT_FALSE(observer.last_online_value);
SendSignal(NM_STATE_CONNECTED_GLOBAL);
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(2, observer.notification_count);
EXPECT_TRUE(observer.last_online_value);
}
TEST_F(NetworkChangeNotifierLinuxTest, IgnoreContinuedOnlineState) {
OfflineObserver observer;
SendResponse(NM_STATE_CONNECTED_SITE);
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(0, observer.notification_count);
SendSignal(NM_STATE_CONNECTED_GLOBAL);
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(0, observer.notification_count);
}
TEST_F(NetworkChangeNotifierLinuxTest, IgnoreContinuedOfflineState) {
OfflineObserver observer;
SendResponse(NM_STATE_DISCONNECTING);
EXPECT_EQ(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(0, observer.notification_count);
SendSignal(NM_STATE_DISCONNECTED);
EXPECT_EQ(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
EXPECT_EQ(0, observer.notification_count);
}
TEST_F(NetworkChangeNotifierLinuxTest, NullResponse) {
RunOnNotifierThread(base::Bind(
response_callback_, static_cast<dbus::Response*>(NULL)));
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
}
TEST_F(NetworkChangeNotifierLinuxTest, EmptyResponse) {
scoped_ptr<dbus::Response> response(dbus::Response::CreateEmpty());
RunOnNotifierThread(base::Bind(response_callback_, response.get()));
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
}
TEST_F(NetworkChangeNotifierLinuxTest, InvalidResponse) {
scoped_ptr<dbus::Response> response(dbus::Response::CreateEmpty());
dbus::MessageWriter writer(response.get());
writer.AppendUint16(20); // Uint16 instead of the expected Uint32
RunOnNotifierThread(base::Bind(response_callback_, response.get()));
EXPECT_NE(NetworkChangeNotifier::CONNECTION_NONE,
NetworkChangeNotifier::GetConnectionType());
}
} // namespace
} // namespace net