01-driver.t 17 KB

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