worker_process_launcher_unittest.cc revision c2e0dbddbe15c98d52c4786dac06cb8952a8ae6d
15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/basictypes.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind.h"
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/memory/ref_counted.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/message_loop.h"
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/win/scoped_handle.h"
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ipc/ipc_channel.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ipc/ipc_channel_proxy.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ipc/ipc_listener.h"
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ipc/ipc_message.h"
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "remoting/base/auto_thread_task_runner.h"
152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "remoting/host/chromoting_messages.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "remoting/host/host_exit_codes.h"
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "remoting/host/win/launch_process_with_token.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "remoting/host/win/worker_process_launcher.h"
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "remoting/host/worker_process_ipc_delegate.h"
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "testing/gmock/include/gmock/gmock.h"
212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "testing/gmock_mutant.h"
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "testing/gtest/include/gtest/gtest.h"
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using base::win::ScopedHandle;
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::_;
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::AnyNumber;
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::CreateFunctor;
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::DoAll;
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::Expectation;
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::Invoke;
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::InvokeWithoutArgs;
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::Return;
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using testing::ReturnPointee;
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace remoting {
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char kIpcSecurityDescriptor[] = "D:(A;;GA;;;AU)";
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class MockProcessLauncherDelegate : public WorkerProcessLauncher::Delegate {
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MockProcessLauncherDelegate() {}
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual ~MockProcessLauncherDelegate() {}
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // IPC::Sender implementation.
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MOCK_METHOD1(Send, bool(IPC::Message*));
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // WorkerProcessLauncher::Delegate interface.
502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MOCK_METHOD0(CloseChannel, void());
512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MOCK_CONST_METHOD0(GetProcessId, DWORD());
522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MOCK_CONST_METHOD1(IsPermanentError, bool(int));
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MOCK_METHOD1(KillProcess, void(DWORD));
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MOCK_METHOD2(LaunchProcess, bool(IPC::Listener*, ScopedHandle*));
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(MockProcessLauncherDelegate);
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class MockIpcDelegate : public WorkerProcessIpcDelegate {
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MockIpcDelegate() {}
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual ~MockIpcDelegate() {}
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // WorkerProcessIpcDelegate interface.
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MOCK_METHOD1(OnChannelConnected, void(int32));
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MOCK_METHOD1(OnMessageReceived, bool(const IPC::Message&));
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  MOCK_METHOD0(OnPermanentError, void());
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(MockIpcDelegate);
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class MockWorkerListener : public IPC::Listener {
752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) public:
762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MockWorkerListener() {}
772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual ~MockWorkerListener() {}
782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MOCK_METHOD3(OnCrash, void(const std::string&, const std::string&, int));
802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // IPC::Listener implementation
822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE;
832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) private:
852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(MockWorkerListener);
862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)};
872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool MockWorkerListener::OnMessageReceived(const IPC::Message& message) {
892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool handled = true;
902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  IPC_BEGIN_MESSAGE_MAP(MockWorkerListener, message)
912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    IPC_MESSAGE_HANDLER(ChromotingDaemonMsg_Crash, OnCrash)
922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    IPC_MESSAGE_UNHANDLED(handled = false)
932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  IPC_END_MESSAGE_MAP()
942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_TRUE(handled);
962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return handled;
982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class WorkerProcessLauncherTest : public testing::Test {
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  WorkerProcessLauncherTest();
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual ~WorkerProcessLauncherTest();
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void SetUp() OVERRIDE;
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void TearDown() OVERRIDE;
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // WorkerProcessLauncher::Delegate mocks
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void KillProcess(DWORD exit_code);
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool LaunchProcess(IPC::Listener* delegate,
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                     ScopedHandle* process_exit_event_out);
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool LaunchProcessAndConnect(IPC::Listener* delegate,
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                               ScopedHandle* process_exit_event_out);
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool FailLaunchAndStopWorker(IPC::Listener* delegate,
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                               ScopedHandle* process_exit_event_out);
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Connects the client end of the channel (the worker process's end).
1202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  void ConnectClient();
1212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Disconnects the client end of the channel.
1232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  void DisconnectClient();
1242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Disconnects the server end of the channel (the launcher's end).
1262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  void DisconnectServer();
1272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Sends a message to the worker process.
1292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool SendToProcess(IPC::Message* message);
1302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Sends a fake message to the launcher.
1322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  void SendFakeMessageToLauncher();
1332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Requests the worker to crash.
1352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  void CrashWorker();
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Starts the worker.
1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void StartWorker();
1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Stops the worker.
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void StopWorker();
1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Quits |message_loop_|.
1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void QuitMainMessageLoop();
1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) protected:
147c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  base::MessageLoop message_loop_;
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_refptr<AutoThreadTaskRunner> task_runner_;
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Receives messages sent to the worker process.
1512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MockWorkerListener client_listener_;
1522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Receives messages sent from the worker process.
1542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  MockIpcDelegate server_listener_;
1552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Implements WorkerProcessLauncher::Delegate.
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_ptr<MockProcessLauncherDelegate> launcher_delegate_;
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The name of the IPC channel.
1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::string channel_name_;
1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Client and server ends of the IPC channel.
1635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_ptr<IPC::ChannelProxy> channel_client_;
1645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_ptr<IPC::ChannelProxy> channel_server_;
1655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Returned as the worker process PID to the launcher.
1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DWORD client_pid_;
1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The worker process launcher.
1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_ptr<WorkerProcessLauncher> launcher_;
1715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The event signalling termination of the worker process.
1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ScopedHandle process_exit_event_;
1745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // True if a permanent error condition should be emulated.
1765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool permanent_error_;
1775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
1785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)WorkerProcessLauncherTest::WorkerProcessLauncherTest()
180c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    : message_loop_(base::MessageLoop::TYPE_IO),
1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      client_pid_(GetCurrentProcessId()),
182c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      permanent_error_(false) {}
1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)WorkerProcessLauncherTest::~WorkerProcessLauncherTest() {
1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WorkerProcessLauncherTest::SetUp() {
1885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  task_runner_ = new AutoThreadTaskRunner(
1895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      message_loop_.message_loop_proxy(),
1905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      base::Bind(&WorkerProcessLauncherTest::QuitMainMessageLoop,
1915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                 base::Unretained(this)));
1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Set up process launcher delegate
1945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  launcher_delegate_.reset(new MockProcessLauncherDelegate());
1955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, Send(_))
1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(AnyNumber())
1972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillRepeatedly(Invoke(this, &WorkerProcessLauncherTest::SendToProcess));
1982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, CloseChannel())
1992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(AnyNumber())
2002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillRepeatedly(Invoke(this,
2012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                             &WorkerProcessLauncherTest::DisconnectServer));
2025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, GetProcessId())
2035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(AnyNumber())
2045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(ReturnPointee(&client_pid_));
2055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, IsPermanentError(_))
2065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(AnyNumber())
2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(ReturnPointee(&permanent_error_));
2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, KillProcess(_))
2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(AnyNumber())
2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(Invoke(this, &WorkerProcessLauncherTest::KillProcess));
2115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Set up IPC delegate.
2132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnMessageReceived(_))
2142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(0);
2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WorkerProcessLauncherTest::TearDown() {
2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WorkerProcessLauncherTest::KillProcess(DWORD exit_code) {
2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BOOL result = SetEvent(process_exit_event_);
2225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_TRUE(result);
2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool WorkerProcessLauncherTest::LaunchProcess(
2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IPC::Listener* delegate,
2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ScopedHandle* process_exit_event_out) {
2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  process_exit_event_.Set(CreateEvent(NULL, TRUE, FALSE, NULL));
2295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!process_exit_event_.IsValid())
2305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
2315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  channel_name_ = IPC::Channel::GenerateUniqueRandomChannelID();
2335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ScopedHandle pipe;
2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!CreateIpcChannel(channel_name_, kIpcSecurityDescriptor, &pipe)) {
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Wrap the pipe into an IPC channel.
2395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  channel_server_.reset(new IPC::ChannelProxy(
2405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      IPC::ChannelHandle(pipe),
2415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      IPC::Channel::MODE_SERVER,
2425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      delegate,
2435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      task_runner_));
2445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return DuplicateHandle(GetCurrentProcess(),
2465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                         process_exit_event_,
2475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                         GetCurrentProcess(),
2485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                         process_exit_event_out->Receive(),
2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                         0,
2505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                         FALSE,
2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                         DUPLICATE_SAME_ACCESS) != FALSE;
2525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool WorkerProcessLauncherTest::LaunchProcessAndConnect(
2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IPC::Listener* delegate,
2565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ScopedHandle* process_exit_event_out) {
2575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!LaunchProcess(delegate, process_exit_event_out))
2585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
2595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  task_runner_->PostTask(
2615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
2622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&WorkerProcessLauncherTest::ConnectClient,
2635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                 base::Unretained(this)));
2645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
2655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool WorkerProcessLauncherTest::FailLaunchAndStopWorker(
2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IPC::Listener* delegate,
2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ScopedHandle* process_exit_event_out) {
2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  task_runner_->PostTask(
2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      FROM_HERE,
2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      base::Bind(&WorkerProcessLauncherTest::StopWorker,
2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                 base::Unretained(this)));
2745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
2755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void WorkerProcessLauncherTest::ConnectClient() {
2785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  channel_client_.reset(new IPC::ChannelProxy(
2795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      IPC::ChannelHandle(channel_name_),
2805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      IPC::Channel::MODE_CLIENT,
2812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      &client_listener_,
2825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      task_runner_));
2832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Pretend that |kLaunchSuccessTimeoutSeconds| passed since launching
2852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // the worker process. This will make the backoff algorithm think that this
2862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // launch attempt was successful and it will not delay the next launch.
2872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  launcher_->ResetLaunchSuccessTimeoutForTest();
2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void WorkerProcessLauncherTest::DisconnectClient() {
2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  channel_client_.reset();
2925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void WorkerProcessLauncherTest::DisconnectServer() {
2952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  channel_server_.reset();
2962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
2972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool WorkerProcessLauncherTest::SendToProcess(IPC::Message* message) {
2992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (channel_server_)
3002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return channel_server_->Send(message);
3012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  delete message;
3032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return false;
3042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
3052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void WorkerProcessLauncherTest::SendFakeMessageToLauncher() {
3072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (channel_client_)
3082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    channel_client_->Send(new ChromotingDesktopNetworkMsg_DisconnectSession());
3092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
3102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void WorkerProcessLauncherTest::CrashWorker() {
3122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  launcher_->Crash(FROM_HERE);
3132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
3142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WorkerProcessLauncherTest::StartWorker() {
3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  launcher_.reset(new WorkerProcessLauncher(
3172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      task_runner_, launcher_delegate_.Pass(), &server_listener_));
3182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  launcher_->SetKillProcessTimeoutForTest(base::TimeDelta::FromMilliseconds(0));
3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WorkerProcessLauncherTest::StopWorker() {
3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  launcher_.reset();
3242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DisconnectClient();
3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  channel_name_.clear();
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  channel_server_.reset();
3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  task_runner_ = NULL;
3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WorkerProcessLauncherTest::QuitMainMessageLoop() {
331c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  message_loop_.PostTask(FROM_HERE, base::MessageLoop::QuitClosure());
3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TEST_F(WorkerProcessLauncherTest, Start) {
3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(1)
3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(Invoke(this, &WorkerProcessLauncherTest::LaunchProcess));
3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnChannelConnected(_))
3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
3412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnPermanentError())
3425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
3435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StartWorker();
3455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StopWorker();
3465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  message_loop_.Run();
3475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Starts and connects to the worker process. Expect OnChannelConnected to be
3505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// called.
3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TEST_F(WorkerProcessLauncherTest, StartAndConnect) {
3525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(1)
3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(Invoke(
3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect));
3565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnChannelConnected(_))
3585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(1)
3595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillOnce(InvokeWithoutArgs(this,
3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                  &WorkerProcessLauncherTest::StopWorker));
3612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnPermanentError())
3625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StartWorker();
3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  message_loop_.Run();
3665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Kills the worker process after the 1st connect and expects it to be
3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// restarted.
3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TEST_F(WorkerProcessLauncherTest, Restart) {
3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(2)
3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(Invoke(
3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect));
3755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Expectation first_connect =
3762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      EXPECT_CALL(server_listener_, OnChannelConnected(_))
3775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          .Times(2)
3785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          .WillOnce(InvokeWithoutArgs(CreateFunctor(
3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)              this, &WorkerProcessLauncherTest::KillProcess, CONTROL_C_EXIT)))
3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          .WillOnce(InvokeWithoutArgs(this,
3815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                      &WorkerProcessLauncherTest::StopWorker));
3825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnPermanentError())
3845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
3855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StartWorker();
3875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  message_loop_.Run();
3885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Drops the IPC channel to the worker process after the 1st connect and expects
3915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// the worker process to be restarted.
3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TEST_F(WorkerProcessLauncherTest, DropIpcChannel) {
3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
3945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(2)
3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(Invoke(
3965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect));
3975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Expectation first_connect =
3992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      EXPECT_CALL(server_listener_, OnChannelConnected(_))
4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          .Times(2)
4015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          .WillOnce(InvokeWithoutArgs(
4022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)              this, &WorkerProcessLauncherTest::DisconnectClient))
4035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          .WillOnce(InvokeWithoutArgs(
4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)              this, &WorkerProcessLauncherTest::StopWorker));
4055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnPermanentError())
4075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StartWorker();
4105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  message_loop_.Run();
4115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns a permanent error exit code and expects OnPermanentError() to be
4145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// invoked.
4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TEST_F(WorkerProcessLauncherTest, PermanentError) {
4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
4175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(1)
4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillRepeatedly(Invoke(
4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect));
4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnChannelConnected(_))
4225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(1)
4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillOnce(InvokeWithoutArgs(CreateFunctor(
4245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::KillProcess, CONTROL_C_EXIT)));
4252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnPermanentError())
4265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(1)
4275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillOnce(InvokeWithoutArgs(this,
4285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                  &WorkerProcessLauncherTest::StopWorker));
4295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  permanent_error_ = true;
4315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StartWorker();
4325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  message_loop_.Run();
4335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns invalid client PID and expects the connection to be rejected.
4365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TEST_F(WorkerProcessLauncherTest, InvalidClientPid) {
4375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
4385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(2)
4395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillOnce(Invoke(
4405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect))
4415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .WillOnce(Invoke(
4425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          this, &WorkerProcessLauncherTest::FailLaunchAndStopWorker));
4435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnChannelConnected(_))
4455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
4462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnPermanentError())
4475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      .Times(0);
4485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  client_pid_ = GetCurrentProcessId() + 4;
4505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StartWorker();
4515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  message_loop_.Run();
4525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Requests the worker to crash and expects it to honor the request.
4552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)TEST_F(WorkerProcessLauncherTest, Crash) {
4562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
4572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(2)
4582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillRepeatedly(Invoke(
4592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect));
4602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnChannelConnected(_))
4622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(2)
4632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillOnce(InvokeWithoutArgs(this,
4642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                  &WorkerProcessLauncherTest::CrashWorker))
4652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillOnce(InvokeWithoutArgs(this,
4662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                  &WorkerProcessLauncherTest::StopWorker));
4672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(client_listener_, OnCrash(_, _, _))
4692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(1)
4702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillOnce(InvokeWithoutArgs(CreateFunctor(
4712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          this, &WorkerProcessLauncherTest::KillProcess,
4722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          EXCEPTION_BREAKPOINT)));
4732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  StartWorker();
4752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  message_loop_.Run();
4762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
4772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Requests the worker to crash and terminates the worker even if it does not
4792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// comply.
4802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)TEST_F(WorkerProcessLauncherTest, CrashAnyway) {
4812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(*launcher_delegate_, LaunchProcess(_, _))
4822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(2)
4832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillRepeatedly(Invoke(
4842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          this, &WorkerProcessLauncherTest::LaunchProcessAndConnect));
4852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(server_listener_, OnChannelConnected(_))
4872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(2)
4882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillOnce(InvokeWithoutArgs(this,
4892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                  &WorkerProcessLauncherTest::CrashWorker))
4902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillOnce(InvokeWithoutArgs(this,
4912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                  &WorkerProcessLauncherTest::StopWorker));
4922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Ignore the crash request and try send another message to the launcher.
4942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  EXPECT_CALL(client_listener_, OnCrash(_, _, _))
4952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .Times(1)
4962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      .WillOnce(InvokeWithoutArgs(
4972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          this, &WorkerProcessLauncherTest::SendFakeMessageToLauncher));
4982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
4992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  StartWorker();
5002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  message_loop_.Run();
5012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
5022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
5035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace remoting
504