1 // Copyright (c) 2012 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 <string>
6
7 #include "base/bind.h"
8 #include "base/callback.h"
9 #include "base/compiler_specific.h"
10 #include "base/files/file_path.h"
11 #include "base/memory/scoped_ptr.h"
12 #include "base/message_loop/message_loop.h"
13 #include "base/run_loop.h"
14 #include "base/synchronization/lock.h"
15 #include "base/threading/thread.h"
16 #include "base/values.h"
17 #include "chrome/test/chromedriver/chrome/status.h"
18 #include "chrome/test/chromedriver/chrome/stub_chrome.h"
19 #include "chrome/test/chromedriver/chrome/stub_web_view.h"
20 #include "chrome/test/chromedriver/chrome/web_view.h"
21 #include "chrome/test/chromedriver/commands.h"
22 #include "chrome/test/chromedriver/element_commands.h"
23 #include "chrome/test/chromedriver/session.h"
24 #include "chrome/test/chromedriver/session_commands.h"
25 #include "chrome/test/chromedriver/window_commands.h"
26 #include "testing/gtest/include/gtest/gtest.h"
27 #include "third_party/webdriver/atoms.h"
28
29 namespace {
30
OnGetStatus(const Status & status,scoped_ptr<base::Value> value,const std::string & session_id)31 void OnGetStatus(const Status& status,
32 scoped_ptr<base::Value> value,
33 const std::string& session_id) {
34 ASSERT_EQ(kOk, status.code());
35 base::DictionaryValue* dict;
36 ASSERT_TRUE(value->GetAsDictionary(&dict));
37 base::Value* unused;
38 ASSERT_TRUE(dict->Get("os.name", &unused));
39 ASSERT_TRUE(dict->Get("os.version", &unused));
40 ASSERT_TRUE(dict->Get("os.arch", &unused));
41 ASSERT_TRUE(dict->Get("build.version", &unused));
42 }
43
44 } // namespace
45
TEST(CommandsTest,GetStatus)46 TEST(CommandsTest, GetStatus) {
47 base::DictionaryValue params;
48 ExecuteGetStatus(params, std::string(), base::Bind(&OnGetStatus));
49 }
50
51 namespace {
52
ExecuteStubQuit(int * count,const base::DictionaryValue & params,const std::string & session_id,const CommandCallback & callback)53 void ExecuteStubQuit(
54 int* count,
55 const base::DictionaryValue& params,
56 const std::string& session_id,
57 const CommandCallback& callback) {
58 if (*count == 0) {
59 EXPECT_STREQ("id", session_id.c_str());
60 } else {
61 EXPECT_STREQ("id2", session_id.c_str());
62 }
63 (*count)++;
64 callback.Run(Status(kOk), scoped_ptr<base::Value>(), session_id);
65 }
66
OnQuitAll(const Status & status,scoped_ptr<base::Value> value,const std::string & session_id)67 void OnQuitAll(const Status& status,
68 scoped_ptr<base::Value> value,
69 const std::string& session_id) {
70 ASSERT_EQ(kOk, status.code());
71 ASSERT_FALSE(value.get());
72 }
73
74 } // namespace
75
TEST(CommandsTest,QuitAll)76 TEST(CommandsTest, QuitAll) {
77 SessionThreadMap map;
78 Session session("id");
79 Session session2("id2");
80 map[session.id] = make_linked_ptr(new base::Thread("1"));
81 map[session2.id] = make_linked_ptr(new base::Thread("2"));
82
83 int count = 0;
84 Command cmd = base::Bind(&ExecuteStubQuit, &count);
85 base::DictionaryValue params;
86 base::MessageLoop loop;
87 ExecuteQuitAll(cmd, &map, params, std::string(), base::Bind(&OnQuitAll));
88 ASSERT_EQ(2, count);
89 }
90
91 namespace {
92
ExecuteSimpleCommand(const std::string & expected_id,base::DictionaryValue * expected_params,base::Value * value,Session * session,const base::DictionaryValue & params,scoped_ptr<base::Value> * return_value)93 Status ExecuteSimpleCommand(
94 const std::string& expected_id,
95 base::DictionaryValue* expected_params,
96 base::Value* value,
97 Session* session,
98 const base::DictionaryValue& params,
99 scoped_ptr<base::Value>* return_value) {
100 EXPECT_EQ(expected_id, session->id);
101 EXPECT_TRUE(expected_params->Equals(¶ms));
102 return_value->reset(value->DeepCopy());
103 session->quit = true;
104 return Status(kOk);
105 }
106
OnSimpleCommand(base::RunLoop * run_loop,const std::string & expected_session_id,base::Value * expected_value,const Status & status,scoped_ptr<base::Value> value,const std::string & session_id)107 void OnSimpleCommand(base::RunLoop* run_loop,
108 const std::string& expected_session_id,
109 base::Value* expected_value,
110 const Status& status,
111 scoped_ptr<base::Value> value,
112 const std::string& session_id) {
113 ASSERT_EQ(kOk, status.code());
114 ASSERT_TRUE(expected_value->Equals(value.get()));
115 ASSERT_EQ(expected_session_id, session_id);
116 run_loop->Quit();
117 }
118
119 } // namespace
120
TEST(CommandsTest,ExecuteSessionCommand)121 TEST(CommandsTest, ExecuteSessionCommand) {
122 SessionThreadMap map;
123 linked_ptr<base::Thread> thread(new base::Thread("1"));
124 ASSERT_TRUE(thread->Start());
125 std::string id("id");
126 thread->message_loop()->PostTask(
127 FROM_HERE,
128 base::Bind(&internal::CreateSessionOnSessionThreadForTesting, id));
129 map[id] = thread;
130
131 base::DictionaryValue params;
132 params.SetInteger("param", 5);
133 base::FundamentalValue expected_value(6);
134 SessionCommand cmd = base::Bind(
135 &ExecuteSimpleCommand, id, ¶ms, &expected_value);
136
137 base::MessageLoop loop;
138 base::RunLoop run_loop;
139 ExecuteSessionCommand(
140 &map,
141 "cmd",
142 cmd,
143 false,
144 params,
145 id,
146 base::Bind(&OnSimpleCommand, &run_loop, id, &expected_value));
147 run_loop.Run();
148 }
149
150 namespace {
151
ShouldNotBeCalled(Session * session,const base::DictionaryValue & params,scoped_ptr<base::Value> * value)152 Status ShouldNotBeCalled(
153 Session* session,
154 const base::DictionaryValue& params,
155 scoped_ptr<base::Value>* value) {
156 EXPECT_TRUE(false);
157 return Status(kOk);
158 }
159
OnNoSuchSession(const Status & status,scoped_ptr<base::Value> value,const std::string & session_id)160 void OnNoSuchSession(const Status& status,
161 scoped_ptr<base::Value> value,
162 const std::string& session_id) {
163 EXPECT_EQ(kNoSuchSession, status.code());
164 EXPECT_FALSE(value.get());
165 }
166
OnNoSuchSessionIsOk(const Status & status,scoped_ptr<base::Value> value,const std::string & session_id)167 void OnNoSuchSessionIsOk(const Status& status,
168 scoped_ptr<base::Value> value,
169 const std::string& session_id) {
170 EXPECT_EQ(kOk, status.code());
171 EXPECT_FALSE(value.get());
172 }
173
174 } // namespace
175
TEST(CommandsTest,ExecuteSessionCommandOnNoSuchSession)176 TEST(CommandsTest, ExecuteSessionCommandOnNoSuchSession) {
177 SessionThreadMap map;
178 base::DictionaryValue params;
179 ExecuteSessionCommand(&map,
180 "cmd",
181 base::Bind(&ShouldNotBeCalled),
182 false,
183 params,
184 "session",
185 base::Bind(&OnNoSuchSession));
186 }
187
TEST(CommandsTest,ExecuteSessionCommandOnNoSuchSessionWhenItExpectsOk)188 TEST(CommandsTest, ExecuteSessionCommandOnNoSuchSessionWhenItExpectsOk) {
189 SessionThreadMap map;
190 base::DictionaryValue params;
191 ExecuteSessionCommand(&map,
192 "cmd",
193 base::Bind(&ShouldNotBeCalled),
194 true,
195 params,
196 "session",
197 base::Bind(&OnNoSuchSessionIsOk));
198 }
199
200 namespace {
201
OnNoSuchSessionAndQuit(base::RunLoop * run_loop,const Status & status,scoped_ptr<base::Value> value,const std::string & session_id)202 void OnNoSuchSessionAndQuit(base::RunLoop* run_loop,
203 const Status& status,
204 scoped_ptr<base::Value> value,
205 const std::string& session_id) {
206 run_loop->Quit();
207 EXPECT_EQ(kNoSuchSession, status.code());
208 EXPECT_FALSE(value.get());
209 }
210
211 } // namespace
212
TEST(CommandsTest,ExecuteSessionCommandOnJustDeletedSession)213 TEST(CommandsTest, ExecuteSessionCommandOnJustDeletedSession) {
214 SessionThreadMap map;
215 linked_ptr<base::Thread> thread(new base::Thread("1"));
216 ASSERT_TRUE(thread->Start());
217 std::string id("id");
218 map[id] = thread;
219
220 base::MessageLoop loop;
221 base::RunLoop run_loop;
222 ExecuteSessionCommand(&map,
223 "cmd",
224 base::Bind(&ShouldNotBeCalled),
225 false,
226 base::DictionaryValue(),
227 "session",
228 base::Bind(&OnNoSuchSessionAndQuit, &run_loop));
229 run_loop.Run();
230 }
231
232 namespace {
233
234 enum TestScenario {
235 kElementExistsQueryOnce = 0,
236 kElementExistsQueryTwice,
237 kElementNotExistsQueryOnce,
238 kElementExistsTimeout
239 };
240
241 class FindElementWebView : public StubWebView {
242 public:
FindElementWebView(bool only_one,TestScenario scenario)243 FindElementWebView(bool only_one, TestScenario scenario)
244 : StubWebView("1"), only_one_(only_one), scenario_(scenario),
245 current_count_(0) {
246 switch (scenario_) {
247 case kElementExistsQueryOnce:
248 case kElementExistsQueryTwice:
249 case kElementExistsTimeout: {
250 if (only_one_) {
251 base::DictionaryValue element;
252 element.SetString("ELEMENT", "1");
253 result_.reset(element.DeepCopy());
254 } else {
255 base::DictionaryValue element1;
256 element1.SetString("ELEMENT", "1");
257 base::DictionaryValue element2;
258 element2.SetString("ELEMENT", "2");
259 base::ListValue list;
260 list.Append(element1.DeepCopy());
261 list.Append(element2.DeepCopy());
262 result_.reset(list.DeepCopy());
263 }
264 break;
265 }
266 case kElementNotExistsQueryOnce: {
267 if (only_one_)
268 result_.reset(base::Value::CreateNullValue());
269 else
270 result_.reset(new base::ListValue());
271 break;
272 }
273 }
274 }
~FindElementWebView()275 virtual ~FindElementWebView() {}
276
Verify(const std::string & expected_frame,const base::ListValue * expected_args,const base::Value * actrual_result)277 void Verify(const std::string& expected_frame,
278 const base::ListValue* expected_args,
279 const base::Value* actrual_result) {
280 EXPECT_EQ(expected_frame, frame_);
281 std::string function;
282 if (only_one_)
283 function = webdriver::atoms::asString(webdriver::atoms::FIND_ELEMENT);
284 else
285 function = webdriver::atoms::asString(webdriver::atoms::FIND_ELEMENTS);
286 EXPECT_EQ(function, function_);
287 ASSERT_TRUE(args_.get());
288 EXPECT_TRUE(expected_args->Equals(args_.get()));
289 ASSERT_TRUE(actrual_result);
290 EXPECT_TRUE(result_->Equals(actrual_result));
291 }
292
293 // Overridden from WebView:
CallFunction(const std::string & frame,const std::string & function,const base::ListValue & args,scoped_ptr<base::Value> * result)294 virtual Status CallFunction(const std::string& frame,
295 const std::string& function,
296 const base::ListValue& args,
297 scoped_ptr<base::Value>* result) OVERRIDE {
298 ++current_count_;
299 if (scenario_ == kElementExistsTimeout ||
300 (scenario_ == kElementExistsQueryTwice && current_count_ == 1)) {
301 // Always return empty result when testing timeout.
302 if (only_one_)
303 result->reset(base::Value::CreateNullValue());
304 else
305 result->reset(new base::ListValue());
306 } else {
307 switch (scenario_) {
308 case kElementExistsQueryOnce:
309 case kElementNotExistsQueryOnce: {
310 EXPECT_EQ(1, current_count_);
311 break;
312 }
313 case kElementExistsQueryTwice: {
314 EXPECT_EQ(2, current_count_);
315 break;
316 }
317 default: {
318 break;
319 }
320 }
321
322 result->reset(result_->DeepCopy());
323 frame_ = frame;
324 function_ = function;
325 args_.reset(args.DeepCopy());
326 }
327 return Status(kOk);
328 }
329
330 private:
331 bool only_one_;
332 TestScenario scenario_;
333 int current_count_;
334 std::string frame_;
335 std::string function_;
336 scoped_ptr<base::ListValue> args_;
337 scoped_ptr<base::Value> result_;
338 };
339
340 } // namespace
341
TEST(CommandsTest,SuccessfulFindElement)342 TEST(CommandsTest, SuccessfulFindElement) {
343 FindElementWebView web_view(true, kElementExistsQueryTwice);
344 Session session("id");
345 session.implicit_wait = base::TimeDelta::FromSeconds(1);
346 session.SwitchToSubFrame("frame_id1", std::string());
347 base::DictionaryValue params;
348 params.SetString("using", "id");
349 params.SetString("value", "a");
350 scoped_ptr<base::Value> result;
351 ASSERT_EQ(kOk,
352 ExecuteFindElement(1, &session, &web_view, params, &result).code());
353 base::DictionaryValue param;
354 param.SetString("id", "a");
355 base::ListValue expected_args;
356 expected_args.Append(param.DeepCopy());
357 web_view.Verify("frame_id1", &expected_args, result.get());
358 }
359
TEST(CommandsTest,FailedFindElement)360 TEST(CommandsTest, FailedFindElement) {
361 FindElementWebView web_view(true, kElementNotExistsQueryOnce);
362 Session session("id");
363 base::DictionaryValue params;
364 params.SetString("using", "id");
365 params.SetString("value", "a");
366 scoped_ptr<base::Value> result;
367 ASSERT_EQ(kNoSuchElement,
368 ExecuteFindElement(1, &session, &web_view, params, &result).code());
369 }
370
TEST(CommandsTest,SuccessfulFindElements)371 TEST(CommandsTest, SuccessfulFindElements) {
372 FindElementWebView web_view(false, kElementExistsQueryTwice);
373 Session session("id");
374 session.implicit_wait = base::TimeDelta::FromSeconds(1);
375 session.SwitchToSubFrame("frame_id2", std::string());
376 base::DictionaryValue params;
377 params.SetString("using", "name");
378 params.SetString("value", "b");
379 scoped_ptr<base::Value> result;
380 ASSERT_EQ(
381 kOk,
382 ExecuteFindElements(1, &session, &web_view, params, &result).code());
383 base::DictionaryValue param;
384 param.SetString("name", "b");
385 base::ListValue expected_args;
386 expected_args.Append(param.DeepCopy());
387 web_view.Verify("frame_id2", &expected_args, result.get());
388 }
389
TEST(CommandsTest,FailedFindElements)390 TEST(CommandsTest, FailedFindElements) {
391 Session session("id");
392 FindElementWebView web_view(false, kElementNotExistsQueryOnce);
393 base::DictionaryValue params;
394 params.SetString("using", "id");
395 params.SetString("value", "a");
396 scoped_ptr<base::Value> result;
397 ASSERT_EQ(
398 kOk,
399 ExecuteFindElements(1, &session, &web_view, params, &result).code());
400 base::ListValue* list;
401 ASSERT_TRUE(result->GetAsList(&list));
402 ASSERT_EQ(0U, list->GetSize());
403 }
404
TEST(CommandsTest,SuccessfulFindChildElement)405 TEST(CommandsTest, SuccessfulFindChildElement) {
406 FindElementWebView web_view(true, kElementExistsQueryTwice);
407 Session session("id");
408 session.implicit_wait = base::TimeDelta::FromSeconds(1);
409 session.SwitchToSubFrame("frame_id3", std::string());
410 base::DictionaryValue params;
411 params.SetString("using", "tag name");
412 params.SetString("value", "div");
413 std::string element_id = "1";
414 scoped_ptr<base::Value> result;
415 ASSERT_EQ(
416 kOk,
417 ExecuteFindChildElement(
418 1, &session, &web_view, element_id, params, &result).code());
419 base::DictionaryValue locator_param;
420 locator_param.SetString("tag name", "div");
421 base::DictionaryValue root_element_param;
422 root_element_param.SetString("ELEMENT", element_id);
423 base::ListValue expected_args;
424 expected_args.Append(locator_param.DeepCopy());
425 expected_args.Append(root_element_param.DeepCopy());
426 web_view.Verify("frame_id3", &expected_args, result.get());
427 }
428
TEST(CommandsTest,FailedFindChildElement)429 TEST(CommandsTest, FailedFindChildElement) {
430 Session session("id");
431 FindElementWebView web_view(true, kElementNotExistsQueryOnce);
432 base::DictionaryValue params;
433 params.SetString("using", "id");
434 params.SetString("value", "a");
435 std::string element_id = "1";
436 scoped_ptr<base::Value> result;
437 ASSERT_EQ(
438 kNoSuchElement,
439 ExecuteFindChildElement(
440 1, &session, &web_view, element_id, params, &result).code());
441 }
442
TEST(CommandsTest,SuccessfulFindChildElements)443 TEST(CommandsTest, SuccessfulFindChildElements) {
444 FindElementWebView web_view(false, kElementExistsQueryTwice);
445 Session session("id");
446 session.implicit_wait = base::TimeDelta::FromSeconds(1);
447 session.SwitchToSubFrame("frame_id4", std::string());
448 base::DictionaryValue params;
449 params.SetString("using", "class name");
450 params.SetString("value", "c");
451 std::string element_id = "1";
452 scoped_ptr<base::Value> result;
453 ASSERT_EQ(
454 kOk,
455 ExecuteFindChildElements(
456 1, &session, &web_view, element_id, params, &result).code());
457 base::DictionaryValue locator_param;
458 locator_param.SetString("class name", "c");
459 base::DictionaryValue root_element_param;
460 root_element_param.SetString("ELEMENT", element_id);
461 base::ListValue expected_args;
462 expected_args.Append(locator_param.DeepCopy());
463 expected_args.Append(root_element_param.DeepCopy());
464 web_view.Verify("frame_id4", &expected_args, result.get());
465 }
466
TEST(CommandsTest,FailedFindChildElements)467 TEST(CommandsTest, FailedFindChildElements) {
468 Session session("id");
469 FindElementWebView web_view(false, kElementNotExistsQueryOnce);
470 base::DictionaryValue params;
471 params.SetString("using", "id");
472 params.SetString("value", "a");
473 std::string element_id = "1";
474 scoped_ptr<base::Value> result;
475 ASSERT_EQ(
476 kOk,
477 ExecuteFindChildElements(
478 1, &session, &web_view, element_id, params, &result).code());
479 base::ListValue* list;
480 ASSERT_TRUE(result->GetAsList(&list));
481 ASSERT_EQ(0U, list->GetSize());
482 }
483
TEST(CommandsTest,TimeoutInFindElement)484 TEST(CommandsTest, TimeoutInFindElement) {
485 Session session("id");
486 FindElementWebView web_view(true, kElementExistsTimeout);
487 session.implicit_wait = base::TimeDelta::FromMilliseconds(2);
488 base::DictionaryValue params;
489 params.SetString("using", "id");
490 params.SetString("value", "a");
491 params.SetString("id", "1");
492 scoped_ptr<base::Value> result;
493 ASSERT_EQ(kNoSuchElement,
494 ExecuteFindElement(1, &session, &web_view, params, &result).code());
495 }
496
497 namespace {
498
499 class ErrorCallFunctionWebView : public StubWebView {
500 public:
ErrorCallFunctionWebView(StatusCode code)501 explicit ErrorCallFunctionWebView(StatusCode code)
502 : StubWebView("1"), code_(code) {}
~ErrorCallFunctionWebView()503 virtual ~ErrorCallFunctionWebView() {}
504
505 // Overridden from WebView:
CallFunction(const std::string & frame,const std::string & function,const base::ListValue & args,scoped_ptr<base::Value> * result)506 virtual Status CallFunction(const std::string& frame,
507 const std::string& function,
508 const base::ListValue& args,
509 scoped_ptr<base::Value>* result) OVERRIDE {
510 return Status(code_);
511 }
512
513 private:
514 StatusCode code_;
515 };
516
517 } // namespace
518
TEST(CommandsTest,ErrorFindElement)519 TEST(CommandsTest, ErrorFindElement) {
520 Session session("id");
521 ErrorCallFunctionWebView web_view(kUnknownError);
522 base::DictionaryValue params;
523 params.SetString("using", "id");
524 params.SetString("value", "a");
525 scoped_ptr<base::Value> value;
526 ASSERT_EQ(kUnknownError,
527 ExecuteFindElement(1, &session, &web_view, params, &value).code());
528 ASSERT_EQ(kUnknownError,
529 ExecuteFindElements(1, &session, &web_view, params, &value).code());
530 }
531
TEST(CommandsTest,ErrorFindChildElement)532 TEST(CommandsTest, ErrorFindChildElement) {
533 Session session("id");
534 ErrorCallFunctionWebView web_view(kStaleElementReference);
535 base::DictionaryValue params;
536 params.SetString("using", "id");
537 params.SetString("value", "a");
538 std::string element_id = "1";
539 scoped_ptr<base::Value> result;
540 ASSERT_EQ(
541 kStaleElementReference,
542 ExecuteFindChildElement(
543 1, &session, &web_view, element_id, params, &result).code());
544 ASSERT_EQ(
545 kStaleElementReference,
546 ExecuteFindChildElements(
547 1, &session, &web_view, element_id, params, &result).code());
548 }
549