01-driver.t 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. use strict;
  2. use warnings;
  3. use JSON;
  4. use Net::Ping;
  5. use Test::More;
  6. use Test::LWP::UserAgent;
  7. use Selenium::Remote::Driver;
  8. BEGIN {
  9. if (defined $ENV{'WD_MOCKING_RECORD'} && ($ENV{'WD_MOCKING_RECORD'}==1)) {
  10. use t::lib::MockSeleniumWebDriver;
  11. my $p = Net::Ping->new("tcp", 2);
  12. $p->port_number(4444);
  13. unless ($p->ping('localhost')) {
  14. plan skip_all => "Selenium server is not running on localhost:4444";
  15. exit;
  16. }
  17. warn "\n\nRecording...\n\n";
  18. }
  19. }
  20. my $record = (defined $ENV{'WD_MOCKING_RECORD'} && ($ENV{'WD_MOCKING_RECORD'}==1))?1:0;
  21. my $os = $^O;
  22. if ($os =~ m/(aix|freebsd|openbsd|sunos|solaris)/) {
  23. $os = 'linux';
  24. }
  25. my $mock_file = "01-driver-mock-$os.json";
  26. if (!$record && !(-e "t/mock-recordings/$mock_file")) {
  27. plan skip_all => "Mocking of tests is not been enabled for this platform";
  28. }
  29. t::lib::MockSeleniumWebDriver::register($record,"t/mock-recordings/$mock_file");
  30. my $driver = Selenium::Remote::Driver->new(browser_name => 'firefox');
  31. my $website = 'http://localhost:63636';
  32. my $ret;
  33. DESIRED_CAPABILITIES: {
  34. # We're using a different test method for these because we needed
  35. # to inspect payload of the POST to /session, and the method of
  36. # recording the RES/REQ pairs doesn't provide any easy way to do
  37. # that.
  38. my $tua = Test::LWP::UserAgent->new;
  39. my $res = {
  40. cmd_return => {},
  41. cmd_status => 'OK',
  42. sessionId => '123124123'
  43. };
  44. $tua->map_response(qr{status}, HTTP::Response->new(200, 'OK'));
  45. my $requests_count = 0;
  46. my $mock_session_handler = sub {
  47. my $request = shift;
  48. $requests_count++;
  49. if ($request->method eq 'POST') {
  50. my $caps = from_json($request->decoded_content)->{desiredCapabilities};
  51. my @keys = keys %$caps;
  52. if (scalar @keys) {
  53. ok(scalar @keys == 2, 'exactly 2 keys passed in if we use desired_capabilities');
  54. my $grep = grep { 'browserName' eq $_ } @keys;
  55. ok($grep, 'and it is the right one');
  56. ok($caps->{superfluous} eq 'thing', 'and we pass through anything else');
  57. ok($caps->{browserName} eq 'firefox', 'and we override the "normal" caps');
  58. ok(!exists $caps->{platform}, 'or ignore them entirely');
  59. }
  60. else {
  61. ok(to_json($caps) eq '{}', 'an empty constructor defaults to an empty hash');
  62. }
  63. return HTTP::Response->new(204, 'OK', ['Content-Type' => 'application/json'], to_json($res));
  64. }
  65. else {
  66. # it's the DELETE when the driver calls
  67. # DESTROY. This should be the last call to /session/.
  68. return HTTP::Response->new(200, 'OK')
  69. }
  70. };
  71. $tua->map_response(qr{session}, $mock_session_handler);
  72. my $caps_driver = Selenium::Remote::Driver->new_from_caps(
  73. auto_close => 0,
  74. browser_name => 'not firefox',
  75. platform => 'WINDOWS',
  76. desired_capabilities => {
  77. 'browserName' => 'firefox',
  78. 'superfluous' => 'thing'
  79. },
  80. ua => $tua
  81. );
  82. ok($caps_driver->auto_close eq 0, 'and other properties are still set');
  83. $caps_driver = Selenium::Remote::Driver->new(
  84. auto_close => 0,
  85. browser_name => 'not firefox',
  86. platform => 'WINDOWS',
  87. desired_capabilities => {
  88. 'browserName' => 'firefox',
  89. 'superfluous' => 'thing'
  90. },
  91. ua => $tua
  92. );
  93. ok($caps_driver->auto_close eq 0, 'and other properties are set if we use the normal constructor');
  94. $caps_driver = Selenium::Remote::Driver->new_from_caps(ua => $tua);
  95. ok($requests_count == 3, 'The new_from_caps section has the correct number of requests to /session/');
  96. }
  97. GRID_STARTUP: {
  98. # Mimicking a grid server; /wd/hub/status fails, and we expect
  99. # grid/api/hub/status to be checked instead.
  100. my $tua = Test::LWP::UserAgent->new;
  101. my $not_ok = sub {
  102. return HTTP::Response->new(500, 'NOTOK');
  103. };
  104. $tua->map_response(qr{wd/hub/status}, $not_ok);
  105. my $grid_status_count = 0;
  106. my $ok = sub {
  107. my $res = {
  108. cmd_return => {},
  109. cmd_status => 'OK',
  110. sessionId => '123124123'
  111. };
  112. $grid_status_count++;
  113. return HTTP::Response->new(200, 'OK', ['Content-Type' => 'application/json'], to_json($res));
  114. };
  115. $tua->map_response(qr{(?:grid/api/hub/status|session)}, $ok);
  116. my $grid_driver = Selenium::Remote::Driver->new(ua => $tua);
  117. ok(defined $grid_driver, 'Grid: Object loaded fine using grid/api/hub/status');
  118. ok($grid_driver->isa('Selenium::Remote::Driver'), 'Grid: ...and of right type');
  119. ok($grid_status_count == 2, 'checked Grid specific status');
  120. }
  121. CHECK_DRIVER: {
  122. ok(defined $driver, 'Object loaded fine...');
  123. ok($driver->isa('Selenium::Remote::Driver'), '...and of right type');
  124. ok(defined $driver->{'session_id'}, 'Established session on remote server');
  125. $ret = $driver->get_capabilities;
  126. is($ret->{'browserName'}, 'firefox', 'Right capabilities');
  127. my $status = $driver->status;
  128. ok($status->{build}->{version},"Got status build.version");
  129. ok($status->{build}->{revision},"Got status build.revision");
  130. ok($status->{build}->{time},"Got status build.time");
  131. }
  132. IME: {
  133. SKIP: {
  134. eval {$driver->available_engines;};
  135. if ($@) {
  136. skip "ime not available on this system",3;
  137. }
  138. };
  139. }
  140. LOAD_PAGE: {
  141. $driver->get("$website/index.html");
  142. pass('Loaded home page');
  143. $ret = $driver->get_title();
  144. is($ret, 'Hello WebDriver', 'Got the title');
  145. $ret = $driver->get_current_url();
  146. ok($ret =~ m/$website/i, 'Got proper URL');
  147. }
  148. WINDOW: {
  149. $ret = $driver->get_current_window_handle();
  150. ok($ret =~ m/^{.*}$/, 'Proper window handle received');
  151. $ret = $driver->get_window_handles();
  152. is(ref $ret, 'ARRAY', 'Received all window handles');
  153. $ret = $driver->set_window_position(100,100);
  154. is($ret, 1, 'Set the window position to 100, 100');
  155. $ret = $driver->get_window_position();
  156. is ($ret->{'x'}, 100, 'Got the right X Co-ordinate');
  157. is ($ret->{'y'}, 100, 'Got the right Y Co-ordinate');
  158. $ret = $driver->set_window_size(640, 480);
  159. is($ret, 1, 'Set the window size to 640x480');
  160. $ret = $driver->get_window_size();
  161. is ($ret->{'height'}, 640, 'Got the right height');
  162. is ($ret->{'width'}, 480, 'Got the right width');
  163. $ret = $driver->get_page_source();
  164. ok($ret =~ m/^<html/i, 'Received page source');
  165. eval {$driver->set_implicit_wait_timeout(20001);};
  166. ok(!$@,"Set implicit wait timeout");
  167. eval {$driver->set_implicit_wait_timeout(0);};
  168. ok(!$@,"Reset implicit wait timeout");
  169. $ret = $driver->get("$website/frameset.html");
  170. $ret = $driver->switch_to_frame('second');
  171. }
  172. COOKIES: {
  173. $driver->get("$website/cookies.html");
  174. $ret = $driver->get_all_cookies();
  175. is(@{$ret}, 2, 'Got 2 cookies');
  176. $ret = $driver->delete_all_cookies();
  177. pass('Deleting cookies...');
  178. $ret = $driver->get_all_cookies();
  179. is(@{$ret}, 0, 'Deleted all cookies.');
  180. $ret = $driver->add_cookie('foo', 'bar', '/', 'localhost', 0);
  181. pass('Adding cookie foo...');
  182. $ret = $driver->get_all_cookies();
  183. is(@{$ret}, 1, 'foo cookie added.');
  184. is($ret->[0]{'secure'}, 0, 'foo cookie insecure.');
  185. $ret = $driver->delete_cookie_named('foo');
  186. pass('Deleting cookie foo...');
  187. $ret = $driver->get_all_cookies();
  188. is(@{$ret}, 0, 'foo cookie deleted.');
  189. $ret = $driver->delete_all_cookies();
  190. }
  191. MOVE: {
  192. $driver->get("$website/index.html");
  193. $driver->get("$website/formPage.html");
  194. $ret = $driver->go_back();
  195. pass('Clicked Back...');
  196. $ret = $driver->get_title();
  197. is($ret, 'Hello WebDriver', 'Got the right title');
  198. $ret = $driver->go_forward();
  199. pass('Clicked Forward...');
  200. $ret = $driver->get_title();
  201. is($ret, 'We Leave From Here', 'Got the right title');
  202. $ret = $driver->refresh();
  203. pass('Clicked Refresh...');
  204. $ret = $driver->get_title();
  205. is($ret, 'We Leave From Here', 'Got the right title');
  206. }
  207. FIND: {
  208. my $elem = $driver->find_element("//input[\@id='checky']");
  209. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Xpath');
  210. $elem = $driver->find_element('checky', 'id');
  211. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Id');
  212. $elem = $driver->find_element('checky', 'name');
  213. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Name');
  214. $elem = $driver->find_element('multi', 'id');
  215. $elem = $driver->find_child_element($elem, "option");
  216. ok($elem->isa('Selenium::Remote::WebElement'), 'Got child WebElement...');
  217. $ret = $elem->get_value();
  218. is($ret, 'Eggs', '...right child WebElement');
  219. $ret = $driver->find_child_elements($elem, "//option[\@selected='selected']");
  220. is(@{$ret}, 4, 'Got 4 WebElements');
  221. my $expected_err = "An element could not be located on the page using the "
  222. . "given search parameters: "
  223. . "element_that_doesnt_exist,id"
  224. # the following needs to always be right before the eval
  225. . " at " . __FILE__ . " line " . (__LINE__+1);
  226. eval { $driver->find_element("element_that_doesnt_exist","id"); };
  227. chomp $@;
  228. is($@,$expected_err.".","find_element croaks properly");
  229. my $elems = $driver->find_elements("//input[\@id='checky']");
  230. is(scalar(@$elems),1, 'Got an arrayref of WebElements');
  231. my @array_elems = $driver->find_elements("//input[\@id='checky']");
  232. is(scalar(@array_elems),1, 'Got an array of WebElements');
  233. is($elems->[0]->get_value(),$array_elems[0]->get_value(), 'and the elements returned are the same');
  234. }
  235. EXECUTE: {
  236. my $script = q{
  237. var arg1 = arguments[0];
  238. var elem = window.document.getElementById(arg1);
  239. return elem;
  240. };
  241. my $elem = $driver->execute_script($script,'checky');
  242. ok($elem->isa('Selenium::Remote::WebElement'), 'Executed script');
  243. is($elem->get_attribute('id'),'checky','Execute found proper element');
  244. $script = q{
  245. var links = window.document.links
  246. var length = links.length
  247. var results = new Array(length)
  248. while(length--) results[length] = links[length];
  249. return results;
  250. };
  251. $elem = $driver->execute_script($script);
  252. ok($elem, 'Got something back from execute_script');
  253. isa_ok($elem, 'ARRAY', 'What we got back is an ARRAY ref');
  254. ok(scalar(@$elem), 'There are elements in our array ref');
  255. foreach my $element (@$elem) {
  256. isa_ok($element, 'Selenium::Remote::WebElement', 'Element was converted to a WebElement object');
  257. }
  258. $script = q{
  259. var arg1 = arguments[0];
  260. var callback = arguments[arguments.length-1];
  261. var elem = window.document.getElementById(arg1);
  262. callback(elem);
  263. };
  264. $elem = $driver->execute_async_script($script,'multi');
  265. ok($elem->isa('Selenium::Remote::WebElement'),'Executed async script');
  266. is($elem->get_attribute('id'),'multi','Async found proper element');
  267. }
  268. ALERT: {
  269. $driver->get("$website/alerts.html");
  270. $driver->find_element("alert",'id')->click;
  271. is($driver->get_alert_text,'cheese','alert text match');
  272. eval {$driver->dismiss_alert;};
  273. ok(!$@,"dismissed alert");
  274. $driver->find_element("prompt",'id')->click;
  275. is($driver->get_alert_text,'Enter your name','prompt text match');
  276. $driver->send_keys_to_prompt("Larry Wall");
  277. eval {$driver->accept_alert;};
  278. ok(!$@,"accepted prompt");
  279. is($driver->get_alert_text,'Larry Wall','keys sent to prompt');
  280. $driver->dismiss_alert;
  281. $driver->find_element("confirm",'id')->click;
  282. is($driver->get_alert_text,"Are you sure?",'confirm text match');
  283. eval {$driver->dismiss_alert;};
  284. ok(!$@,"dismissed confirm");
  285. is($driver->get_alert_text,'false',"dismissed confirmed correct");
  286. $driver->accept_alert;
  287. $driver->find_element("confirm",'id')->click;
  288. eval {$driver->accept_alert;};
  289. ok(!$@,"accepted confirm");
  290. is($driver->get_alert_text,'true',"accept confirm correct");
  291. $driver->accept_alert;
  292. }
  293. PAUSE: {
  294. my $starttime=time();
  295. $driver->pause();
  296. my $endtime=time();
  297. ok($starttime <= $endtime-1,"starttime <= endtime+1"); # Slept at least 1 second
  298. ok($starttime >= $endtime-2,"starttime >= endtime-2"); # Slept at most 2 seconds
  299. }
  300. AUTO_CLOSE: {
  301. my $stayOpen = Selenium::Remote::Driver->new(
  302. browser_name => 'firefox',
  303. auto_close => 0
  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. QUIT: {
  316. $ret = $driver->quit();
  317. ok((not defined $driver->{'session_id'}), 'Killed the remote session');
  318. }
  319. done_testing;