blob: 9a9140fc388035b7b46d319c33b42533b7a2a631 [file] [log] [blame]
zijiehe36763d82017-06-19 23:52:311// Copyright 2017 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "remoting/host/desktop_session_agent.h"
6
7#include <memory>
8
9#include "base/bind.h"
Sebastien Marchand17fa2782019-01-25 19:28:1010#include "base/bind_helpers.h"
zijiehe36763d82017-06-19 23:52:3111#include "base/callback.h"
12#include "base/location.h"
13#include "base/memory/ref_counted.h"
14#include "base/memory/weak_ptr.h"
zijiehe36763d82017-06-19 23:52:3115#include "base/run_loop.h"
16#include "base/single_thread_task_runner.h"
Gabriel Charettec7108742019-08-23 03:31:4017#include "base/test/task_environment.h"
zijiehef401c5182017-07-19 00:21:4618#include "base/threading/thread_task_runner_handle.h"
zijiehe36763d82017-06-19 23:52:3119#include "ipc/ipc_channel_proxy.h"
20#include "ipc/ipc_listener.h"
21#include "remoting/base/auto_thread_task_runner.h"
22#include "remoting/host/chromoting_messages.h"
23#include "remoting/host/desktop_environment_options.h"
24#include "remoting/host/fake_desktop_environment.h"
25#include "remoting/host/screen_resolution.h"
26#include "testing/gtest/include/gtest/gtest.h"
27
28namespace remoting {
29
30namespace {
31
32class FakeDelegate : public DesktopSessionAgent::Delegate {
33 public:
34 FakeDelegate(scoped_refptr<base::SingleThreadTaskRunner> runner);
35 ~FakeDelegate() override = default;
36
37 DesktopEnvironmentFactory& desktop_environment_factory() override {
38 return factory_;
39 }
40
41 void OnNetworkProcessDisconnected() override {}
42
43 base::WeakPtr<Delegate> GetWeakPtr() { return weak_ptr_.GetWeakPtr(); }
44
45 private:
46 FakeDesktopEnvironmentFactory factory_;
47
Jeremy Roman7c5cfabd2019-08-12 15:45:2748 base::WeakPtrFactory<FakeDelegate> weak_ptr_{this};
zijiehe36763d82017-06-19 23:52:3149};
50
51FakeDelegate::FakeDelegate(scoped_refptr<base::SingleThreadTaskRunner> runner)
Jeremy Roman7c5cfabd2019-08-12 15:45:2752 : factory_(runner) {}
zijiehe36763d82017-06-19 23:52:3153
54class ProcessStatsListener : public IPC::Listener {
55 public:
Evan Stadeaf37ba32020-06-26 20:36:3756 ProcessStatsListener(base::RepeatingClosure action_after_received)
zijiehe36763d82017-06-19 23:52:3157 : action_after_received_(action_after_received) {}
58
59 ~ProcessStatsListener() override = default;
60
61 private:
62 // IPC::Listener implementation.
63 bool OnMessageReceived(const IPC::Message& message) override;
64
65 void OnProcessResourceUsage(
66 const remoting::protocol::AggregatedProcessResourceUsage& usage);
67
Evan Stadeaf37ba32020-06-26 20:36:3768 const base::RepeatingClosure action_after_received_;
zijiehe36763d82017-06-19 23:52:3169};
70
71bool ProcessStatsListener::OnMessageReceived(const IPC::Message& message) {
72 bool handled = false;
73 IPC_BEGIN_MESSAGE_MAP(ProcessStatsListener, message)
74 IPC_MESSAGE_HANDLER(ChromotingAnyToNetworkMsg_ReportProcessStats,
75 OnProcessResourceUsage);
76 IPC_MESSAGE_UNHANDLED(handled = false);
77 IPC_END_MESSAGE_MAP()
78 return handled;
79}
80
81void ProcessStatsListener::OnProcessResourceUsage(
82 const remoting::protocol::AggregatedProcessResourceUsage& usage) {
83 action_after_received_.Run();
84}
85
86} // namespace
87
88class DesktopSessionAgentTest : public ::testing::Test {
89 public:
90 DesktopSessionAgentTest();
91 ~DesktopSessionAgentTest() override = default;
92
93 void Shutdown();
94
95 protected:
Gabriel Charettedf0a7442019-09-05 17:32:3596 base::test::SingleThreadTaskEnvironment task_environment_;
zijiehe36763d82017-06-19 23:52:3197 base::RunLoop run_loop_;
98 scoped_refptr<AutoThreadTaskRunner> task_runner_;
99 scoped_refptr<DesktopSessionAgent> agent_;
100};
101
102DesktopSessionAgentTest::DesktopSessionAgentTest()
Gabriel Charettedfa36042019-08-19 17:30:11103 : task_runner_(
104 new AutoThreadTaskRunner(task_environment_.GetMainThreadTaskRunner(),
105 run_loop_.QuitClosure())),
Carlos Caballero1df57eb22019-05-30 17:07:12106 agent_(new DesktopSessionAgent(task_runner_,
107 task_runner_,
108 task_runner_,
109 task_runner_)) {}
zijiehe36763d82017-06-19 23:52:31110
111void DesktopSessionAgentTest::Shutdown() {
112 task_runner_ = nullptr;
113 agent_->Stop();
114 agent_ = nullptr;
115}
116
117TEST_F(DesktopSessionAgentTest, StartProcessStatsReport) {
118 std::unique_ptr<FakeDelegate> delegate(new FakeDelegate(task_runner_));
119 std::unique_ptr<IPC::ChannelProxy> proxy;
Evan Stade94ffcb62020-07-28 18:51:19120 ProcessStatsListener listener(base::BindRepeating(
121 [](DesktopSessionAgentTest* test, std::unique_ptr<FakeDelegate>* delegate,
122 std::unique_ptr<IPC::ChannelProxy>* proxy) {
zijiehe36763d82017-06-19 23:52:31123 test->Shutdown();
124 delegate->reset();
125 proxy->reset();
126 },
Evan Stade94ffcb62020-07-28 18:51:19127 base::Unretained(this), base::Unretained(&delegate),
zijiehe36763d82017-06-19 23:52:31128 base::Unretained(&proxy)));
129 proxy = IPC::ChannelProxy::Create(
130 agent_->Start(delegate->GetWeakPtr()).release(),
Hajime Hoshiff15e972017-11-09 06:37:09131 IPC::Channel::MODE_CLIENT, &listener, task_runner_,
132 base::ThreadTaskRunnerHandle::Get());
zijiehe36763d82017-06-19 23:52:31133 ASSERT_TRUE(proxy->Send(new ChromotingNetworkDesktopMsg_StartSessionAgent(
134 "jid", ScreenResolution(), DesktopEnvironmentOptions())));
135 ASSERT_TRUE(proxy->Send(new ChromotingNetworkToAnyMsg_StartProcessStatsReport(
136 base::TimeDelta::FromMilliseconds(1))));
137 run_loop_.Run();
138}
139
140TEST_F(DesktopSessionAgentTest, StartProcessStatsReportWithInvalidInterval) {
141 std::unique_ptr<FakeDelegate> delegate(new FakeDelegate(task_runner_));
142 std::unique_ptr<IPC::ChannelProxy> proxy;
Peter Kasting341e1fb2018-02-24 00:03:01143 ProcessStatsListener listener{base::DoNothing()};
zijiehe36763d82017-06-19 23:52:31144 proxy = IPC::ChannelProxy::Create(
145 agent_->Start(delegate->GetWeakPtr()).release(),
Hajime Hoshiff15e972017-11-09 06:37:09146 IPC::Channel::MODE_CLIENT, &listener, task_runner_,
147 base::ThreadTaskRunnerHandle::Get());
zijiehe36763d82017-06-19 23:52:31148 ASSERT_TRUE(proxy->Send(new ChromotingNetworkDesktopMsg_StartSessionAgent(
149 "jid", ScreenResolution(), DesktopEnvironmentOptions())));
150 ASSERT_TRUE(proxy->Send(new ChromotingNetworkToAnyMsg_StartProcessStatsReport(
151 base::TimeDelta::FromMilliseconds(-1))));
152 ASSERT_TRUE(proxy->Send(
153 new ChromotingNetworkToAnyMsg_StopProcessStatsReport()));
kylechar2caf4d62019-02-15 20:03:42154 task_runner_->PostDelayedTask(
155 FROM_HERE,
156 base::BindOnce(
157 [](DesktopSessionAgentTest* test,
158 std::unique_ptr<FakeDelegate>* delegate,
159 std::unique_ptr<IPC::ChannelProxy>* proxy) {
160 test->Shutdown();
161 delegate->reset();
162 proxy->reset();
163 },
164 base::Unretained(this), base::Unretained(&delegate),
165 base::Unretained(&proxy)),
zijiehe36763d82017-06-19 23:52:31166 base::TimeDelta::FromMilliseconds(1));
167 run_loop_.Run();
168}
169
170TEST_F(DesktopSessionAgentTest, StartThenStopProcessStatsReport) {
171 std::unique_ptr<FakeDelegate> delegate(new FakeDelegate(task_runner_));
172 std::unique_ptr<IPC::ChannelProxy> proxy;
Peter Kasting341e1fb2018-02-24 00:03:01173 ProcessStatsListener listener{base::DoNothing()};
zijiehe36763d82017-06-19 23:52:31174 proxy = IPC::ChannelProxy::Create(
175 agent_->Start(delegate->GetWeakPtr()).release(),
Hajime Hoshiff15e972017-11-09 06:37:09176 IPC::Channel::MODE_CLIENT, &listener, task_runner_,
177 base::ThreadTaskRunnerHandle::Get());
zijiehe36763d82017-06-19 23:52:31178 ASSERT_TRUE(proxy->Send(new ChromotingNetworkDesktopMsg_StartSessionAgent(
179 "jid", ScreenResolution(), DesktopEnvironmentOptions())));
180 ASSERT_TRUE(proxy->Send(new ChromotingNetworkToAnyMsg_StartProcessStatsReport(
181 base::TimeDelta::FromMilliseconds(1))));
182 ASSERT_TRUE(proxy->Send(
183 new ChromotingNetworkToAnyMsg_StopProcessStatsReport()));
kylechar2caf4d62019-02-15 20:03:42184 task_runner_->PostDelayedTask(
185 FROM_HERE,
186 base::BindOnce(
187 [](DesktopSessionAgentTest* test,
188 std::unique_ptr<FakeDelegate>* delegate,
189 std::unique_ptr<IPC::ChannelProxy>* proxy) {
190 test->Shutdown();
191 delegate->reset();
192 proxy->reset();
193 },
194 base::Unretained(this), base::Unretained(&delegate),
195 base::Unretained(&proxy)),
zijiehe36763d82017-06-19 23:52:31196 base::TimeDelta::FromMilliseconds(1));
197 run_loop_.Run();
198}
199
zijiehef401c5182017-07-19 00:21:46200TEST_F(DesktopSessionAgentTest, SendAggregatedProcessResourceUsage) {
201 std::unique_ptr<IPC::Channel> receiver;
202 std::unique_ptr<IPC::Channel> sender;
Evan Stade94ffcb62020-07-28 18:51:19203 ProcessStatsListener listener(base::BindRepeating(
204 [](DesktopSessionAgentTest* test, std::unique_ptr<IPC::Channel>* receiver,
205 std::unique_ptr<IPC::Channel>* sender) {
zijiehef401c5182017-07-19 00:21:46206 test->Shutdown();
207 base::ThreadTaskRunnerHandle::Get()->DeleteSoon(
208 FROM_HERE, receiver->release());
209 base::ThreadTaskRunnerHandle::Get()->DeleteSoon(
210 FROM_HERE, sender->release());
211 },
Evan Stade94ffcb62020-07-28 18:51:19212 base::Unretained(this), base::Unretained(&receiver),
zijiehef401c5182017-07-19 00:21:46213 base::Unretained(&sender)));
214 mojo::MessagePipe pipe;
215 receiver = IPC::Channel::CreateServer(
216 pipe.handle1.release(),
217 &listener,
218 task_runner_);
219 ASSERT_TRUE(receiver->Connect());
220 sender = IPC::Channel::CreateClient(
221 pipe.handle0.release(),
222 &listener,
223 task_runner_);
224 ASSERT_TRUE(sender->Connect());
225 protocol::AggregatedProcessResourceUsage aggregated;
226 for (int i = 0; i < 2; i++) {
227 *aggregated.add_usages() = protocol::ProcessResourceUsage();
228 }
229 ASSERT_TRUE(sender->Send(
230 new ChromotingAnyToNetworkMsg_ReportProcessStats(aggregated)));
231 run_loop_.Run();
232}
233
234TEST_F(DesktopSessionAgentTest, SendEmptyAggregatedProcessResourceUsage) {
235 std::unique_ptr<IPC::Channel> receiver;
236 std::unique_ptr<IPC::Channel> sender;
Evan Stade94ffcb62020-07-28 18:51:19237 ProcessStatsListener listener(base::BindRepeating(
238 [](DesktopSessionAgentTest* test, std::unique_ptr<IPC::Channel>* receiver,
239 std::unique_ptr<IPC::Channel>* sender) {
zijiehef401c5182017-07-19 00:21:46240 test->Shutdown();
241 base::ThreadTaskRunnerHandle::Get()->DeleteSoon(
242 FROM_HERE, receiver->release());
243 base::ThreadTaskRunnerHandle::Get()->DeleteSoon(
244 FROM_HERE, sender->release());
245 },
Evan Stade94ffcb62020-07-28 18:51:19246 base::Unretained(this), base::Unretained(&receiver),
zijiehef401c5182017-07-19 00:21:46247 base::Unretained(&sender)));
248 mojo::MessagePipe pipe;
249 receiver = IPC::Channel::CreateServer(
250 pipe.handle1.release(),
251 &listener,
252 task_runner_);
253 ASSERT_TRUE(receiver->Connect());
254 sender = IPC::Channel::CreateClient(
255 pipe.handle0.release(),
256 &listener,
257 task_runner_);
258 ASSERT_TRUE(sender->Connect());
259 ASSERT_TRUE(sender->Send(new ChromotingAnyToNetworkMsg_ReportProcessStats(
260 protocol::AggregatedProcessResourceUsage())));
261 run_loop_.Run();
262}
263
zijiehe36763d82017-06-19 23:52:31264} // namespace remoting