 Chromium Code Reviews
 Chromium Code Reviews Issue 12096071:
  Adding a unit test to verify the IPC channel between the network and desktop processes.  (Closed) 
  Base URL: svn://svn.chromium.org/chrome/trunk/src
    
  
    Issue 12096071:
  Adding a unit test to verify the IPC channel between the network and desktop processes.  (Closed) 
  Base URL: svn://svn.chromium.org/chrome/trunk/src| Index: remoting/host/ipc_desktop_environment_unittest.cc | 
| diff --git a/remoting/host/ipc_desktop_environment_unittest.cc b/remoting/host/ipc_desktop_environment_unittest.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..437c0dc6a2d18477f259259bbbe471905a40a8d1 | 
| --- /dev/null | 
| +++ b/remoting/host/ipc_desktop_environment_unittest.cc | 
| @@ -0,0 +1,325 @@ | 
| +// Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
| 
Sergey Ulanov
2013/02/04 21:04:41
2013 please
 
alexeypa (please no reviews)
2013/02/04 23:45:58
Done.
 | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +#include "base/bind.h" | 
| +#include "base/bind_helpers.h" | 
| +#include "base/callback.h" | 
| +#include "base/memory/ref_counted.h" | 
| +#include "base/memory/scoped_ptr.h" | 
| +#include "base/memory/weak_ptr.h" | 
| +#include "base/message_loop.h" | 
| +#include "base/process.h" | 
| +#include "base/process_util.h" | 
| +#include "base/run_loop.h" | 
| +#include "base/win/scoped_handle.h" | 
| +#include "ipc/ipc_channel.h" | 
| +#include "ipc/ipc_channel_proxy.h" | 
| +#include "ipc/ipc_listener.h" | 
| +#include "ipc/ipc_message.h" | 
| +#include "ipc/ipc_platform_file.h" | 
| +#include "media/video/capture/screen/screen_capturer_fake.h" | 
| +#include "media/video/capture/screen/screen_capturer_mock_objects.h" | 
| +#include "remoting/base/auto_thread.h" | 
| +#include "remoting/base/auto_thread_task_runner.h" | 
| +#include "remoting/host/chromoting_messages.h" | 
| +#include "remoting/host/desktop_process.h" | 
| +#include "remoting/host/desktop_session_connector.h" | 
| +#include "remoting/host/desktop_session_proxy.h" | 
| +#include "remoting/host/host_mock_objects.h" | 
| +#include "remoting/host/ipc_desktop_environment.h" | 
| +#include "remoting/protocol/protocol_mock_objects.h" | 
| +#include "testing/gmock/include/gmock/gmock.h" | 
| +#include "testing/gtest/include/gtest/gtest.h" | 
| + | 
| +using testing::_; | 
| +using testing::AnyNumber; | 
| +using testing::Return; | 
| + | 
| +namespace remoting { | 
| + | 
| +namespace { | 
| + | 
| +class MockDaemonListener : public IPC::Listener { | 
| + public: | 
| + MockDaemonListener() {} | 
| + virtual ~MockDaemonListener() {} | 
| + | 
| + virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE; | 
| + | 
| + MOCK_METHOD1(OnDesktopAttached, void(IPC::PlatformFileForTransit)); | 
| + MOCK_METHOD1(OnChannelConnected, void(int32)); | 
| + MOCK_METHOD0(OnChannelError, void()); | 
| + | 
| + private: | 
| + DISALLOW_COPY_AND_ASSIGN(MockDaemonListener); | 
| +}; | 
| + | 
| +bool MockDaemonListener::OnMessageReceived(const IPC::Message& message) { | 
| + bool handled = true; | 
| + IPC_BEGIN_MESSAGE_MAP(MockDaemonListener, message) | 
| + IPC_MESSAGE_HANDLER(ChromotingDesktopDaemonMsg_DesktopAttached, | 
| + OnDesktopAttached) | 
| + IPC_MESSAGE_UNHANDLED(handled = false) | 
| + IPC_END_MESSAGE_MAP() | 
| + | 
| + EXPECT_TRUE(handled); | 
| + return handled; | 
| +} | 
| + | 
| +} // namespace | 
| + | 
| +class IpcDesktopEnvironmentTest | 
| + : public testing::Test, | 
| + public DesktopSessionConnector { | 
| + public: | 
| + IpcDesktopEnvironmentTest(); | 
| + virtual ~IpcDesktopEnvironmentTest(); | 
| + | 
| + virtual void SetUp() OVERRIDE; | 
| + virtual void TearDown() OVERRIDE; | 
| + | 
| + // DesktopSessionConnector implementation. | 
| + virtual void ConnectTerminal( | 
| + scoped_refptr<DesktopSessionProxy> desktop_session_proxy) OVERRIDE; | 
| + virtual void DisconnectTerminal( | 
| + scoped_refptr<DesktopSessionProxy> desktop_session_proxy) OVERRIDE; | 
| + virtual void OnDesktopSessionAgentAttached( | 
| + int terminal_id, | 
| + base::ProcessHandle desktop_process, | 
| + IPC::PlatformFileForTransit desktop_pipe) OVERRIDE; | 
| + virtual void OnTerminalDisconnected(int terminal_id) OVERRIDE; | 
| + | 
| + // Creates a DesktopEnvironment with a fake media::ScreenCapturer, to mock | 
| + // DesktopEnvironmentFactory::Create(). | 
| + DesktopEnvironment* CreateDesktopEnvironment(); | 
| + | 
| + // Creates a dummy EventExecutor, to mock | 
| + // DesktopEnvironment::CreateEventExecutor(). | 
| + EventExecutor* CreateEventExecutor(); | 
| + | 
| + // Creates a fake media::ScreenCapturer, to mock | 
| + // DesktopEnvironment::CreateVideoCapturer(). | 
| + media::ScreenCapturer* CreateVideoCapturer(); | 
| + | 
| + void DeleteDesktopEnvironment(); | 
| + | 
| + protected: | 
| + void OnDisconnectCallback(); | 
| + | 
| + // Invoked when ChromotingDesktopDaemonMsg_DesktopAttached message is | 
| + // received. | 
| + void OnDesktopAttached(IPC::PlatformFileForTransit desktop_pipe); | 
| + | 
| + // Invoked when the daemon-to-desktop channel is closed. | 
| + void OnDesktopSessionClosed(); | 
| + | 
| + MessageLoop message_loop_; | 
| + base::RunLoop run_loop_; | 
| + scoped_refptr<AutoThreadTaskRunner> task_runner_; | 
| + | 
| + // Factory for weak pointers to DesktopSessionConnector interface. | 
| + base::WeakPtrFactory<DesktopSessionConnector> connector_factory_; | 
| + | 
| + // The daemons's end of the daemon-to-desktop channel. | 
| + scoped_ptr<IPC::ChannelProxy> daemon_channel_; | 
| + | 
| + // Name of the daemon-to-desktop channel. | 
| + std::string daemon_channel_name_; | 
| + | 
| + // Delegate that is passed to |daemon_channel_|. | 
| + MockDaemonListener daemon_listener_; | 
| + | 
| + scoped_ptr<IpcDesktopEnvironment> desktop_environment_; | 
| + | 
| + // Event executor created by |desktop_environment_|. | 
| + scoped_ptr<EventExecutor> event_executor_; | 
| + | 
| + // Screen capturer created by |desktop_environment_|. | 
| + scoped_ptr<media::ScreenCapturer> video_capturer_; | 
| + | 
| + // Represents the desktop process running in a user session. | 
| + scoped_ptr<DesktopProcess> desktop_process_; | 
| + | 
| + // Points to the DesktopSessionProxy instance created by | 
| + // IpdDesktopEnvironment. | 
| + scoped_refptr<DesktopSessionProxy> desktop_session_proxy_; | 
| + | 
| + media::MockScreenCapturerDelegate screen_capturer_delegate_; | 
| +}; | 
| + | 
| +IpcDesktopEnvironmentTest::IpcDesktopEnvironmentTest() | 
| + : message_loop_(MessageLoop::TYPE_UI), | 
| + connector_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) { | 
| +} | 
| + | 
| +IpcDesktopEnvironmentTest::~IpcDesktopEnvironmentTest() { | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::SetUp() { | 
| + // Arrange to run |message_loop_| until no components depend on it. | 
| + task_runner_ = new AutoThreadTaskRunner( | 
| + message_loop_.message_loop_proxy(), run_loop_.QuitClosure()); | 
| + | 
| + scoped_refptr<AutoThreadTaskRunner> io_task_runner = | 
| + AutoThread::CreateWithType("IPC thread", task_runner_, | 
| + MessageLoop::TYPE_IO); | 
| + | 
| + // Set expectation that the DaemonProcess will send DesktopAttached message | 
| + // once it is ready. | 
| + EXPECT_CALL(daemon_listener_, OnChannelConnected(_)); | 
| + EXPECT_CALL(daemon_listener_, OnDesktopAttached(_)) | 
| + .WillOnce(Invoke(this, &IpcDesktopEnvironmentTest::OnDesktopAttached)); | 
| + EXPECT_CALL(daemon_listener_, OnChannelError()) | 
| + .Times(AnyNumber()) | 
| + .WillOnce(Invoke(this, | 
| + &IpcDesktopEnvironmentTest::OnDesktopSessionClosed)); | 
| + | 
| + // Create the daemon end of the daemon-to-desktop channel. | 
| + daemon_channel_name_ = IPC::Channel::GenerateUniqueRandomChannelID(); | 
| + daemon_channel_.reset(new IPC::ChannelProxy( | 
| + IPC::ChannelHandle(daemon_channel_name_), | 
| + IPC::Channel::MODE_SERVER, | 
| + &daemon_listener_, | 
| + io_task_runner)); | 
| + | 
| + // Create an IpcDesktopEnvironment instance. | 
| + desktop_environment_.reset(new IpcDesktopEnvironment( | 
| + task_runner_, io_task_runner, "user@domain/rest-of-jid", | 
| + base::Bind(&IpcDesktopEnvironmentTest::OnDisconnectCallback, | 
| + base::Unretained(this)), | 
| + connector_factory_.GetWeakPtr())); | 
| + | 
| + // Create the event executor. | 
| + scoped_ptr<protocol::MockClipboardStub> clipboard_stub( | 
| + new protocol::MockClipboardStub()); | 
| + EXPECT_CALL(*clipboard_stub, InjectClipboardEvent(_)) | 
| + .Times(0); | 
| + | 
| + event_executor_ = | 
| + desktop_environment_->CreateEventExecutor(task_runner_, task_runner_); | 
| + event_executor_->Start(clipboard_stub.PassAs<protocol::ClipboardStub>()); | 
| + | 
| + // Create the screen capturer. | 
| + video_capturer_ = | 
| + desktop_environment_->CreateVideoCapturer(task_runner_, task_runner_); | 
| + video_capturer_->Start(&screen_capturer_delegate_); | 
| 
Sergey Ulanov
2013/02/04 21:04:41
It looks wrong that all this code is in SetUp() me
 
alexeypa (please no reviews)
2013/02/04 23:45:58
Done.
 | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::TearDown() { | 
| 
Sergey Ulanov
2013/02/04 21:04:41
nit: No need to override this method if it's empty
 
alexeypa (please no reviews)
2013/02/04 23:45:58
Done.
 | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::ConnectTerminal( | 
| + scoped_refptr<DesktopSessionProxy> desktop_session_proxy) { | 
| + EXPECT_TRUE(!desktop_process_); | 
| + EXPECT_TRUE(!desktop_session_proxy_.get()); | 
| + EXPECT_TRUE(task_runner_.get()); | 
| + | 
| + desktop_session_proxy_ = desktop_session_proxy; | 
| + | 
| + // Create and start the desktop process. | 
| + desktop_process_.reset(new DesktopProcess(task_runner_, | 
| + daemon_channel_name_)); | 
| + | 
| + scoped_ptr<MockDesktopEnvironmentFactory> desktop_environment_factory( | 
| + new MockDesktopEnvironmentFactory()); | 
| + EXPECT_CALL(*desktop_environment_factory, CreatePtr()) | 
| + .Times(AnyNumber()) | 
| + .WillRepeatedly(Invoke( | 
| + this, &IpcDesktopEnvironmentTest::CreateDesktopEnvironment)); | 
| + EXPECT_CALL(*desktop_environment_factory, SupportsAudioCapture()) | 
| + .Times(AnyNumber()) | 
| + .WillRepeatedly(Return(false)); | 
| + | 
| + // TODO(alexeypa): Fix DesktopProcess to use the desktop environment | 
| + // to create the disconnect window instead of directly calling | 
| + // DisconnectWindow::Create(). This will take care of "Uninteresting mock | 
| + // function call" warnings printed when DisconnectWindow::Show() and | 
| + // DisconnectWindow::Hide() are called. | 
| + EXPECT_TRUE(desktop_process_->Start(desktop_environment_factory.Pass())); | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::DisconnectTerminal( | 
| + scoped_refptr<DesktopSessionProxy> desktop_session_proxy) { | 
| + EXPECT_TRUE(desktop_session_proxy_.get()); | 
| + EXPECT_EQ(desktop_session_proxy_.get(), desktop_session_proxy.get()); | 
| + | 
| + desktop_session_proxy_ = NULL; | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::OnDesktopSessionAgentAttached( | 
| + int terminal_id, | 
| + base::ProcessHandle desktop_process, | 
| + IPC::PlatformFileForTransit desktop_pipe) { | 
| + NOTIMPLEMENTED(); | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::OnTerminalDisconnected(int terminal_id) { | 
| + NOTIMPLEMENTED(); | 
| +} | 
| + | 
| +DesktopEnvironment* IpcDesktopEnvironmentTest::CreateDesktopEnvironment() { | 
| + MockDesktopEnvironment* desktop_environment = new MockDesktopEnvironment(); | 
| + EXPECT_CALL(*desktop_environment, CreateAudioCapturerPtr(_)) | 
| + .Times(0); | 
| + EXPECT_CALL(*desktop_environment, CreateEventExecutorPtr(_, _)) | 
| + .Times(AnyNumber()) | 
| + .WillRepeatedly( | 
| + InvokeWithoutArgs(this, | 
| + &IpcDesktopEnvironmentTest::CreateEventExecutor)); | 
| + EXPECT_CALL(*desktop_environment, CreateVideoCapturerPtr(_, _)) | 
| + .Times(AnyNumber()) | 
| + .WillRepeatedly( | 
| + InvokeWithoutArgs(this, | 
| + &IpcDesktopEnvironmentTest::CreateVideoCapturer)); | 
| + | 
| + return desktop_environment; | 
| +} | 
| + | 
| +EventExecutor* IpcDesktopEnvironmentTest::CreateEventExecutor() { | 
| + MockEventExecutor* event_executor = new MockEventExecutor(); | 
| + EXPECT_CALL(*event_executor, StartPtr(_)); | 
| + return event_executor; | 
| +} | 
| + | 
| +media::ScreenCapturer* IpcDesktopEnvironmentTest::CreateVideoCapturer() { | 
| + return new media::ScreenCapturerFake(); | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::DeleteDesktopEnvironment() { | 
| + desktop_environment_.reset(); | 
| + event_executor_.reset(); | 
| + video_capturer_.reset(); | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::OnDisconnectCallback() { | 
| + NOTIMPLEMENTED(); | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::OnDesktopAttached( | 
| + IPC::PlatformFileForTransit desktop_pipe) { | 
| + // Instruct DesktopSessionProxy to connect to the network-to-desktop pipe. | 
| + EXPECT_TRUE(desktop_session_proxy_->AttachToDesktop( | 
| + base::GetCurrentProcessHandle(), desktop_pipe)); | 
| + | 
| + // Once |desktop_session_proxy_| is attahced to the desktop, request a single | 
| + // frame to be captured. | 
| + video_capturer_->CaptureFrame(); | 
| 
Sergey Ulanov
2013/02/04 21:04:41
I think it's better to move this call to the body
 
alexeypa (please no reviews)
2013/02/04 23:45:58
Done.
 | 
| +} | 
| + | 
| +void IpcDesktopEnvironmentTest::OnDesktopSessionClosed() { | 
| + daemon_channel_.reset(); | 
| + desktop_process_.reset(); | 
| +} | 
| + | 
| +TEST_F(IpcDesktopEnvironmentTest, Basic) { | 
| + // Wait for the first frame to be captured and exit. | 
| + EXPECT_CALL(screen_capturer_delegate_, OnCaptureCompleted(_)) | 
| + .WillOnce(InvokeWithoutArgs( | 
| + this, &IpcDesktopEnvironmentTest::DeleteDesktopEnvironment)); | 
| + | 
| + task_runner_ = NULL; | 
| + run_loop_.Run(); | 
| +} | 
| + | 
| +} // namespace remoting |