01-driver.t 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. use strict;
  2. use warnings;
  3. use JSON;
  4. use Test::More;
  5. use LWP::UserAgent;
  6. use Test::LWP::UserAgent;
  7. use IO::Socket::INET;
  8. use Selenium::Remote::Driver;
  9. use Selenium::Remote::Mock::Commands;
  10. use Selenium::Remote::Mock::RemoteConnection;
  11. use FindBin;
  12. use lib $FindBin::Bin . '/lib';
  13. use TestHarness;
  14. my %selenium_args = %{ TestHarness->new(
  15. this_file => $FindBin::Script
  16. )->base_caps };
  17. my $driver = Selenium::Remote::Driver->new(%selenium_args);
  18. my $website = 'http://localhost:63636';
  19. my $ret;
  20. DESIRED_CAPABILITIES: {
  21. # We're using a different test method for these because we needed
  22. # to inspect payload of the POST to /session, and the method of
  23. # recording the RES/REQ pairs doesn't provide any easy way to do
  24. # that.
  25. my $tua = Test::LWP::UserAgent->new;
  26. my $res = {
  27. cmd_return => {},
  28. cmd_status => 'OK',
  29. sessionId => '123124123'
  30. };
  31. $tua->map_response(qr{status}, HTTP::Response->new(200, 'OK'));
  32. my $requests_count = 0;
  33. my $mock_session_handler = sub {
  34. my $request = shift;
  35. $requests_count++;
  36. if ($request->method eq 'POST') {
  37. my $caps = from_json($request->decoded_content)->{desiredCapabilities};
  38. my @keys = keys %$caps;
  39. if (scalar @keys) {
  40. ok(scalar @keys == 2, 'exactly 2 keys passed in if we use desired_capabilities');
  41. my $grep = grep { 'browserName' eq $_ } @keys;
  42. ok($grep, 'and it is the right one');
  43. ok($caps->{superfluous} eq 'thing', 'and we pass through anything else');
  44. ok($caps->{browserName} eq 'firefox', 'and we override the "normal" caps');
  45. ok(!exists $caps->{platform}, 'or ignore them entirely');
  46. }
  47. else {
  48. ok(to_json($caps) eq '{}', 'an empty constructor defaults to an empty hash');
  49. }
  50. return HTTP::Response->new(204, 'OK', ['Content-Type' => 'application/json'], to_json($res));
  51. }
  52. else {
  53. # it's the DELETE when the driver calls
  54. # DESTROY. This should be the last call to /session/.
  55. return HTTP::Response->new(200, 'OK')
  56. }
  57. };
  58. $tua->map_response(qr{session}, $mock_session_handler);
  59. my $caps_driver = Selenium::Remote::Driver->new_from_caps(
  60. auto_close => 0,
  61. browser_name => 'not firefox',
  62. platform => 'WINDOWS',
  63. desired_capabilities => {
  64. 'browserName' => 'firefox',
  65. 'superfluous' => 'thing'
  66. },
  67. ua => $tua
  68. );
  69. ok($caps_driver->auto_close eq 0, 'and other properties are still set');
  70. $caps_driver = Selenium::Remote::Driver->new(
  71. auto_close => 0,
  72. browser_name => 'not firefox',
  73. platform => 'WINDOWS',
  74. desired_capabilities => {
  75. 'browserName' => 'firefox',
  76. 'superfluous' => 'thing'
  77. },
  78. ua => $tua
  79. );
  80. ok($caps_driver->auto_close eq 0, 'and other properties are set if we use the normal constructor');
  81. $caps_driver = Selenium::Remote::Driver->new_from_caps(ua => $tua);
  82. ok($requests_count == 3, 'The new_from_caps section has the correct number of requests to /session/');
  83. }
  84. GRID_STARTUP: {
  85. # Mimicking a grid server; /wd/hub/status fails, and we expect
  86. # grid/api/hub/status to be checked instead.
  87. my $tua = Test::LWP::UserAgent->new;
  88. my $not_ok = sub {
  89. return HTTP::Response->new(500, 'NOTOK');
  90. };
  91. $tua->map_response(qr{wd/hub/status}, $not_ok);
  92. my $grid_status_count = 0;
  93. my $ok = sub {
  94. my $res = {
  95. cmd_return => {},
  96. cmd_status => 'OK',
  97. sessionId => '123124123'
  98. };
  99. $grid_status_count++;
  100. return HTTP::Response->new(200, 'OK', ['Content-Type' => 'application/json'], to_json($res));
  101. };
  102. $tua->map_response(qr{(?:grid/api/hub/status|session)}, $ok);
  103. my $grid_driver = Selenium::Remote::Driver->new(ua => $tua);
  104. ok(defined $grid_driver, 'Grid: Object loaded fine using grid/api/hub/status');
  105. ok($grid_driver->isa('Selenium::Remote::Driver'), 'Grid: ...and of right type');
  106. ok($grid_status_count == 2, 'checked Grid specific status');
  107. }
  108. CHECK_DRIVER: {
  109. ok(defined $driver, 'Object loaded fine...');
  110. ok($driver->isa('Selenium::Remote::Driver'), '...and of right type');
  111. ok(defined $driver->{'session_id'}, 'Established session on remote server');
  112. $ret = $driver->get_capabilities;
  113. is($ret->{'browserName'}, 'firefox', 'Right capabilities');
  114. my $status = $driver->status;
  115. ok($status->{build}->{version},"Got status build.version");
  116. ok($status->{build}->{revision},"Got status build.revision");
  117. ok($status->{build}->{time},"Got status build.time");
  118. }
  119. IME: {
  120. SKIP: {
  121. eval {$driver->available_engines;};
  122. if ($@) {
  123. skip "ime not available on this system",3;
  124. }
  125. }
  126. }
  127. LOAD_PAGE: {
  128. $driver->get("$website/index.html");
  129. pass('Loaded home page');
  130. $ret = $driver->get_title();
  131. is($ret, 'Hello WebDriver', 'Got the title');
  132. $ret = $driver->get_current_url();
  133. ok($ret =~ m/$website/i, 'Got proper URL');
  134. }
  135. WINDOW: {
  136. $ret = $driver->get_current_window_handle();
  137. ok($ret =~ m/^{.*}$/, 'Proper window handle received');
  138. $ret = $driver->get_window_handles();
  139. is(ref $ret, 'ARRAY', 'Received all window handles');
  140. $ret = $driver->set_window_position(100,100);
  141. is($ret, 1, 'Set the window position to 100, 100');
  142. $ret = $driver->get_window_position();
  143. is ($ret->{'x'}, 100, 'Got the right X Co-ordinate');
  144. is ($ret->{'y'}, 100, 'Got the right Y Co-ordinate');
  145. $ret = $driver->set_window_size(640, 480);
  146. is($ret, 1, 'Set the window size to 640x480');
  147. $ret = $driver->get_window_size();
  148. is ($ret->{'height'}, 640, 'Got the right height');
  149. is ($ret->{'width'}, 480, 'Got the right width');
  150. $ret = $driver->maximize_window();
  151. is ($ret, 1, "Got confirmation from maximize");
  152. SKIP: {
  153. skip 'headless browsers don\'t get maximized', 2
  154. unless $^O =~ /darwin|MSWin32/;
  155. $ret = $driver->get_window_size();
  156. ok ($ret->{'height'} > 640, 'Height has increased');
  157. ok ($ret->{'width'} > 480, 'Width has increased');
  158. }
  159. $ret = $driver->get_page_source();
  160. ok($ret =~ m/^<html/i, 'Received page source');
  161. eval {$driver->set_implicit_wait_timeout(20001);};
  162. ok(!$@,"Set implicit wait timeout");
  163. eval {$driver->set_implicit_wait_timeout(0);};
  164. ok(!$@,"Reset implicit wait timeout");
  165. $ret = $driver->get("$website/frameset.html");
  166. $ret = $driver->switch_to_frame('second');
  167. }
  168. COOKIES: {
  169. $driver->get("$website/cookies.html");
  170. $ret = $driver->get_all_cookies();
  171. is(@{$ret}, 2, 'Got 2 cookies');
  172. $ret = $driver->delete_all_cookies();
  173. pass('Deleting cookies...');
  174. $ret = $driver->get_all_cookies();
  175. is(@{$ret}, 0, 'Deleted all cookies.');
  176. $ret = $driver->add_cookie('foo', 'bar', '/', 'localhost', 0);
  177. pass('Adding cookie foo...');
  178. $ret = $driver->get_all_cookies();
  179. is(@{$ret}, 1, 'foo cookie added.');
  180. is($ret->[0]{'secure'}, 0, 'foo cookie insecure.');
  181. $ret = $driver->delete_cookie_named('foo');
  182. pass('Deleting cookie foo...');
  183. $ret = $driver->get_all_cookies();
  184. is(@{$ret}, 0, 'foo cookie deleted.');
  185. $ret = $driver->delete_all_cookies();
  186. }
  187. MOVE: {
  188. $driver->get("$website/index.html");
  189. $driver->get("$website/formPage.html");
  190. $ret = $driver->go_back();
  191. pass('Clicked Back...');
  192. $ret = $driver->get_title();
  193. is($ret, 'Hello WebDriver', 'Got the right title');
  194. $ret = $driver->go_forward();
  195. pass('Clicked Forward...');
  196. $ret = $driver->get_title();
  197. is($ret, 'We Leave From Here', 'Got the right title');
  198. $ret = $driver->refresh();
  199. pass('Clicked Refresh...');
  200. $ret = $driver->get_title();
  201. is($ret, 'We Leave From Here', 'Got the right title');
  202. }
  203. FIND: {
  204. my $elem = $driver->find_element("//input[\@id='checky']");
  205. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Xpath');
  206. $elem = $driver->find_element('checky', 'id');
  207. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Id');
  208. $elem = $driver->find_element('checky', 'name');
  209. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Name');
  210. $elem = $driver->find_element('multi', 'id');
  211. $elem = $driver->find_child_element($elem, "option");
  212. ok($elem->isa('Selenium::Remote::WebElement'), 'Got child WebElement...');
  213. $ret = $elem->get_value();
  214. is($ret, 'Eggs', '...right child WebElement');
  215. $ret = $driver->find_child_elements($elem, "//option[\@selected='selected']");
  216. is(@{$ret}, 4, 'Got 4 WebElements');
  217. my $expected_err = "An element could not be located on the page using the "
  218. . "given search parameters: "
  219. . "element_that_doesnt_exist,id"
  220. # the following needs to always be right before the eval
  221. . " at " . __FILE__ . " line " . (__LINE__+1);
  222. eval { $driver->find_element("element_that_doesnt_exist","id"); };
  223. chomp $@;
  224. like($@,qr/$expected_err/,"find_element croaks properly");
  225. my $elems = $driver->find_elements("//input[\@id='checky']");
  226. is(scalar(@$elems),1, 'Got an arrayref of WebElements');
  227. my @array_elems = $driver->find_elements("//input[\@id='checky']");
  228. is(scalar(@array_elems),1, 'Got an array of WebElements');
  229. is($elems->[0]->get_value(),$array_elems[0]->get_value(), 'and the elements returned are the same');
  230. }
  231. EXECUTE: {
  232. my $script = q{
  233. var arg1 = arguments[0];
  234. var elem = window.document.getElementById(arg1);
  235. return elem;
  236. };
  237. my $elem = $driver->execute_script($script,'checky');
  238. ok($elem->isa('Selenium::Remote::WebElement'), 'Executed script');
  239. is($elem->get_attribute('id'),'checky','Execute found proper element');
  240. $script = q{
  241. var links = window.document.links
  242. var length = links.length
  243. var results = new Array(length)
  244. while(length--) results[length] = links[length];
  245. return results;
  246. };
  247. $elem = $driver->execute_script($script);
  248. ok($elem, 'Got something back from execute_script');
  249. isa_ok($elem, 'ARRAY', 'What we got back is an ARRAY ref');
  250. ok(scalar(@$elem), 'There are elements in our array ref');
  251. foreach my $element (@$elem) {
  252. isa_ok($element, 'Selenium::Remote::WebElement', 'Element was converted to a WebElement object');
  253. }
  254. $script = q{
  255. var arg1 = arguments[0];
  256. var callback = arguments[arguments.length-1];
  257. var elem = window.document.getElementById(arg1);
  258. callback(elem);
  259. };
  260. $elem = $driver->execute_async_script($script,'multi');
  261. ok($elem->isa('Selenium::Remote::WebElement'),'Executed async script');
  262. is($elem->get_attribute('id'),'multi','Async found proper element');
  263. $script = 'return ""';
  264. my $empty_string = $driver->execute_script($script);
  265. cmp_ok($empty_string, 'eq', '', 'Empty strings are returned properly');
  266. }
  267. ALERT: {
  268. $driver->get("$website/alerts.html");
  269. $driver->find_element("alert",'id')->click;
  270. is($driver->get_alert_text,'cheese','alert text match');
  271. eval {$driver->dismiss_alert;};
  272. ok(!$@,"dismissed alert");
  273. $driver->find_element("prompt",'id')->click;
  274. is($driver->get_alert_text,'Enter your name','prompt text match');
  275. $driver->send_keys_to_prompt("Larry Wall");
  276. eval {$driver->accept_alert;};
  277. ok(!$@,"accepted prompt");
  278. is($driver->get_alert_text,'Larry Wall','keys sent to prompt');
  279. $driver->dismiss_alert;
  280. $driver->find_element("confirm",'id')->click;
  281. is($driver->get_alert_text,"Are you sure?",'confirm text match');
  282. eval {$driver->dismiss_alert;};
  283. ok(!$@,"dismissed confirm");
  284. is($driver->get_alert_text,'false',"dismissed confirmed correct");
  285. $driver->accept_alert;
  286. $driver->find_element("confirm",'id')->click;
  287. eval {$driver->accept_alert;};
  288. ok(!$@,"accepted confirm");
  289. is($driver->get_alert_text,'true',"accept confirm correct");
  290. $driver->accept_alert;
  291. }
  292. PAUSE: {
  293. my $starttime=time();
  294. $driver->pause();
  295. my $endtime=time();
  296. ok($starttime <= $endtime-1,"starttime <= endtime+1"); # Slept at least 1 second
  297. ok($starttime >= $endtime-2,"starttime >= endtime-2"); # Slept at most 2 seconds
  298. }
  299. AUTO_CLOSE: {
  300. my %stay_open_selenium_args = %selenium_args;
  301. $stay_open_selenium_args{auto_close} = 0;
  302. my $stayOpen = Selenium::Remote::Driver->new(
  303. %stay_open_selenium_args
  304. );
  305. $stayOpen->DESTROY();
  306. ok(defined $stayOpen->{'session_id'}, 'auto close in init hash is respected');
  307. $stayOpen->auto_close(1);
  308. $stayOpen->DESTROY();
  309. ok(!defined $stayOpen->{'session_id'}, 'true for auto close is still respected');
  310. $driver->auto_close(0);
  311. $driver->DESTROY();
  312. ok(defined $driver->{'session_id'}, 'changing autoclose on the fly keeps the session open');
  313. $driver->auto_close(1);
  314. }
  315. INNER_WINDOW_SIZE: {
  316. my %normal_selenium_args = %selenium_args;
  317. my $normal = Selenium::Remote::Driver->new(%normal_selenium_args)->get_window_size;
  318. my %resized_selenium_args = %selenium_args;
  319. $resized_selenium_args{inner_window_size} = [ 640,480];
  320. my $resized = Selenium::Remote::Driver->new(
  321. %resized_selenium_args
  322. )->get_window_size;
  323. ok($normal->{height} != $resized->{height}, 'inner window size: height is immediately changed');
  324. ok($normal->{width} != $resized->{width}, 'inner window size: width is immediately changed');
  325. }
  326. BASE_URL: {
  327. {
  328. package MySeleniumRemoteDriver;
  329. use Moo;
  330. extends 'Selenium::Remote::Driver';
  331. sub _execute_command { $_[2]->{url} }
  332. 1;
  333. }
  334. my @tests = ({
  335. base_url => 'http://example.com',
  336. url => '/foo',
  337. expected => 'http://example.com/foo',
  338. },{
  339. base_url => 'http://example.com/',
  340. url => '/foo',
  341. expected => 'http://example.com/foo',
  342. },{
  343. base_url => 'http://example.com',
  344. url => 'foo',
  345. expected => 'http://example.com/foo',
  346. },{
  347. base_url => 'http://example.com/a',
  348. url => '/foo',
  349. expected => 'http://example.com/a/foo',
  350. },{
  351. base_url => 'http://example.com/a',
  352. url => 'foo',
  353. expected => 'http://example.com/a/foo',
  354. },{
  355. base_url => 'http://example.com/a',
  356. url => 'http://blog.example.com/foo',
  357. expected => 'http://blog.example.com/foo',
  358. });
  359. my $mock_commands = Selenium::Remote::Mock::Commands->new;
  360. for my $test (@tests) {
  361. my $base_url_driver = MySeleniumRemoteDriver->new(
  362. browser_name => 'firefox',
  363. base_url => $test->{base_url},
  364. remote_conn => Selenium::Remote::Mock::RemoteConnection->new(
  365. spec => {
  366. get =>
  367. sub { my ( undef, $params ) = @_; return $params->{url} }
  368. },
  369. mock_cmds => $mock_commands
  370. ),
  371. commands => $mock_commands,
  372. );
  373. my $got = $base_url_driver->get($test->{url});
  374. is $got, $test->{expected}, "base_url + $test->{url}";
  375. }
  376. }
  377. USER_AGENT: {
  378. my $ua = $driver->get_user_agent;
  379. ok($ua =~ /Firefox/, 'we can get a user agent');
  380. }
  381. STORAGE: {
  382. my $chrome;
  383. my %selenium_chrome_args = ( browser_name => 'chrome');
  384. $selenium_chrome_args{remote_conn} = $selenium_args{remote_conn};
  385. SKIP: {
  386. eval {
  387. $chrome = Selenium::Remote::Driver->new(%selenium_chrome_args);
  388. $chrome->get($website);
  389. };
  390. if ($@ || !defined $chrome) {
  391. skip 'FirefoxDriver does not support Storage APIs; Chromedriver must be configured to perform storage tests', 3;
  392. }
  393. my ($key, $value) = ('testKey', 'testValue');
  394. $chrome->execute_script('localStorage.' . $key . ' = "' . $value . '"; return 1');
  395. my $actual = $chrome->get_local_storage_item($key);
  396. ok($actual eq $value, 'can retrieve local storage by key');
  397. ok($chrome->delete_local_storage_item($key), 'can delete local storage by key');
  398. my $now_empty = $chrome->get_local_storage_item($key);
  399. ok(!(defined $now_empty), 'retrieving an empty or deleted local storage key returns undef');
  400. }
  401. }
  402. QUIT: {
  403. $ret = $driver->quit();
  404. ok((not defined $driver->{'session_id'}), 'Killed the remote session');
  405. }
  406. done_testing;