01-driver.t 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607
  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 Test::MockModule v0.13;
  9. use Selenium::Remote::Driver;
  10. use Selenium::Remote::Mock::Commands;
  11. use Selenium::Remote::Mock::RemoteConnection;
  12. use Selenium::Waiter;
  13. use Carp;
  14. use FindBin;
  15. use lib $FindBin::Bin . '/lib';
  16. use TestHarness;
  17. use Test::Fatal;
  18. $Selenium::Remote::Driver::FORCE_WD2 = 1;
  19. my $harness = TestHarness->new(
  20. this_file => $FindBin::Script
  21. );
  22. my %selenium_args = %{ $harness->base_caps };
  23. my $driver = Selenium::Remote::Driver->new(%selenium_args);
  24. my $domain = $harness->domain;
  25. my $website = $harness->website;
  26. my $ret;
  27. my $chrome;
  28. eval { $chrome = Selenium::Remote::Driver->new(
  29. %selenium_args,
  30. browser_name => 'chrome'
  31. ); };
  32. DESIRED_CAPABILITIES: {
  33. # We're using a different test method for these because we needed
  34. # to inspect payload of the POST to /session, and the method of
  35. # recording the RES/REQ pairs doesn't provide any easy way to do
  36. # that.
  37. my $tua = Test::LWP::UserAgent->new;
  38. my $res = {
  39. cmd_return => {},
  40. cmd_status => 'OK',
  41. sessionId => '123124123'
  42. };
  43. $tua->map_response(qr{status}, HTTP::Response->new(200, 'OK'));
  44. my $requests_count = 0;
  45. my $mock_session_handler = sub {
  46. my $request = shift;
  47. $requests_count++;
  48. if ($request->method eq 'POST') {
  49. my $caps = from_json($request->decoded_content)->{desiredCapabilities};
  50. my @keys = keys %$caps;
  51. if (scalar @keys) {
  52. ok(scalar @keys == 2, 'exactly 2 keys passed in if we use desired_capabilities');
  53. my $grep = grep { 'browserName' eq $_ } @keys;
  54. ok($grep, 'and it is the right one');
  55. ok($caps->{superfluous} eq 'thing', 'and we pass through anything else');
  56. ok($caps->{browserName} eq 'firefox', 'and we override the "normal" caps');
  57. ok(!exists $caps->{platform}, 'or ignore them entirely');
  58. }
  59. else {
  60. ok(to_json($caps) eq '{}', 'an empty constructor defaults to an empty hash');
  61. }
  62. return HTTP::Response->new(204, 'OK', ['Content-Type' => 'application/json'], to_json($res));
  63. }
  64. else {
  65. # it's the DELETE when the driver calls
  66. # DESTROY. This should be the last call to /session/.
  67. return HTTP::Response->new(200, 'OK')
  68. }
  69. };
  70. $tua->map_response(qr{session}, $mock_session_handler);
  71. my $caps_driver = Selenium::Remote::Driver->new_from_caps(
  72. auto_close => 0,
  73. browser_name => 'not firefox',
  74. platform => 'WINDOWS',
  75. desired_capabilities => {
  76. 'browserName' => 'firefox',
  77. 'superfluous' => 'thing'
  78. },
  79. ua => $tua
  80. );
  81. ok($caps_driver->auto_close eq 0, 'and other properties are still set');
  82. $caps_driver = Selenium::Remote::Driver->new(
  83. auto_close => 0,
  84. browser_name => 'not firefox',
  85. platform => 'WINDOWS',
  86. desired_capabilities => {
  87. 'browserName' => 'firefox',
  88. 'superfluous' => 'thing'
  89. },
  90. ua => $tua
  91. );
  92. ok($caps_driver->auto_close eq 0, 'and other properties are set if we use the normal constructor');
  93. $caps_driver = Selenium::Remote::Driver->new_from_caps(ua => $tua);
  94. ok($requests_count == 3, 'The new_from_caps section has the correct number of requests to /session/');
  95. }
  96. GRID_STARTUP: {
  97. # Mimicking a grid server; /wd/hub/status fails, and we expect
  98. # grid/api/hub/status to be checked instead.
  99. my $tua = Test::LWP::UserAgent->new;
  100. my $not_ok = sub {
  101. return HTTP::Response->new(500, 'NOTOK');
  102. };
  103. $tua->map_response(qr{wd/hub/status}, $not_ok);
  104. my $grid_status_count = 0;
  105. my $ok = sub {
  106. my $res = {
  107. cmd_return => {},
  108. cmd_status => 'OK',
  109. sessionId => '123124123'
  110. };
  111. $grid_status_count++;
  112. return HTTP::Response->new(200, 'OK', ['Content-Type' => 'application/json'], to_json($res));
  113. };
  114. $tua->map_response(qr{(?:grid/api/hub/status|session)}, $ok);
  115. my $mock = Test::MockModule->new('Selenium::Remote::RemoteConnection');
  116. $mock->mock('check_status', sub { $grid_status_count++; 1 });
  117. my $grid_driver = Selenium::Remote::Driver->new(ua => $tua);
  118. ok(defined $grid_driver, 'Grid: Object loaded fine using grid/api/hub/status');
  119. ok($grid_driver->isa('Selenium::Remote::Driver'), 'Grid: ...and of right type');
  120. ok($grid_status_count == 2, 'checked Grid specific status');
  121. }
  122. CHECK_DRIVER: {
  123. ok(defined $driver, 'Object loaded fine...');
  124. ok($driver->isa('Selenium::Remote::Driver'), '...and of right type');
  125. ok($driver->does('Selenium::Remote::Driver::CanSetWebdriverContext'),
  126. 'and can set webdriver context prefix');
  127. ok(defined $driver->{'session_id'}, 'Established session on remote server');
  128. $ret = $driver->get_capabilities;
  129. is($ret->{'browserName'}, 'firefox', 'Right capabilities');
  130. my $status = $driver->status;
  131. ok($status->{build}->{version},"Got status build.version");
  132. ok($status->{build}->{revision},"Got status build.revision");
  133. ok($status->{build}->{time},"Got status build.time");
  134. }
  135. IME: {
  136. SKIP: {
  137. eval {$driver->available_engines;};
  138. if ($@) {
  139. skip "ime not available on this system",3;
  140. }
  141. }
  142. }
  143. LOAD_PAGE: {
  144. $driver->get("$website/index.html");
  145. note('Loaded home page');
  146. $ret = $driver->get_title();
  147. is($ret, 'Hello WebDriver', 'Got the title');
  148. $ret = $driver->get_current_url();
  149. ok($ret =~ m/$website/i, 'Got proper URL');
  150. }
  151. WINDOW: {
  152. SKIP: {
  153. skip 'window position is busted in 3.4.0', 8;
  154. $ret = $driver->get_current_window_handle();
  155. $ret = $driver->get_window_handles();
  156. is(ref $ret, 'ARRAY', 'Received all window handles');
  157. $ret = $driver->set_window_position(100,100);
  158. is($ret, 1, 'Set the window position to 100, 100');
  159. $ret = $driver->get_window_position();
  160. is ($ret->{'x'}, 100, 'Got the right X Co-ordinate');
  161. is ($ret->{'y'}, 100, 'Got the right Y Co-ordinate');
  162. $ret = $driver->set_window_size(640, 480);
  163. is($ret, 1, 'Set the window size to 640x480');
  164. $ret = $driver->get_window_size();
  165. is ($ret->{'height'}, 640, 'Got the right height');
  166. is ($ret->{'width'}, 480, 'Got the right width');
  167. $ret = $driver->maximize_window();
  168. is ($ret, 1, "Got confirmation from maximize");
  169. }
  170. SKIP: {
  171. skip 'window position is busted in 3.4.0', 2;
  172. # skip 'headless browsers don\'t get maximized', 2
  173. # unless $^O =~ /darwin|MSWin32/;
  174. $ret = $driver->get_window_size();
  175. ok ($ret->{'height'} > 640, 'Height has increased');
  176. ok ($ret->{'width'} > 480, 'Width has increased');
  177. }
  178. $ret = $driver->get_page_source();
  179. ok($ret =~ m/^<html/i, 'Received page source');
  180. # Using a string instead of an int exercises the
  181. # _coerce_timeout_ms functionality
  182. eval {$driver->set_implicit_wait_timeout("20001");};
  183. ok(!$@,"Set implicit wait timeout with string");
  184. eval {$driver->set_implicit_wait_timeout(0);};
  185. ok(!$@,"Reset implicit wait timeout with integer");
  186. my $invalid_ms = 'invalid timeout ms';
  187. ok( exception{ $driver->set_timeout('script', $invalid_ms) },
  188. 'Coerce ms arguments for set_timeout' );
  189. ok( exception{ $driver->set_async_script_timeout( $invalid_ms ) },
  190. 'Coerce ms arguments for set_async_script_timeout' );
  191. ok( exception{ $driver->set_implicit_wait_timeout( $invalid_ms ) },
  192. 'Coerce ms arguments for set_implicit_wait_timeout' );
  193. $ret = $driver->get("$website/frameset.html");
  194. $ret = $driver->switch_to_frame('second');
  195. SKIP: {
  196. skip 'Cannot rotate desktop browsers', 3;
  197. ok($driver->get_orientation eq 'PORTRAIT', 'Can get default orientation');
  198. $ret = $driver->set_orientation('LANDSCAPE');
  199. ok($ret, 'Can change orientation to LANDSCAPE');
  200. ok($driver->get_orientation eq 'LANDSCAPE', 'Can get changed orientation');
  201. }
  202. }
  203. COOKIES: {
  204. $driver->get("$website/cookies.html");
  205. $ret = $driver->get_all_cookies();
  206. is(@{$ret}, 2, 'Got 2 cookies');
  207. $ret = $driver->delete_all_cookies();
  208. note('Deleting cookies...');
  209. $ret = $driver->get_all_cookies();
  210. is(@{$ret}, 0, 'Deleted all cookies.');
  211. $driver->debug_on();
  212. $ret = $driver->add_cookie('foo', 'bar', '/', $domain, 0);
  213. note('Adding cookie foo...');
  214. $ret = $driver->get_all_cookies();
  215. is(@{$ret}, 1, 'foo cookie added.');
  216. is($ret->[0]{'secure'}, 0, 'foo cookie insecure.');
  217. $ret = $driver->delete_cookie_named('foo');
  218. note('Deleting cookie foo...');
  219. $ret = $driver->get_all_cookies();
  220. is(@{$ret}, 0, 'foo cookie deleted.');
  221. $ret = $driver->delete_all_cookies();
  222. }
  223. MOVE: {
  224. $driver->get("$website/index.html");
  225. $driver->get("$website/formPage.html");
  226. $ret = $driver->go_back();
  227. note('Clicked Back...');
  228. $ret = $driver->get_title();
  229. is($ret, 'Hello WebDriver', 'Got the right title');
  230. $ret = $driver->go_forward();
  231. note('Clicked Forward...');
  232. $ret = $driver->get_title();
  233. is($ret, 'We Leave From Here', 'Got the right title');
  234. $ret = $driver->refresh();
  235. note('Clicked Refresh...');
  236. $ret = $driver->get_title();
  237. is($ret, 'We Leave From Here', 'Got the right title');
  238. }
  239. FIND: {
  240. my $elem = $driver->find_element("//input[\@id='checky']");
  241. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Xpath');
  242. $elem = $driver->find_element('checky', 'id');
  243. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Id');
  244. $elem = $driver->find_element('checky', 'name');
  245. ok($elem->isa('Selenium::Remote::WebElement'), 'Got WebElement via Name');
  246. $driver->default_finder('css');
  247. $elem = $driver->find_element('#multi');
  248. $elem = $driver->find_child_element($elem, "option[selected]");
  249. ok($elem->isa('Selenium::Remote::WebElement'), 'Got child WebElement...');
  250. $driver->default_finder('xpath');
  251. $elem = $driver->find_element('multi', 'id');
  252. $elem = $driver->find_child_element($elem, "option");
  253. ok($elem->isa('Selenium::Remote::WebElement'), 'Got child WebElement...');
  254. $ret = $elem->get_value();
  255. is($ret, 'Eggs', '...right child WebElement');
  256. $ret = $driver->find_child_elements($elem, "//option[\@selected='selected']");
  257. is(@{$ret}, 4, 'Got 4 WebElements');
  258. my $expected_err = "An element could not be located on the page using the "
  259. . "given search parameters: "
  260. . "element_that_doesnt_exist,id"
  261. # the following needs to always be right before the eval
  262. . " at " . __FILE__ . " line " . (__LINE__+1);
  263. eval { $driver->find_element("element_that_doesnt_exist","id"); };
  264. chomp $@;
  265. like($@,qr/\Q$expected_err\E/,"find_element croaks properly");
  266. my $elems = $driver->find_elements("//input[\@id='checky']");
  267. is(scalar(@$elems),1, 'Got an arrayref of WebElements');
  268. my @array_elems = $driver->find_elements("//input[\@id='checky']");
  269. is(scalar(@array_elems),1, 'Got an array of WebElements');
  270. is($elems->[0]->get_value(),$array_elems[0]->get_value(), 'and the elements returned are the same');
  271. }
  272. EXECUTE: {
  273. my $script = q{
  274. var arg1 = arguments[0];
  275. var elem = window.document.getElementById(arg1);
  276. return elem;
  277. };
  278. my $elem = $driver->execute_script($script,'checky');
  279. ok($elem->isa('Selenium::Remote::WebElement'), 'Executed script');
  280. is($elem->get_attribute('id'),'checky','Execute found proper element');
  281. $script = q{
  282. var links = window.document.links
  283. var length = links.length
  284. var results = new Array(length)
  285. while(length--) results[length] = links[length];
  286. return results;
  287. };
  288. $elem = $driver->execute_script($script);
  289. ok($elem, 'Got something back from execute_script');
  290. isa_ok($elem, 'ARRAY', 'What we got back is an ARRAY ref');
  291. ok(scalar(@$elem), 'There are elements in our array ref');
  292. foreach my $element (@$elem) {
  293. isa_ok($element, 'Selenium::Remote::WebElement', 'Element was converted to a WebElement object');
  294. }
  295. $script = q{
  296. var arg1 = arguments[0];
  297. var callback = arguments[arguments.length-1];
  298. var elem = window.document.getElementById(arg1);
  299. callback(elem);
  300. };
  301. $elem = $driver->execute_async_script($script,'multi');
  302. ok($elem->isa('Selenium::Remote::WebElement'),'Executed async script');
  303. is($elem->get_attribute('id'),'multi','Async found proper element');
  304. $script = 'return ""';
  305. my $empty_string = $driver->execute_script($script);
  306. cmp_ok($empty_string, 'eq', '', 'Empty strings are returned properly');
  307. }
  308. ALERT: {
  309. $driver->get("$website/alerts.html");
  310. $driver->find_element("alert",'id')->click;
  311. is($driver->get_alert_text,'cheese','alert text match');
  312. eval {$driver->dismiss_alert;};
  313. ok(!$@,"dismissed alert");
  314. $driver->find_element("prompt",'id')->click;
  315. is($driver->get_alert_text,'Enter your name','prompt text match');
  316. $driver->send_keys_to_prompt("Larry Wall");
  317. eval {$driver->accept_alert;};
  318. ok(!$@,"accepted prompt");
  319. is(wait_until { $driver->get_alert_text },'Larry Wall','keys sent to prompt');
  320. $driver->dismiss_alert;
  321. $driver->find_element("confirm",'id')->click;
  322. is($driver->get_alert_text,"Are you sure?",'confirm text match');
  323. eval {$driver->dismiss_alert;};
  324. ok(!$@,"dismissed confirm");
  325. is(wait_until { $driver->get_alert_text },'false',"dismissed confirmed correct");
  326. $driver->accept_alert;
  327. $driver->find_element("confirm",'id')->click;
  328. eval {$driver->accept_alert;};
  329. ok(!$@,"accepted confirm");
  330. is(wait_until { $driver->get_alert_text } ,'true',"accept confirm correct");
  331. $driver->accept_alert;
  332. }
  333. PAUSE: {
  334. my $starttime=time();
  335. $driver->pause();
  336. my $endtime=time();
  337. ok($starttime <= $endtime-1,"starttime <= endtime+1"); # Slept at least 1 second
  338. ok($starttime >= $endtime-2,"starttime >= endtime-2"); # Slept at most 2 seconds
  339. }
  340. AUTO_CLOSE: {
  341. my %stay_open_selenium_args = %selenium_args;
  342. $stay_open_selenium_args{auto_close} = 0;
  343. my $stayOpen = Selenium::Remote::Driver->new(
  344. %stay_open_selenium_args
  345. );
  346. $stayOpen->DESTROY();
  347. ok(defined $stayOpen->{'session_id'}, 'auto close in init hash is respected');
  348. $stayOpen->auto_close(1);
  349. $stayOpen->DESTROY();
  350. ok(!defined $stayOpen->{'session_id'}, 'true for auto close is still respected');
  351. $driver->auto_close(0);
  352. $driver->DESTROY();
  353. ok(defined $driver->{'session_id'}, 'changing autoclose on the fly keeps the session open');
  354. $driver->auto_close(1);
  355. }
  356. INNER_WINDOW_SIZE: {
  357. SKIP: {
  358. skip 'window position is busted in 3.4.0', 2;
  359. my %normal_selenium_args = %selenium_args;
  360. my $normal = Selenium::Remote::Driver->new(%normal_selenium_args)->get_window_size;
  361. my %resized_selenium_args = %selenium_args;
  362. $resized_selenium_args{inner_window_size} = [ 640,480];
  363. my $resized = Selenium::Remote::Driver->new(
  364. %resized_selenium_args
  365. )->get_window_size;
  366. ok($normal->{height} != $resized->{height}, 'inner window size: height is immediately changed');
  367. ok($normal->{width} != $resized->{width}, 'inner window size: width is immediately changed');
  368. }
  369. }
  370. BASE_URL: {
  371. {
  372. package MySeleniumRemoteDriver;
  373. use Moo;
  374. extends 'Selenium::Remote::Driver';
  375. sub _execute_command { $_[2]->{url} }
  376. 1;
  377. }
  378. my @tests = ({
  379. base_url => 'http://example.com',
  380. url => '/foo',
  381. expected => 'http://example.com/foo',
  382. },{
  383. base_url => 'http://example.com/',
  384. url => '/foo',
  385. expected => 'http://example.com/foo',
  386. },{
  387. base_url => 'http://example.com',
  388. url => 'foo',
  389. expected => 'http://example.com/foo',
  390. },{
  391. base_url => 'http://example.com/a',
  392. url => '/foo',
  393. expected => 'http://example.com/a/foo',
  394. },{
  395. base_url => 'http://example.com/a',
  396. url => 'foo',
  397. expected => 'http://example.com/a/foo',
  398. },{
  399. base_url => 'http://example.com/a',
  400. url => 'http://blog.example.com/foo',
  401. expected => 'http://blog.example.com/foo',
  402. });
  403. my $mock_commands = Selenium::Remote::Mock::Commands->new;
  404. for my $test (@tests) {
  405. my $base_url_driver = MySeleniumRemoteDriver->new(
  406. browser_name => 'firefox',
  407. base_url => $test->{base_url},
  408. remote_conn => Selenium::Remote::Mock::RemoteConnection->new(
  409. spec => {
  410. get =>
  411. sub { my ( undef, $params ) = @_; return $params->{url} }
  412. },
  413. mock_cmds => $mock_commands
  414. ),
  415. commands => $mock_commands,
  416. );
  417. my $got = $base_url_driver->get($test->{url});
  418. is $got, $test->{expected}, "base_url + $test->{url}";
  419. }
  420. }
  421. USER_AGENT: {
  422. my $ua = $driver->get_user_agent;
  423. ok($ua =~ /Firefox/, 'we can get a user agent');
  424. }
  425. STORAGE: {
  426. SKIP: {
  427. eval {
  428. $chrome->get($website);
  429. };
  430. if ($@ || !defined $chrome) {
  431. skip 'FirefoxDriver does not support Storage APIs; Chromedriver must be configured to perform storage tests', 3;
  432. }
  433. my ($key, $value) = ('testKey', 'testValue');
  434. $chrome->execute_script('localStorage.' . $key . ' = "' . $value . '"; return 1');
  435. my $actual = $chrome->get_local_storage_item($key);
  436. ok($actual eq $value, 'can retrieve local storage by key');
  437. ok($chrome->delete_local_storage_item($key), 'can delete local storage by key');
  438. my $now_empty = $chrome->get_local_storage_item($key);
  439. ok(!(defined $now_empty), 'retrieving an empty or deleted local storage key returns undef');
  440. }
  441. }
  442. HTML5: {
  443. SKIP: {
  444. skip 'HTML5 Application Cache is not supported by firefox or chrome', 1 if 1;
  445. $driver->get($website);
  446. my $status = $driver->cache_status;
  447. ok($status, 'we can get application cache status');
  448. }
  449. SKIP: {
  450. skip 'Geolocation requires Chrome to test', 2 unless $chrome;
  451. my $ret = $chrome->set_geolocation( location => {
  452. latitude => 40.714353,
  453. longitude => -74.005973,
  454. altitude => 0.056747
  455. });
  456. ok($ret, 'can set geolocation');
  457. # GET geolocation has a cast Long to Double error in Chromedriver
  458. # my $ret = {};
  459. # eval { $ret = $chrome->get_geolocation };
  460. # ok(exists $ret->{location}, 'get_geolocation returns a location dictionary.');
  461. }
  462. }
  463. LOGS: {
  464. skip 'Log Types requires Chrome to test', 5 unless $chrome;
  465. $chrome->get($website);
  466. my $types = $chrome->get_log_types;
  467. ok(scalar @$types >= 4, 'Can get log types');
  468. foreach (@$types) {
  469. my $log = $chrome->get_log($_);
  470. ok(defined $log, 'Can get logs from the ' . $_);
  471. }
  472. }
  473. UPLOAD: {
  474. #Webdriver only returns the full filename if there isn't any path components in it, so test both cases
  475. my $testFile = "UEsDBBQACAAIAFtuNEYAAAAAAAAAAAAAAAAKABUAdXBsb2FkVGVzdFVUCQADjbG+VJ6xvlRVeAQA\n6APoAytJLS4BAFBLBwgMfn/YBgAAAAQAAABQSwECFAMUAAgACABbbjRGDH5/2AYAAAAEAAAACgAN\nAAAAAAAAAAAApIEAAAAAdXBsb2FkVGVzdFVUBQABjbG+VFV4AABQSwUGAAAAAAEAAQBFAAAAUwAA\nAAAA\n";
  476. my $otherTestFile = "UEsDBBQACAAIAFtuNEYAAAAAAAAAAAAAAAAMABUAdC91cGxvYWRUZXN0VVQJAAOesb5UnrG+VFV4\nBADoA+gDK0ktLgEAUEsHCAx+f9gGAAAABAAAAFBLAQIUAxQACAAIAFtuNEYMfn/YBgAAAAQAAAAM\nAA0AAAAAAAAAAACkgQAAAAB0L3VwbG9hZFRlc3RVVAUAAZ6xvlRVeAAAUEsFBgAAAAABAAEARwAA\nAFUAAAAAAA==\n";
  477. like( my $path1 = $driver->upload_file('uploadTest',$testFile),qr/uploadTest$/,'upload_file returns FULL path to the file: cwd');
  478. ok(-f $path1) if $harness->record;
  479. my $fake_driver;
  480. if ($harness->record) {
  481. $fake_driver = $driver;
  482. } else {
  483. #Going to have to use a custom UA to test this, since bogosity ensues
  484. my $ftua = Test::LWP::UserAgent->new;
  485. my $fake_header = bless( {'content-type' => 'application/json; charset=utf-8'}, 'HTTP::Headers' );
  486. #Seems a bit heavy handed, but shouldn't be a problem.
  487. $ftua->map_response(qr{.*}, HTTP::Response->new(200, 'OK', $fake_header, '{"sessionId":"89726c41-409f-421e-95a8-8b1fa482fa33","status":0,"state":"success","value":"/tmp/89726c41-409f-421e-95a8-8b1fa482fa33/upload1105744174029267337file/uploadTest","class":"org.openqa.selenium.remote.Response","hCode":516517658}'));
  488. $fake_driver = Selenium::Remote::Driver->new(
  489. ua => $ftua
  490. );
  491. }
  492. #In the case this is not mocked, it tests a real issue (.. in paths), if not, it makes sure the zip/base64 bits at least don't make us explode.
  493. like( my $path2 = $fake_driver->upload_file($FindBin::Bin . '/uploadTest'),qr/uploadTest$/,'upload_file: zip/base64 branch' );
  494. like( my $path3 = $fake_driver->upload_file($FindBin::Bin . '/../t/uploadTest'),qr/uploadTest$/,'upload_file: zip/base64 branch with .. in path' );
  495. unlike( $path3,qr/\.\./,'upload_file: zip/base64 branch with .. in path' );
  496. ok(-f $path2) if $harness->record;
  497. ok(-f $path3) if $harness->record;
  498. #Negative tests to verify that our expected behavior codepath is travelled by tests
  499. like( exception { $driver->upload_file('@@@SomeFileThatDoesNotExist@@@')},qr/no such file/i,"Passing missing file terminates program");
  500. SKIP: {
  501. skip 'purposefully excluding this test from the recording', 1
  502. if $harness->record;
  503. like( exception { $driver->upload_file(__FILE__) },qr/501/,"Passing this file rightly fails due to mock not being present");
  504. }
  505. }
  506. ERROR: {
  507. # driver behaviour on error
  508. $driver->error_handler(sub {my ($self,$error_msg) = @_; croak("Got message: $error_msg");});
  509. like( exception { $driver->accept_alert }, qr/^Got message:/, "Error handler catches correctly an error");
  510. $driver->clear_error_handler;
  511. unlike( exception { $driver->accept_alert }, qr/^Got message:/, "Error handler was correctly cleared");
  512. like( exception { $driver->error_handler( 'hello' ) }, qr/must be a code ref/, 'we only accept code refs as error handlers');
  513. }
  514. QUIT: {
  515. $ret = $driver->quit();
  516. ok((not defined $driver->{'session_id'}), 'Killed the remote session');
  517. }
  518. COERCION: {
  519. my $string = 'string';
  520. like( exception { Selenium::Remote::Driver::_coerce_number( $string ) },
  521. qr/Expecting a number/,
  522. 'Can coerce numbers'
  523. );
  524. like( exception { Selenium::Remote::Driver::_coerce_timeout_ms() },
  525. qr/Expecting a timeout/,
  526. 'Can coerce missing timeouts'
  527. );
  528. like( exception { Selenium::Remote::Driver::_coerce_timeout_ms( $string ) },
  529. qr/Expecting a number/,
  530. 'Can coerce non-numeric timeouts'
  531. );
  532. }
  533. done_testing;