OLD | NEW |
1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include <stddef.h> | 5 #include <stddef.h> |
6 #include <stdint.h> | 6 #include <stdint.h> |
7 | 7 |
8 #include <memory> | 8 #include <memory> |
9 #include <utility> | 9 #include <utility> |
10 | 10 |
11 #include "base/bind.h" | 11 #include "base/bind.h" |
12 #include "base/guid.h" | 12 #include "base/guid.h" |
13 #include "base/macros.h" | 13 #include "base/macros.h" |
14 #include "base/memory/ptr_util.h" | 14 #include "base/memory/ptr_util.h" |
15 #include "base/process/process.h" | 15 #include "base/process/process.h" |
16 #include "base/run_loop.h" | 16 #include "base/run_loop.h" |
17 #include "base/test/test_suite.h" | 17 #include "base/test/test_suite.h" |
18 #include "mojo/public/cpp/bindings/binding_set.h" | 18 #include "mojo/public/cpp/bindings/binding_set.h" |
| 19 #include "services/service_manager/public/cpp/binder_registry.h" |
19 #include "services/service_manager/public/cpp/interface_factory.h" | 20 #include "services/service_manager/public/cpp/interface_factory.h" |
20 #include "services/service_manager/public/cpp/interface_registry.h" | |
21 #include "services/service_manager/public/cpp/service_test.h" | 21 #include "services/service_manager/public/cpp/service_test.h" |
22 #include "services/service_manager/public/interfaces/service_manager.mojom.h" | 22 #include "services/service_manager/public/interfaces/service_manager.mojom.h" |
23 #include "services/service_manager/tests/connect/connect_test.mojom.h" | 23 #include "services/service_manager/tests/connect/connect_test.mojom.h" |
24 #include "services/service_manager/tests/util.h" | 24 #include "services/service_manager/tests/util.h" |
25 | 25 |
26 // Tests that multiple services can be packaged in a single service by | 26 // Tests that multiple services can be packaged in a single service by |
27 // implementing ServiceFactory; that these services can be specified by | 27 // implementing ServiceFactory; that these services can be specified by |
28 // the package's manifest and are thus registered with the PackageManager. | 28 // the package's manifest and are thus registered with the PackageManager. |
29 | 29 |
30 namespace service_manager { | 30 namespace service_manager { |
(...skipping 27 matching lines...) Loading... |
58 void ReceiveConnectionResult(mojom::ConnectResult* out_result, | 58 void ReceiveConnectionResult(mojom::ConnectResult* out_result, |
59 Identity* out_target, | 59 Identity* out_target, |
60 base::RunLoop* loop, | 60 base::RunLoop* loop, |
61 int32_t in_result, | 61 int32_t in_result, |
62 const service_manager::Identity& in_identity) { | 62 const service_manager::Identity& in_identity) { |
63 *out_result = static_cast<mojom::ConnectResult>(in_result); | 63 *out_result = static_cast<mojom::ConnectResult>(in_result); |
64 *out_target = in_identity; | 64 *out_target = in_identity; |
65 loop->Quit(); | 65 loop->Quit(); |
66 } | 66 } |
67 | 67 |
| 68 void StartServiceResponse(base::RunLoop* quit_loop, |
| 69 mojom::ConnectResult* out_result, |
| 70 Identity* out_resolved_identity, |
| 71 mojom::ConnectResult result, |
| 72 const Identity& resolved_identity) { |
| 73 if (quit_loop) |
| 74 quit_loop->Quit(); |
| 75 if (out_result) |
| 76 *out_result = result; |
| 77 if (out_resolved_identity) |
| 78 *out_resolved_identity = resolved_identity; |
| 79 } |
| 80 |
68 void QuitLoop(base::RunLoop* loop) { | 81 void QuitLoop(base::RunLoop* loop) { |
69 loop->Quit(); | 82 loop->Quit(); |
70 } | 83 } |
71 | 84 |
72 } // namespace | 85 } // namespace |
73 | 86 |
74 class ConnectTest : public test::ServiceTest, | 87 class ConnectTest : public test::ServiceTest, |
75 public InterfaceFactory<test::mojom::ExposedInterface>, | 88 public InterfaceFactory<test::mojom::ExposedInterface>, |
76 public test::mojom::ExposedInterface { | 89 public test::mojom::ExposedInterface { |
77 public: | 90 public: |
78 ConnectTest() : ServiceTest("connect_unittests") {} | 91 ConnectTest() : ServiceTest("connect_unittests") {} |
79 ~ConnectTest() override {} | 92 ~ConnectTest() override {} |
80 | 93 |
81 protected: | 94 protected: |
82 std::unique_ptr<Connection> ConnectTo(const Identity& target) { | |
83 std::unique_ptr<Connection> connection = connector()->Connect(target); | |
84 base::RunLoop loop; | |
85 connection->AddConnectionCompletedClosure(base::Bind(&QuitLoop, &loop)); | |
86 loop.Run(); | |
87 return connection; | |
88 } | |
89 | |
90 void CompareConnectionState( | 95 void CompareConnectionState( |
91 const std::string& connection_local_name, | 96 const std::string& connection_local_name, |
92 const std::string& connection_remote_name, | 97 const std::string& connection_remote_name, |
93 const std::string& connection_remote_userid, | 98 const std::string& connection_remote_userid, |
94 const std::string& initialize_local_name, | 99 const std::string& initialize_local_name, |
95 const std::string& initialize_userid) { | 100 const std::string& initialize_userid) { |
96 EXPECT_EQ(connection_remote_name, | 101 EXPECT_EQ(connection_remote_name, |
97 connection_state_->connection_remote_name); | 102 connection_state_->connection_remote_name); |
98 EXPECT_EQ(connection_remote_userid, | 103 EXPECT_EQ(connection_remote_userid, |
99 connection_state_->connection_remote_userid); | 104 connection_state_->connection_remote_userid); |
100 EXPECT_EQ(initialize_local_name, connection_state_->initialize_local_name); | 105 EXPECT_EQ(initialize_local_name, connection_state_->initialize_local_name); |
101 EXPECT_EQ(initialize_userid, connection_state_->initialize_userid); | 106 EXPECT_EQ(initialize_userid, connection_state_->initialize_userid); |
102 } | 107 } |
103 | 108 |
104 private: | 109 private: |
105 class TestService : public test::ServiceTestClient { | 110 class TestService : public test::ServiceTestClient { |
106 public: | 111 public: |
107 explicit TestService(ConnectTest* connect_test) | 112 explicit TestService(ConnectTest* connect_test) |
108 : test::ServiceTestClient(connect_test), | 113 : test::ServiceTestClient(connect_test), connect_test_(connect_test) { |
109 connect_test_(connect_test) {} | 114 registry_.AddInterface<test::mojom::ExposedInterface>(connect_test_); |
| 115 } |
110 ~TestService() override {} | 116 ~TestService() override {} |
111 | 117 |
112 private: | 118 private: |
113 bool OnConnect(const ServiceInfo& remote_info, | 119 void OnBindInterface( |
114 InterfaceRegistry* registry) override { | 120 const ServiceInfo& source_info, |
115 registry->AddInterface<test::mojom::ExposedInterface>(connect_test_); | 121 const std::string& interface_name, |
116 return true; | 122 mojo::ScopedMessagePipeHandle interface_pipe) override { |
| 123 registry_.BindInterface(source_info.identity, interface_name, |
| 124 std::move(interface_pipe)); |
117 } | 125 } |
118 | 126 |
119 ConnectTest* connect_test_; | 127 ConnectTest* connect_test_; |
| 128 BinderRegistry registry_; |
120 | 129 |
121 DISALLOW_COPY_AND_ASSIGN(TestService); | 130 DISALLOW_COPY_AND_ASSIGN(TestService); |
122 }; | 131 }; |
123 | 132 |
124 // test::ServiceTest: | 133 // test::ServiceTest: |
125 void SetUp() override { | 134 void SetUp() override { |
126 test::ServiceTest::SetUp(); | 135 test::ServiceTest::SetUp(); |
127 // We need to connect to the package first to force the service manager to | 136 // We need to connect to the package first to force the service manager to |
128 // read the | 137 // read the |
129 // package app's manifest and register aliases for the applications it | 138 // package app's manifest and register aliases for the applications it |
130 // provides. | 139 // provides. |
131 test::mojom::ConnectTestServicePtr root_service; | 140 test::mojom::ConnectTestServicePtr root_service; |
132 std::unique_ptr<Connection> connection = | 141 connector()->BindInterface(kTestPackageName, &root_service); |
133 connector()->Connect(kTestPackageName); | |
134 connection->GetInterface(&root_service); | |
135 base::RunLoop run_loop; | 142 base::RunLoop run_loop; |
136 std::string root_name; | 143 std::string root_name; |
137 root_service->GetTitle( | 144 root_service->GetTitle( |
138 base::Bind(&ReceiveOneString, &root_name, &run_loop)); | 145 base::Bind(&ReceiveOneString, &root_name, &run_loop)); |
139 run_loop.Run(); | 146 run_loop.Run(); |
140 } | 147 } |
141 std::unique_ptr<Service> CreateService() override { | 148 std::unique_ptr<Service> CreateService() override { |
142 return base::MakeUnique<TestService>(this); | 149 return base::MakeUnique<TestService>(this); |
143 } | 150 } |
144 | 151 |
145 // InterfaceFactory<test::mojom::ExposedInterface>: | 152 // InterfaceFactory<test::mojom::ExposedInterface>: |
146 void Create(const Identity& remote_identity, | 153 void Create(const Identity& remote_identity, |
147 test::mojom::ExposedInterfaceRequest request) override { | 154 test::mojom::ExposedInterfaceRequest request) override { |
148 bindings_.AddBinding(this, std::move(request)); | 155 bindings_.AddBinding(this, std::move(request)); |
149 } | 156 } |
150 | 157 |
151 void ConnectionAccepted(test::mojom::ConnectionStatePtr state) override { | 158 void ConnectionAccepted(test::mojom::ConnectionStatePtr state) override { |
152 connection_state_ = std::move(state); | 159 connection_state_ = std::move(state); |
153 } | 160 } |
154 | 161 |
155 test::mojom::ConnectionStatePtr connection_state_; | 162 test::mojom::ConnectionStatePtr connection_state_; |
156 | 163 |
157 mojo::BindingSet<test::mojom::ExposedInterface> bindings_; | 164 mojo::BindingSet<test::mojom::ExposedInterface> bindings_; |
158 | 165 |
159 DISALLOW_COPY_AND_ASSIGN(ConnectTest); | 166 DISALLOW_COPY_AND_ASSIGN(ConnectTest); |
160 }; | 167 }; |
161 | 168 |
162 // Ensure the connection was properly established and that a round trip | 169 // Ensure the connection was properly established and that a round trip |
163 // method call/response is completed. | 170 // method call/response is completed. |
164 TEST_F(ConnectTest, Connect) { | 171 TEST_F(ConnectTest, BindInterface) { |
165 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppName); | |
166 test::mojom::ConnectTestServicePtr service; | 172 test::mojom::ConnectTestServicePtr service; |
167 connection->GetInterface(&service); | 173 connector()->BindInterface(kTestAppName, &service); |
168 base::RunLoop run_loop; | 174 base::RunLoop run_loop; |
169 std::string title; | 175 std::string title; |
170 service->GetTitle(base::Bind(&ReceiveOneString, &title, &run_loop)); | 176 service->GetTitle(base::Bind(&ReceiveOneString, &title, &run_loop)); |
171 run_loop.Run(); | 177 run_loop.Run(); |
172 EXPECT_EQ("APP", title); | 178 EXPECT_EQ("APP", title); |
173 EXPECT_FALSE(connection->IsPending()); | |
174 EXPECT_EQ(connection->GetRemoteIdentity().name(), kTestAppName); | |
175 } | 179 } |
176 | 180 |
177 TEST_F(ConnectTest, Instances) { | 181 TEST_F(ConnectTest, Instances) { |
178 Identity identity_a(kTestAppName, mojom::kInheritUserID, "A"); | 182 Identity identity_a(kTestAppName, mojom::kInheritUserID, "A"); |
179 std::unique_ptr<Connection> connection_a1 = ConnectTo(identity_a); | |
180 std::unique_ptr<Connection> connection_a2 = ConnectTo(identity_a); | |
181 std::string instance_a1, instance_a2; | 183 std::string instance_a1, instance_a2; |
182 test::mojom::ConnectTestServicePtr service_a1; | 184 test::mojom::ConnectTestServicePtr service_a1; |
183 { | 185 { |
184 connection_a1->GetInterface(&service_a1); | 186 connector()->BindInterface(identity_a, &service_a1); |
185 base::RunLoop loop; | 187 base::RunLoop loop; |
186 service_a1->GetInstance(base::Bind(&ReceiveOneString, &instance_a1, &loop)); | 188 service_a1->GetInstance(base::Bind(&ReceiveOneString, &instance_a1, &loop)); |
187 loop.Run(); | 189 loop.Run(); |
188 } | 190 } |
189 test::mojom::ConnectTestServicePtr service_a2; | 191 test::mojom::ConnectTestServicePtr service_a2; |
190 { | 192 { |
191 connection_a2->GetInterface(&service_a2); | 193 connector()->BindInterface(identity_a, &service_a2); |
192 base::RunLoop loop; | 194 base::RunLoop loop; |
193 service_a2->GetInstance(base::Bind(&ReceiveOneString, &instance_a2, &loop)); | 195 service_a2->GetInstance(base::Bind(&ReceiveOneString, &instance_a2, &loop)); |
194 loop.Run(); | 196 loop.Run(); |
195 } | 197 } |
196 EXPECT_EQ(instance_a1, instance_a2); | 198 EXPECT_EQ(instance_a1, instance_a2); |
197 | 199 |
198 Identity identity_b(kTestAppName, mojom::kInheritUserID, "B"); | 200 Identity identity_b(kTestAppName, mojom::kInheritUserID, "B"); |
199 std::unique_ptr<Connection> connection_b = ConnectTo(identity_b); | |
200 std::string instance_b; | 201 std::string instance_b; |
201 test::mojom::ConnectTestServicePtr service_b; | 202 test::mojom::ConnectTestServicePtr service_b; |
202 { | 203 { |
203 connection_b->GetInterface(&service_b); | 204 connector()->BindInterface(identity_b, &service_b); |
204 base::RunLoop loop; | 205 base::RunLoop loop; |
205 service_b->GetInstance(base::Bind(&ReceiveOneString, &instance_b, &loop)); | 206 service_b->GetInstance(base::Bind(&ReceiveOneString, &instance_b, &loop)); |
206 loop.Run(); | 207 loop.Run(); |
207 } | 208 } |
208 | 209 |
209 EXPECT_NE(instance_a1, instance_b); | 210 EXPECT_NE(instance_a1, instance_b); |
210 } | 211 } |
211 | 212 |
212 // BlockedInterface should not be exposed to this application because it is not | 213 // BlockedInterface should not be exposed to this application because it is not |
213 // in our CapabilityFilter whitelist. | 214 // in our CapabilityFilter whitelist. |
214 TEST_F(ConnectTest, BlockedInterface) { | 215 TEST_F(ConnectTest, BlockedInterface) { |
215 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppName); | |
216 base::RunLoop run_loop; | 216 base::RunLoop run_loop; |
217 test::mojom::BlockedInterfacePtr blocked; | 217 test::mojom::BlockedInterfacePtr blocked; |
218 connection->GetInterface(&blocked); | 218 connector()->BindInterface(kTestAppName, &blocked); |
219 blocked.set_connection_error_handler(base::Bind(&QuitLoop, &run_loop)); | 219 blocked.set_connection_error_handler(base::Bind(&QuitLoop, &run_loop)); |
220 std::string title = "unchanged"; | 220 std::string title = "unchanged"; |
221 blocked->GetTitleBlocked(base::Bind(&ReceiveOneString, &title, &run_loop)); | 221 blocked->GetTitleBlocked(base::Bind(&ReceiveOneString, &title, &run_loop)); |
222 run_loop.Run(); | 222 run_loop.Run(); |
223 EXPECT_EQ("unchanged", title); | 223 EXPECT_EQ("unchanged", title); |
224 } | 224 } |
225 | 225 |
226 // Connects to an app provided by a package. | 226 // Connects to an app provided by a package. |
227 TEST_F(ConnectTest, PackagedApp) { | 227 TEST_F(ConnectTest, PackagedApp) { |
228 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppAName); | |
229 test::mojom::ConnectTestServicePtr service_a; | 228 test::mojom::ConnectTestServicePtr service_a; |
230 connection->GetInterface(&service_a); | 229 connector()->BindInterface(kTestAppAName, &service_a); |
| 230 Connector::TestApi test_api(connector()); |
| 231 Identity resolved_identity; |
| 232 test_api.SetStartServiceCallback( |
| 233 base::Bind(&StartServiceResponse, nullptr, nullptr, &resolved_identity)); |
231 base::RunLoop run_loop; | 234 base::RunLoop run_loop; |
232 std::string a_name; | 235 std::string a_name; |
233 service_a->GetTitle(base::Bind(&ReceiveOneString, &a_name, &run_loop)); | 236 service_a->GetTitle(base::Bind(&ReceiveOneString, &a_name, &run_loop)); |
234 run_loop.Run(); | 237 run_loop.Run(); |
235 EXPECT_EQ("A", a_name); | 238 EXPECT_EQ("A", a_name); |
236 EXPECT_FALSE(connection->IsPending()); | 239 EXPECT_EQ(resolved_identity.name(), kTestAppAName); |
237 EXPECT_EQ(connection->GetRemoteIdentity().name(), kTestAppAName); | |
238 } | 240 } |
239 | 241 |
240 // Ask the target application to attempt to connect to a third application | 242 // Ask the target application to attempt to connect to a third application |
241 // provided by a package whose id is permitted by the primary target's | 243 // provided by a package whose id is permitted by the primary target's |
242 // CapabilityFilter but whose package is not. The connection should be | 244 // CapabilityFilter but whose package is not. The connection should be |
243 // allowed regardless of the target's CapabilityFilter with respect to the | 245 // allowed regardless of the target's CapabilityFilter with respect to the |
244 // package. | 246 // package. |
245 TEST_F(ConnectTest, BlockedPackage) { | 247 TEST_F(ConnectTest, BlockedPackage) { |
246 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppName); | |
247 test::mojom::StandaloneAppPtr standalone_app; | 248 test::mojom::StandaloneAppPtr standalone_app; |
248 connection->GetInterface(&standalone_app); | 249 connector()->BindInterface(kTestAppName, &standalone_app); |
249 base::RunLoop run_loop; | 250 base::RunLoop run_loop; |
250 std::string title; | 251 std::string title; |
251 standalone_app->ConnectToAllowedAppInBlockedPackage( | 252 standalone_app->ConnectToAllowedAppInBlockedPackage( |
252 base::Bind(&ReceiveOneString, &title, &run_loop)); | 253 base::Bind(&ReceiveOneString, &title, &run_loop)); |
253 run_loop.Run(); | 254 run_loop.Run(); |
254 EXPECT_EQ("A", title); | 255 EXPECT_EQ("A", title); |
255 } | 256 } |
256 | 257 |
257 // BlockedInterface should not be exposed to this application because it is not | 258 // BlockedInterface should not be exposed to this application because it is not |
258 // in our CapabilityFilter whitelist. | 259 // in our CapabilityFilter whitelist. |
259 TEST_F(ConnectTest, PackagedApp_BlockedInterface) { | 260 TEST_F(ConnectTest, PackagedApp_BlockedInterface) { |
260 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppAName); | |
261 base::RunLoop run_loop; | 261 base::RunLoop run_loop; |
262 test::mojom::BlockedInterfacePtr blocked; | 262 test::mojom::BlockedInterfacePtr blocked; |
263 connection->GetInterface(&blocked); | 263 connector()->BindInterface(kTestAppAName, &blocked); |
264 blocked.set_connection_error_handler(base::Bind(&QuitLoop, &run_loop)); | 264 blocked.set_connection_error_handler(base::Bind(&QuitLoop, &run_loop)); |
265 run_loop.Run(); | 265 run_loop.Run(); |
266 } | 266 } |
267 | 267 |
268 // Connection to another application provided by the same package, blocked | 268 // Connection to another application provided by the same package, blocked |
269 // because it's not in the capability filter whitelist. | 269 // because it's not in the capability filter whitelist. |
270 TEST_F(ConnectTest, BlockedPackagedApplication) { | 270 TEST_F(ConnectTest, BlockedPackagedApplication) { |
271 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppBName); | |
272 test::mojom::ConnectTestServicePtr service_b; | 271 test::mojom::ConnectTestServicePtr service_b; |
273 connection->GetInterface(&service_b); | 272 connector()->BindInterface(kTestAppBName, &service_b); |
| 273 Connector::TestApi test_api(connector()); |
| 274 mojom::ConnectResult result; |
| 275 test_api.SetStartServiceCallback( |
| 276 base::Bind(&StartServiceResponse, nullptr, &result, nullptr)); |
274 base::RunLoop run_loop; | 277 base::RunLoop run_loop; |
275 connection->SetConnectionLostClosure(base::Bind(&QuitLoop, &run_loop)); | 278 service_b.set_connection_error_handler(run_loop.QuitClosure()); |
276 run_loop.Run(); | 279 run_loop.Run(); |
277 EXPECT_FALSE(connection->IsPending()); | 280 EXPECT_EQ(mojom::ConnectResult::ACCESS_DENIED, result); |
278 EXPECT_EQ(mojom::ConnectResult::ACCESS_DENIED, connection->GetResult()); | |
279 } | 281 } |
280 | 282 |
281 TEST_F(ConnectTest, CapabilityClasses) { | 283 TEST_F(ConnectTest, CapabilityClasses) { |
282 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppName); | |
283 test::mojom::StandaloneAppPtr standalone_app; | 284 test::mojom::StandaloneAppPtr standalone_app; |
284 connection->GetInterface(&standalone_app); | 285 connector()->BindInterface(kTestAppName, &standalone_app); |
285 std::string string1, string2; | 286 std::string string1, string2; |
286 base::RunLoop loop; | 287 base::RunLoop loop; |
287 standalone_app->ConnectToClassInterface( | 288 standalone_app->ConnectToClassInterface( |
288 base::Bind(&ReceiveTwoStrings, &string1, &string2, &loop)); | 289 base::Bind(&ReceiveTwoStrings, &string1, &string2, &loop)); |
289 loop.Run(); | 290 loop.Run(); |
290 EXPECT_EQ("PONG", string1); | 291 EXPECT_EQ("PONG", string1); |
291 EXPECT_EQ("CLASS APP", string2); | 292 EXPECT_EQ("CLASS APP", string2); |
292 } | 293 } |
293 | 294 |
294 TEST_F(ConnectTest, ConnectWithoutExplicitClassBlocked) { | 295 TEST_F(ConnectTest, ConnectWithoutExplicitClassBlocked) { |
295 // We not be able to bind a ClassInterfacePtr since the connect_unittest app | 296 // We not be able to bind a ClassInterfacePtr since the connect_unittest app |
296 // does not explicitly request the "class" capability from | 297 // does not explicitly request the "class" capability from |
297 // connect_test_class_app. This test will hang if it is bound. | 298 // connect_test_class_app. This test will hang if it is bound. |
298 std::unique_ptr<Connection> connection = | |
299 connector()->Connect(kTestClassAppName); | |
300 test::mojom::ClassInterfacePtr class_interface; | 299 test::mojom::ClassInterfacePtr class_interface; |
301 connection->GetInterface(&class_interface); | 300 connector()->BindInterface(kTestClassAppName, &class_interface); |
302 base::RunLoop loop; | 301 base::RunLoop loop; |
303 class_interface.set_connection_error_handler(base::Bind(&QuitLoop, &loop)); | 302 class_interface.set_connection_error_handler(base::Bind(&QuitLoop, &loop)); |
304 loop.Run(); | 303 loop.Run(); |
305 } | 304 } |
306 | 305 |
307 TEST_F(ConnectTest, ConnectAsDifferentUser_Allowed) { | 306 TEST_F(ConnectTest, ConnectAsDifferentUser_Allowed) { |
308 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppName); | |
309 test::mojom::UserIdTestPtr user_id_test; | 307 test::mojom::UserIdTestPtr user_id_test; |
310 connection->GetInterface(&user_id_test); | 308 connector()->BindInterface(kTestAppName, &user_id_test); |
311 mojom::ConnectResult result; | 309 mojom::ConnectResult result; |
312 Identity target(kTestClassAppName, base::GenerateGUID()); | 310 Identity target(kTestClassAppName, base::GenerateGUID()); |
313 Identity result_identity; | 311 Identity result_identity; |
314 { | 312 { |
315 base::RunLoop loop; | 313 base::RunLoop loop; |
316 user_id_test->ConnectToClassAppAsDifferentUser( | 314 user_id_test->ConnectToClassAppAsDifferentUser( |
317 target, | 315 target, |
318 base::Bind(&ReceiveConnectionResult, &result, &result_identity, &loop)); | 316 base::Bind(&ReceiveConnectionResult, &result, &result_identity, &loop)); |
319 loop.Run(); | 317 loop.Run(); |
320 } | 318 } |
321 EXPECT_EQ(result, mojom::ConnectResult::SUCCEEDED); | 319 EXPECT_EQ(result, mojom::ConnectResult::SUCCEEDED); |
322 EXPECT_EQ(target, result_identity); | 320 EXPECT_EQ(target, result_identity); |
323 } | 321 } |
324 | 322 |
325 TEST_F(ConnectTest, ConnectAsDifferentUser_Blocked) { | 323 TEST_F(ConnectTest, ConnectAsDifferentUser_Blocked) { |
326 std::unique_ptr<Connection> connection = connector()->Connect(kTestAppAName); | |
327 test::mojom::UserIdTestPtr user_id_test; | 324 test::mojom::UserIdTestPtr user_id_test; |
328 connection->GetInterface(&user_id_test); | 325 connector()->BindInterface(kTestAppAName, &user_id_test); |
329 mojom::ConnectResult result; | 326 mojom::ConnectResult result; |
330 Identity target(kTestClassAppName, base::GenerateGUID()); | 327 Identity target(kTestClassAppName, base::GenerateGUID()); |
331 Identity result_identity; | 328 Identity result_identity; |
332 { | 329 { |
333 base::RunLoop loop; | 330 base::RunLoop loop; |
334 user_id_test->ConnectToClassAppAsDifferentUser( | 331 user_id_test->ConnectToClassAppAsDifferentUser( |
335 target, | 332 target, |
336 base::Bind(&ReceiveConnectionResult, &result, &result_identity, &loop)); | 333 base::Bind(&ReceiveConnectionResult, &result, &result_identity, &loop)); |
337 loop.Run(); | 334 loop.Run(); |
338 } | 335 } |
339 EXPECT_EQ(mojom::ConnectResult::ACCESS_DENIED, result); | 336 EXPECT_EQ(mojom::ConnectResult::ACCESS_DENIED, result); |
340 EXPECT_FALSE(target == result_identity); | 337 EXPECT_FALSE(target == result_identity); |
341 } | 338 } |
342 | 339 |
343 // There are various other tests (service manager, lifecycle) that test valid | 340 // There are various other tests (service manager, lifecycle) that test valid |
344 // client | 341 // client |
345 // process specifications. This is the only one for blocking. | 342 // process specifications. This is the only one for blocking. |
346 TEST_F(ConnectTest, ConnectToClientProcess_Blocked) { | 343 TEST_F(ConnectTest, ConnectToClientProcess_Blocked) { |
347 base::Process process; | 344 base::Process process; |
348 std::unique_ptr<service_manager::Connection> connection = | 345 mojom::ConnectResult result = |
349 service_manager::test::LaunchAndConnectToProcess( | 346 service_manager::test::LaunchAndConnectToProcess( |
350 #if defined(OS_WIN) | 347 #if defined(OS_WIN) |
351 "connect_test_exe.exe", | 348 "connect_test_exe.exe", |
352 #else | 349 #else |
353 "connect_test_exe", | 350 "connect_test_exe", |
354 #endif | 351 #endif |
355 service_manager::Identity("connect_test_exe", | 352 service_manager::Identity("connect_test_exe", |
356 service_manager::mojom::kInheritUserID), | 353 service_manager::mojom::kInheritUserID), |
357 connector(), &process); | 354 connector(), &process); |
358 EXPECT_EQ(connection->GetResult(), mojom::ConnectResult::ACCESS_DENIED); | 355 EXPECT_EQ(result, mojom::ConnectResult::ACCESS_DENIED); |
359 } | 356 } |
360 | 357 |
361 // Verifies that a client with the "all_users" capability class can receive | 358 // Verifies that a client with the "all_users" capability class can receive |
362 // connections from clients run as other users. | 359 // connections from clients run as other users. |
363 TEST_F(ConnectTest, AllUsersSingleton) { | 360 TEST_F(ConnectTest, AllUsersSingleton) { |
364 // Connect to an instance with an explicitly different user_id. This supplied | 361 // Connect to an instance with an explicitly different user_id. This supplied |
365 // user id should be ignored by the service manager (which will generate its | 362 // user id should be ignored by the service manager (which will generate its |
366 // own | 363 // own |
367 // synthetic user id for all-user singleton instances). | 364 // synthetic user id for all-user singleton instances). |
368 const std::string singleton_userid = base::GenerateGUID(); | 365 const std::string singleton_userid = base::GenerateGUID(); |
369 Identity singleton_id(kTestSingletonAppName, singleton_userid); | 366 Identity singleton_id(kTestSingletonAppName, singleton_userid); |
370 std::unique_ptr<Connection> connection = connector()->Connect(singleton_id); | 367 connector()->StartService(singleton_id); |
| 368 Identity first_resolved_identity; |
371 { | 369 { |
372 base::RunLoop loop; | 370 base::RunLoop loop; |
373 connection->AddConnectionCompletedClosure(base::Bind(&QuitLoop, &loop)); | 371 Connector::TestApi test_api(connector()); |
| 372 test_api.SetStartServiceCallback(base::Bind( |
| 373 &StartServiceResponse, &loop, nullptr, &first_resolved_identity)); |
374 loop.Run(); | 374 loop.Run(); |
375 EXPECT_NE(connection->GetRemoteIdentity().user_id(), singleton_userid); | 375 EXPECT_NE(first_resolved_identity.user_id(), singleton_userid); |
376 } | 376 } |
377 // This connects using the current client's user_id. It should be bound to the | 377 // This connects using the current client's user_id. It should be bound to the |
378 // same service started above, with the same service manager-generated user | 378 // same service started above, with the same service manager-generated user |
379 // id. | 379 // id. |
380 std::unique_ptr<Connection> inherit_connection = | 380 connector()->StartService(kTestSingletonAppName); |
381 connector()->Connect(kTestSingletonAppName); | |
382 { | 381 { |
383 base::RunLoop loop; | 382 base::RunLoop loop; |
384 inherit_connection->AddConnectionCompletedClosure( | 383 Connector::TestApi test_api(connector()); |
385 base::Bind(&QuitLoop, &loop)); | 384 Identity resolved_identity; |
| 385 test_api.SetStartServiceCallback( |
| 386 base::Bind(&StartServiceResponse, &loop, nullptr, &resolved_identity)); |
386 loop.Run(); | 387 loop.Run(); |
387 EXPECT_EQ(inherit_connection->GetRemoteIdentity().user_id(), | 388 EXPECT_EQ(resolved_identity.user_id(), first_resolved_identity.user_id()); |
388 connection->GetRemoteIdentity().user_id()); | |
389 } | 389 } |
390 } | 390 } |
391 | 391 |
392 } // namespace service_manager | 392 } // namespace service_manager |
OLD | NEW |