capybara 2.18.0 → 3.38.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (315) hide show
  1. checksums.yaml +4 -4
  2. data/.yardopts +1 -1
  3. data/History.md +889 -12
  4. data/License.txt +1 -1
  5. data/README.md +108 -78
  6. data/lib/capybara/config.rb +29 -57
  7. data/lib/capybara/cucumber.rb +2 -3
  8. data/lib/capybara/driver/base.rb +35 -18
  9. data/lib/capybara/driver/node.rb +40 -10
  10. data/lib/capybara/dsl.rb +10 -7
  11. data/lib/capybara/helpers.rb +70 -31
  12. data/lib/capybara/minitest/spec.rb +173 -83
  13. data/lib/capybara/minitest.rb +219 -112
  14. data/lib/capybara/node/actions.rb +274 -171
  15. data/lib/capybara/node/base.rb +42 -34
  16. data/lib/capybara/node/document.rb +15 -3
  17. data/lib/capybara/node/document_matchers.rb +19 -21
  18. data/lib/capybara/node/element.rb +362 -135
  19. data/lib/capybara/node/finders.rb +149 -137
  20. data/lib/capybara/node/matchers.rb +369 -209
  21. data/lib/capybara/node/simple.rb +59 -26
  22. data/lib/capybara/queries/active_element_query.rb +18 -0
  23. data/lib/capybara/queries/ancestor_query.rb +12 -9
  24. data/lib/capybara/queries/base_query.rb +39 -28
  25. data/lib/capybara/queries/current_path_query.rb +21 -27
  26. data/lib/capybara/queries/match_query.rb +14 -7
  27. data/lib/capybara/queries/selector_query.rb +658 -149
  28. data/lib/capybara/queries/sibling_query.rb +11 -9
  29. data/lib/capybara/queries/style_query.rb +45 -0
  30. data/lib/capybara/queries/text_query.rb +56 -38
  31. data/lib/capybara/queries/title_query.rb +8 -11
  32. data/lib/capybara/rack_test/browser.rb +113 -42
  33. data/lib/capybara/rack_test/css_handlers.rb +6 -4
  34. data/lib/capybara/rack_test/driver.rb +22 -17
  35. data/lib/capybara/rack_test/errors.rb +6 -0
  36. data/lib/capybara/rack_test/form.rb +93 -58
  37. data/lib/capybara/rack_test/node.rb +188 -81
  38. data/lib/capybara/rails.rb +3 -7
  39. data/lib/capybara/registration_container.rb +41 -0
  40. data/lib/capybara/registrations/drivers.rb +42 -0
  41. data/lib/capybara/registrations/patches/puma_ssl.rb +29 -0
  42. data/lib/capybara/registrations/servers.rb +53 -0
  43. data/lib/capybara/result.rb +96 -62
  44. data/lib/capybara/rspec/features.rb +17 -50
  45. data/lib/capybara/rspec/matcher_proxies.rb +52 -15
  46. data/lib/capybara/rspec/matchers/base.rb +111 -0
  47. data/lib/capybara/rspec/matchers/become_closed.rb +33 -0
  48. data/lib/capybara/rspec/matchers/compound.rb +88 -0
  49. data/lib/capybara/rspec/matchers/count_sugar.rb +37 -0
  50. data/lib/capybara/rspec/matchers/have_ancestor.rb +28 -0
  51. data/lib/capybara/rspec/matchers/have_current_path.rb +29 -0
  52. data/lib/capybara/rspec/matchers/have_selector.rb +77 -0
  53. data/lib/capybara/rspec/matchers/have_sibling.rb +27 -0
  54. data/lib/capybara/rspec/matchers/have_text.rb +33 -0
  55. data/lib/capybara/rspec/matchers/have_title.rb +29 -0
  56. data/lib/capybara/rspec/matchers/match_selector.rb +27 -0
  57. data/lib/capybara/rspec/matchers/match_style.rb +43 -0
  58. data/lib/capybara/rspec/matchers/spatial_sugar.rb +39 -0
  59. data/lib/capybara/rspec/matchers.rb +142 -311
  60. data/lib/capybara/rspec.rb +7 -11
  61. data/lib/capybara/selector/builders/css_builder.rb +84 -0
  62. data/lib/capybara/selector/builders/xpath_builder.rb +71 -0
  63. data/lib/capybara/selector/css.rb +89 -17
  64. data/lib/capybara/selector/definition/button.rb +68 -0
  65. data/lib/capybara/selector/definition/checkbox.rb +26 -0
  66. data/lib/capybara/selector/definition/css.rb +10 -0
  67. data/lib/capybara/selector/definition/datalist_input.rb +35 -0
  68. data/lib/capybara/selector/definition/datalist_option.rb +25 -0
  69. data/lib/capybara/selector/definition/element.rb +28 -0
  70. data/lib/capybara/selector/definition/field.rb +40 -0
  71. data/lib/capybara/selector/definition/fieldset.rb +14 -0
  72. data/lib/capybara/selector/definition/file_field.rb +13 -0
  73. data/lib/capybara/selector/definition/fillable_field.rb +33 -0
  74. data/lib/capybara/selector/definition/frame.rb +17 -0
  75. data/lib/capybara/selector/definition/id.rb +6 -0
  76. data/lib/capybara/selector/definition/label.rb +62 -0
  77. data/lib/capybara/selector/definition/link.rb +54 -0
  78. data/lib/capybara/selector/definition/link_or_button.rb +16 -0
  79. data/lib/capybara/selector/definition/option.rb +27 -0
  80. data/lib/capybara/selector/definition/radio_button.rb +27 -0
  81. data/lib/capybara/selector/definition/select.rb +81 -0
  82. data/lib/capybara/selector/definition/table.rb +109 -0
  83. data/lib/capybara/selector/definition/table_row.rb +21 -0
  84. data/lib/capybara/selector/definition/xpath.rb +5 -0
  85. data/lib/capybara/selector/definition.rb +280 -0
  86. data/lib/capybara/selector/filter.rb +2 -17
  87. data/lib/capybara/selector/filter_set.rb +80 -33
  88. data/lib/capybara/selector/filters/base.rb +50 -6
  89. data/lib/capybara/selector/filters/expression_filter.rb +8 -26
  90. data/lib/capybara/selector/filters/locator_filter.rb +29 -0
  91. data/lib/capybara/selector/filters/node_filter.rb +16 -12
  92. data/lib/capybara/selector/regexp_disassembler.rb +211 -0
  93. data/lib/capybara/selector/selector.rb +93 -210
  94. data/lib/capybara/selector/xpath_extensions.rb +17 -0
  95. data/lib/capybara/selector.rb +227 -526
  96. data/lib/capybara/selenium/atoms/getAttribute.min.js +1 -0
  97. data/lib/capybara/selenium/atoms/isDisplayed.min.js +1 -0
  98. data/lib/capybara/selenium/atoms/src/getAttribute.js +161 -0
  99. data/lib/capybara/selenium/atoms/src/isDisplayed.js +454 -0
  100. data/lib/capybara/selenium/driver.rb +332 -261
  101. data/lib/capybara/selenium/driver_specializations/chrome_driver.rb +117 -0
  102. data/lib/capybara/selenium/driver_specializations/edge_driver.rb +124 -0
  103. data/lib/capybara/selenium/driver_specializations/firefox_driver.rb +89 -0
  104. data/lib/capybara/selenium/driver_specializations/internet_explorer_driver.rb +26 -0
  105. data/lib/capybara/selenium/driver_specializations/safari_driver.rb +24 -0
  106. data/lib/capybara/selenium/extensions/file_input_click_emulation.rb +34 -0
  107. data/lib/capybara/selenium/extensions/find.rb +110 -0
  108. data/lib/capybara/selenium/extensions/html5_drag.rb +226 -0
  109. data/lib/capybara/selenium/extensions/modifier_keys_stack.rb +28 -0
  110. data/lib/capybara/selenium/extensions/scroll.rb +76 -0
  111. data/lib/capybara/selenium/logger_suppressor.rb +44 -0
  112. data/lib/capybara/selenium/node.rb +545 -144
  113. data/lib/capybara/selenium/nodes/chrome_node.rb +137 -0
  114. data/lib/capybara/selenium/nodes/edge_node.rb +104 -0
  115. data/lib/capybara/selenium/nodes/firefox_node.rb +136 -0
  116. data/lib/capybara/selenium/nodes/ie_node.rb +22 -0
  117. data/lib/capybara/selenium/nodes/safari_node.rb +118 -0
  118. data/lib/capybara/selenium/patches/action_pauser.rb +26 -0
  119. data/lib/capybara/selenium/patches/atoms.rb +18 -0
  120. data/lib/capybara/selenium/patches/is_displayed.rb +16 -0
  121. data/lib/capybara/selenium/patches/logs.rb +45 -0
  122. data/lib/capybara/selenium/patches/pause_duration_fix.rb +9 -0
  123. data/lib/capybara/selenium/patches/persistent_client.rb +20 -0
  124. data/lib/capybara/server/animation_disabler.rb +81 -0
  125. data/lib/capybara/server/checker.rb +44 -0
  126. data/lib/capybara/server/middleware.rb +71 -0
  127. data/lib/capybara/server.rb +59 -67
  128. data/lib/capybara/session/config.rb +81 -67
  129. data/lib/capybara/session/matchers.rb +28 -20
  130. data/lib/capybara/session.rb +336 -365
  131. data/lib/capybara/spec/public/jquery.js +5 -5
  132. data/lib/capybara/spec/public/offset.js +6 -0
  133. data/lib/capybara/spec/public/test.js +147 -12
  134. data/lib/capybara/spec/session/accept_alert_spec.rb +12 -11
  135. data/lib/capybara/spec/session/accept_confirm_spec.rb +6 -5
  136. data/lib/capybara/spec/session/accept_prompt_spec.rb +10 -10
  137. data/lib/capybara/spec/session/active_element_spec.rb +31 -0
  138. data/lib/capybara/spec/session/all_spec.rb +160 -56
  139. data/lib/capybara/spec/session/ancestor_spec.rb +27 -24
  140. data/lib/capybara/spec/session/assert_all_of_selectors_spec.rb +68 -38
  141. data/lib/capybara/spec/session/assert_current_path_spec.rb +75 -0
  142. data/lib/capybara/spec/session/assert_selector_spec.rb +143 -0
  143. data/lib/capybara/spec/session/assert_style_spec.rb +26 -0
  144. data/lib/capybara/spec/session/assert_text_spec.rb +258 -0
  145. data/lib/capybara/spec/session/{assert_title.rb → assert_title_spec.rb} +22 -12
  146. data/lib/capybara/spec/session/attach_file_spec.rb +144 -69
  147. data/lib/capybara/spec/session/body_spec.rb +12 -13
  148. data/lib/capybara/spec/session/check_spec.rb +117 -55
  149. data/lib/capybara/spec/session/choose_spec.rb +64 -31
  150. data/lib/capybara/spec/session/click_button_spec.rb +231 -173
  151. data/lib/capybara/spec/session/click_link_or_button_spec.rb +55 -35
  152. data/lib/capybara/spec/session/click_link_spec.rb +82 -58
  153. data/lib/capybara/spec/session/current_scope_spec.rb +12 -11
  154. data/lib/capybara/spec/session/current_url_spec.rb +57 -39
  155. data/lib/capybara/spec/session/dismiss_confirm_spec.rb +4 -4
  156. data/lib/capybara/spec/session/dismiss_prompt_spec.rb +3 -2
  157. data/lib/capybara/spec/session/element/{assert_match_selector.rb → assert_match_selector_spec.rb} +11 -9
  158. data/lib/capybara/spec/session/element/match_css_spec.rb +18 -10
  159. data/lib/capybara/spec/session/element/match_xpath_spec.rb +9 -7
  160. data/lib/capybara/spec/session/element/matches_selector_spec.rb +71 -57
  161. data/lib/capybara/spec/session/evaluate_async_script_spec.rb +8 -7
  162. data/lib/capybara/spec/session/evaluate_script_spec.rb +29 -8
  163. data/lib/capybara/spec/session/execute_script_spec.rb +10 -8
  164. data/lib/capybara/spec/session/fill_in_spec.rb +134 -43
  165. data/lib/capybara/spec/session/find_button_spec.rb +25 -24
  166. data/lib/capybara/spec/session/find_by_id_spec.rb +10 -9
  167. data/lib/capybara/spec/session/find_field_spec.rb +37 -41
  168. data/lib/capybara/spec/session/find_link_spec.rb +36 -17
  169. data/lib/capybara/spec/session/find_spec.rb +251 -144
  170. data/lib/capybara/spec/session/first_spec.rb +79 -51
  171. data/lib/capybara/spec/session/frame/frame_title_spec.rb +23 -0
  172. data/lib/capybara/spec/session/frame/frame_url_spec.rb +23 -0
  173. data/lib/capybara/spec/session/frame/switch_to_frame_spec.rb +33 -20
  174. data/lib/capybara/spec/session/frame/within_frame_spec.rb +50 -32
  175. data/lib/capybara/spec/session/go_back_spec.rb +2 -1
  176. data/lib/capybara/spec/session/go_forward_spec.rb +2 -1
  177. data/lib/capybara/spec/session/has_all_selectors_spec.rb +31 -31
  178. data/lib/capybara/spec/session/has_ancestor_spec.rb +46 -0
  179. data/lib/capybara/spec/session/has_any_selectors_spec.rb +29 -0
  180. data/lib/capybara/spec/session/has_button_spec.rb +100 -13
  181. data/lib/capybara/spec/session/has_css_spec.rb +272 -137
  182. data/lib/capybara/spec/session/has_current_path_spec.rb +59 -60
  183. data/lib/capybara/spec/session/has_field_spec.rb +139 -59
  184. data/lib/capybara/spec/session/has_link_spec.rb +43 -6
  185. data/lib/capybara/spec/session/has_none_selectors_spec.rb +42 -40
  186. data/lib/capybara/spec/session/has_select_spec.rb +107 -72
  187. data/lib/capybara/spec/session/has_selector_spec.rb +120 -71
  188. data/lib/capybara/spec/session/has_sibling_spec.rb +50 -0
  189. data/lib/capybara/spec/session/has_table_spec.rb +172 -5
  190. data/lib/capybara/spec/session/has_text_spec.rb +106 -62
  191. data/lib/capybara/spec/session/has_title_spec.rb +20 -14
  192. data/lib/capybara/spec/session/has_xpath_spec.rb +57 -38
  193. data/lib/capybara/spec/session/{headers.rb → headers_spec.rb} +3 -2
  194. data/lib/capybara/spec/session/html_spec.rb +14 -6
  195. data/lib/capybara/spec/session/matches_style_spec.rb +35 -0
  196. data/lib/capybara/spec/session/node_spec.rb +1002 -153
  197. data/lib/capybara/spec/session/node_wrapper_spec.rb +39 -0
  198. data/lib/capybara/spec/session/refresh_spec.rb +12 -6
  199. data/lib/capybara/spec/session/reset_session_spec.rb +82 -35
  200. data/lib/capybara/spec/session/{response_code.rb → response_code_spec.rb} +2 -1
  201. data/lib/capybara/spec/session/save_and_open_page_spec.rb +3 -2
  202. data/lib/capybara/spec/session/save_and_open_screenshot_spec.rb +8 -12
  203. data/lib/capybara/spec/session/save_page_spec.rb +42 -55
  204. data/lib/capybara/spec/session/save_screenshot_spec.rb +16 -14
  205. data/lib/capybara/spec/session/screenshot_spec.rb +2 -2
  206. data/lib/capybara/spec/session/scroll_spec.rb +117 -0
  207. data/lib/capybara/spec/session/select_spec.rb +107 -81
  208. data/lib/capybara/spec/session/selectors_spec.rb +52 -19
  209. data/lib/capybara/spec/session/sibling_spec.rb +10 -10
  210. data/lib/capybara/spec/session/text_spec.rb +37 -21
  211. data/lib/capybara/spec/session/title_spec.rb +17 -5
  212. data/lib/capybara/spec/session/uncheck_spec.rb +42 -22
  213. data/lib/capybara/spec/session/unselect_spec.rb +39 -38
  214. data/lib/capybara/spec/session/visit_spec.rb +85 -53
  215. data/lib/capybara/spec/session/window/become_closed_spec.rb +24 -20
  216. data/lib/capybara/spec/session/window/current_window_spec.rb +5 -3
  217. data/lib/capybara/spec/session/window/open_new_window_spec.rb +5 -3
  218. data/lib/capybara/spec/session/window/switch_to_window_spec.rb +27 -22
  219. data/lib/capybara/spec/session/window/window_opened_by_spec.rb +12 -6
  220. data/lib/capybara/spec/session/window/window_spec.rb +97 -63
  221. data/lib/capybara/spec/session/window/windows_spec.rb +12 -10
  222. data/lib/capybara/spec/session/window/within_window_spec.rb +31 -86
  223. data/lib/capybara/spec/session/within_spec.rb +83 -44
  224. data/lib/capybara/spec/spec_helper.rb +53 -43
  225. data/lib/capybara/spec/test_app.rb +158 -43
  226. data/lib/capybara/spec/views/animated.erb +49 -0
  227. data/lib/capybara/spec/views/form.erb +154 -42
  228. data/lib/capybara/spec/views/frame_child.erb +4 -3
  229. data/lib/capybara/spec/views/frame_one.erb +2 -1
  230. data/lib/capybara/spec/views/frame_parent.erb +1 -1
  231. data/lib/capybara/spec/views/frame_two.erb +1 -1
  232. data/lib/capybara/spec/views/initial_alert.erb +2 -1
  233. data/lib/capybara/spec/views/layout.erb +10 -0
  234. data/lib/capybara/spec/views/obscured.erb +47 -0
  235. data/lib/capybara/spec/views/offset.erb +33 -0
  236. data/lib/capybara/spec/views/path.erb +2 -2
  237. data/lib/capybara/spec/views/popup_one.erb +1 -1
  238. data/lib/capybara/spec/views/popup_two.erb +1 -1
  239. data/lib/capybara/spec/views/react.erb +45 -0
  240. data/lib/capybara/spec/views/scroll.erb +21 -0
  241. data/lib/capybara/spec/views/spatial.erb +31 -0
  242. data/lib/capybara/spec/views/tables.erb +68 -1
  243. data/lib/capybara/spec/views/with_animation.erb +81 -0
  244. data/lib/capybara/spec/views/with_base_tag.erb +2 -2
  245. data/lib/capybara/spec/views/with_dragula.erb +24 -0
  246. data/lib/capybara/spec/views/with_fixed_header_footer.erb +2 -1
  247. data/lib/capybara/spec/views/with_hover.erb +3 -2
  248. data/lib/capybara/spec/views/with_hover1.erb +10 -0
  249. data/lib/capybara/spec/views/with_html.erb +67 -12
  250. data/lib/capybara/spec/views/with_html5_svg.erb +20 -0
  251. data/lib/capybara/spec/views/with_jquery_animation.erb +24 -0
  252. data/lib/capybara/spec/views/with_js.erb +30 -5
  253. data/lib/capybara/spec/views/with_jstree.erb +26 -0
  254. data/lib/capybara/spec/views/with_namespace.erb +21 -0
  255. data/lib/capybara/spec/views/with_scope.erb +2 -2
  256. data/lib/capybara/spec/views/with_scope_other.erb +6 -0
  257. data/lib/capybara/spec/views/with_shadow.erb +31 -0
  258. data/lib/capybara/spec/views/with_slow_unload.erb +2 -1
  259. data/lib/capybara/spec/views/with_sortable_js.erb +21 -0
  260. data/lib/capybara/spec/views/with_unload_alert.erb +1 -0
  261. data/lib/capybara/spec/views/with_windows.erb +1 -1
  262. data/lib/capybara/spec/views/within_frames.erb +5 -2
  263. data/lib/capybara/version.rb +2 -1
  264. data/lib/capybara/window.rb +36 -34
  265. data/lib/capybara.rb +129 -103
  266. data/spec/basic_node_spec.rb +60 -34
  267. data/spec/capybara_spec.rb +63 -88
  268. data/spec/counter_spec.rb +35 -0
  269. data/spec/css_builder_spec.rb +101 -0
  270. data/spec/css_splitter_spec.rb +38 -0
  271. data/spec/dsl_spec.rb +85 -64
  272. data/spec/filter_set_spec.rb +27 -9
  273. data/spec/fixtures/certificate.pem +25 -0
  274. data/spec/fixtures/key.pem +27 -0
  275. data/spec/fixtures/selenium_driver_rspec_failure.rb +6 -5
  276. data/spec/fixtures/selenium_driver_rspec_success.rb +6 -5
  277. data/spec/minitest_spec.rb +45 -7
  278. data/spec/minitest_spec_spec.rb +94 -63
  279. data/spec/per_session_config_spec.rb +14 -13
  280. data/spec/rack_test_spec.rb +194 -125
  281. data/spec/regexp_dissassembler_spec.rb +250 -0
  282. data/spec/result_spec.rb +102 -50
  283. data/spec/rspec/features_spec.rb +37 -31
  284. data/spec/rspec/scenarios_spec.rb +9 -7
  285. data/spec/rspec/shared_spec_matchers.rb +449 -422
  286. data/spec/rspec/views_spec.rb +5 -3
  287. data/spec/rspec_matchers_spec.rb +27 -11
  288. data/spec/rspec_spec.rb +109 -89
  289. data/spec/sauce_spec_chrome.rb +43 -0
  290. data/spec/selector_spec.rb +397 -68
  291. data/spec/selenium_spec_chrome.rb +186 -40
  292. data/spec/selenium_spec_chrome_remote.rb +103 -0
  293. data/spec/selenium_spec_edge.rb +49 -0
  294. data/spec/selenium_spec_firefox.rb +194 -41
  295. data/spec/selenium_spec_firefox_remote.rb +82 -0
  296. data/spec/selenium_spec_ie.rb +149 -0
  297. data/spec/selenium_spec_safari.rb +162 -0
  298. data/spec/server_spec.rb +201 -102
  299. data/spec/session_spec.rb +53 -16
  300. data/spec/shared_selenium_node.rb +79 -0
  301. data/spec/shared_selenium_session.rb +474 -122
  302. data/spec/spec_helper.rb +93 -7
  303. data/spec/xpath_builder_spec.rb +93 -0
  304. metadata +360 -73
  305. data/.yard/templates_custom/default/class/html/selectors.erb +0 -38
  306. data/.yard/templates_custom/default/class/html/setup.rb +0 -17
  307. data/.yard/yard_extensions.rb +0 -78
  308. data/lib/capybara/query.rb +0 -7
  309. data/lib/capybara/rspec/compound.rb +0 -95
  310. data/lib/capybara/spec/session/assert_current_path.rb +0 -72
  311. data/lib/capybara/spec/session/assert_selector.rb +0 -148
  312. data/lib/capybara/spec/session/assert_text.rb +0 -234
  313. data/lib/capybara/spec/session/source_spec.rb +0 -0
  314. data/lib/capybara/spec/views/with_title.erb +0 -5
  315. data/spec/selenium_spec_marionette.rb +0 -127
@@ -1,48 +1,103 @@
1
1
  # frozen_string_literal: true
2
- require "uri"
2
+
3
+ require 'uri'
4
+ require 'English'
3
5
 
4
6
  class Capybara::Selenium::Driver < Capybara::Driver::Base
7
+ include Capybara::Selenium::Find
5
8
 
6
9
  DEFAULT_OPTIONS = {
7
- :browser => :firefox,
8
- clear_local_storage: false,
9
- clear_session_storage: false
10
- }
11
- SPECIAL_OPTIONS = [:browser, :clear_local_storage, :clear_session_storage]
10
+ browser: :firefox,
11
+ clear_local_storage: nil,
12
+ clear_session_storage: nil
13
+ }.freeze
14
+ SPECIAL_OPTIONS = %i[browser clear_local_storage clear_session_storage timeout native_displayed].freeze
15
+ CAPS_VERSION = Gem::Requirement.new('>= 4.0.0.alpha6')
12
16
 
13
17
  attr_reader :app, :options
14
18
 
19
+ class << self
20
+ attr_reader :selenium_webdriver_version
21
+
22
+ def load_selenium
23
+ require 'selenium-webdriver'
24
+ require 'capybara/selenium/logger_suppressor'
25
+ require 'capybara/selenium/patches/atoms'
26
+ require 'capybara/selenium/patches/is_displayed'
27
+ require 'capybara/selenium/patches/action_pauser'
28
+
29
+ # Look up the version of `selenium-webdriver` to
30
+ # see if it's a version we support.
31
+ #
32
+ # By default, we use Gem.loaded_specs to determine
33
+ # the version number. However, in some cases, such
34
+ # as when loading `selenium-webdriver` outside of
35
+ # Rubygems, we fall back to referencing
36
+ # Selenium::WebDriver::VERSION. Ideally we'd
37
+ # use the constant in all cases, but earlier versions
38
+ # of `selenium-webdriver` didn't provide the constant.
39
+ @selenium_webdriver_version =
40
+ if Gem.loaded_specs['selenium-webdriver']
41
+ Gem.loaded_specs['selenium-webdriver'].version
42
+ else
43
+ Gem::Version.new(Selenium::WebDriver::VERSION)
44
+ end
45
+
46
+ unless Gem::Requirement.new('>= 3.142.7').satisfied_by? @selenium_webdriver_version
47
+ warn "Warning: You're using an unsupported version of selenium-webdriver, please upgrade."
48
+ end
49
+
50
+ @selenium_webdriver_version
51
+ rescue LoadError => e
52
+ raise e unless e.message.include?('selenium-webdriver')
53
+
54
+ raise LoadError, "Capybara's selenium driver is unable to load `selenium-webdriver`, please install the gem and add `gem 'selenium-webdriver'` to your Gemfile if you are using bundler."
55
+ end
56
+
57
+ attr_reader :specializations
58
+
59
+ def register_specialization(browser_name, specialization)
60
+ @specializations ||= {}
61
+ @specializations[browser_name] = specialization
62
+ end
63
+ end
64
+
15
65
  def browser
16
66
  unless @browser
17
- if firefox?
18
- options[:desired_capabilities] ||= {}
19
- options[:desired_capabilities].merge!({ unexpectedAlertBehaviour: "ignore" })
67
+ options[:http_client] ||= begin
68
+ require 'capybara/selenium/patches/persistent_client'
69
+ if options[:timeout]
70
+ ::Capybara::Selenium::PersistentClient.new(read_timeout: options[:timeout])
71
+ else
72
+ ::Capybara::Selenium::PersistentClient.new
73
+ end
20
74
  end
75
+ processed_options = options.reject { |key, _val| SPECIAL_OPTIONS.include?(key) }
21
76
 
22
- @processed_options = options.reject { |key,_val| SPECIAL_OPTIONS.include?(key) }
23
- @browser = Selenium::WebDriver.for(options[:browser], @processed_options)
24
-
25
- @w3c = ((defined?(Selenium::WebDriver::Remote::W3CCapabilities) && @browser.capabilities.is_a?(Selenium::WebDriver::Remote::W3CCapabilities)) ||
26
- (defined?(Selenium::WebDriver::Remote::W3C::Capabilities) && @browser.capabilities.is_a?(Selenium::WebDriver::Remote::W3C::Capabilities)))
27
- main = Process.pid
28
- at_exit do
29
- # Store the exit status of the test run since it goes away after calling the at_exit proc...
30
- @exit_status = $!.status if $!.is_a?(SystemExit)
31
- quit if Process.pid == main
32
- exit @exit_status if @exit_status # Force exit with stored status
77
+ @browser = if options[:browser] == :firefox &&
78
+ RUBY_VERSION >= '3.0' &&
79
+ Capybara::Selenium::Driver.selenium_webdriver_version <= Gem::Version.new('4.0.0.alpha1')
80
+ # selenium-webdriver 3.x doesn't correctly pass options through for Firefox with Ruby 3 so workaround that
81
+ Selenium::WebDriver::Firefox::Driver.new(**processed_options)
82
+ else
83
+ Selenium::WebDriver.for(options[:browser], processed_options)
33
84
  end
85
+
86
+ specialize_driver
87
+ setup_exit_handler
34
88
  end
35
89
  @browser
36
90
  end
37
91
 
38
- def initialize(app, options={})
39
- load_selenium
40
- @session = nil
92
+ def initialize(app, **options)
93
+ super()
94
+ self.class.load_selenium
41
95
  @app = app
42
96
  @browser = nil
43
97
  @exit_status = nil
44
- @frame_handles = {}
98
+ @frame_handles = Hash.new { |hash, handle| hash[handle] = [] }
45
99
  @options = DEFAULT_OPTIONS.merge(options)
100
+ @node_class = ::Capybara::Selenium::Node
46
101
  end
47
102
 
48
103
  def visit(path)
@@ -50,10 +105,7 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
50
105
  end
51
106
 
52
107
  def refresh
53
- accept_modal(nil, wait: 0.1) do
54
- browser.navigate.refresh
55
- end
56
- rescue Capybara::ModalNotFound
108
+ browser.navigate.refresh
57
109
  end
58
110
 
59
111
  def go_back
@@ -66,6 +118,8 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
66
118
 
67
119
  def html
68
120
  browser.page_source
121
+ rescue Selenium::WebDriver::Error::JavascriptError => e
122
+ raise unless e.message.include?('documentElement is null')
69
123
  end
70
124
 
71
125
  def title
@@ -76,19 +130,11 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
76
130
  browser.current_url
77
131
  end
78
132
 
79
- def find_xpath(selector)
80
- browser.find_elements(:xpath, selector).map { |node| Capybara::Selenium::Node.new(self, node) }
81
- end
82
-
83
- def find_css(selector)
84
- browser.find_elements(:css, selector).map { |node| Capybara::Selenium::Node.new(self, node) }
85
- end
86
-
87
133
  def wait?; true; end
88
134
  def needs_server?; true; end
89
135
 
90
136
  def execute_script(script, *args)
91
- browser.execute_script(script, *args.map { |arg| arg.is_a?(Capybara::Selenium::Node) ? arg.native : arg} )
137
+ browser.execute_script(script, *native_args(args))
92
138
  end
93
139
 
94
140
  def evaluate_script(script, *args)
@@ -98,83 +144,69 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
98
144
 
99
145
  def evaluate_async_script(script, *args)
100
146
  browser.manage.timeouts.script_timeout = Capybara.default_max_wait_time
101
- result = browser.execute_async_script(script, *args.map { |arg| arg.is_a?(Capybara::Selenium::Node) ? arg.native : arg} )
147
+ result = browser.execute_async_script(script, *native_args(args))
102
148
  unwrap_script_result(result)
103
149
  end
104
150
 
105
- def save_screenshot(path, _options={})
151
+ def active_element
152
+ build_node(native_active_element)
153
+ end
154
+
155
+ def send_keys(*args)
156
+ # Should this call the specialized nodes rather than native???
157
+ native_active_element.send_keys(*args)
158
+ end
159
+
160
+ def save_screenshot(path, **_options)
106
161
  browser.save_screenshot(path)
107
162
  end
108
163
 
109
164
  def reset!
110
165
  # Use instance variable directly so we avoid starting the browser just to reset the session
111
- if @browser
112
- navigated = false
113
- start_time = Capybara::Helpers.monotonic_time
114
- begin
115
- if !navigated
116
- # Only trigger a navigation if we haven't done it already, otherwise it
117
- # can trigger an endless series of unload modals
118
- begin
119
- @browser.manage.delete_all_cookies
120
- if options[:clear_session_storage]
121
- if @browser.respond_to? :session_storage
122
- @browser.session_storage.clear
123
- else
124
- warn "sessionStorage clear requested but is not available for this driver"
125
- end
126
- end
127
- if options[:clear_local_storage]
128
- if @browser.respond_to? :local_storage
129
- @browser.local_storage.clear
130
- else
131
- warn "localStorage clear requested but is not available for this driver"
132
- end
133
- end
134
- rescue Selenium::WebDriver::Error::UnhandledError
135
- # delete_all_cookies fails when we've previously gone
136
- # to about:blank, so we rescue this error and do nothing
137
- # instead.
138
- end
139
- @browser.navigate.to("about:blank")
140
- end
141
- navigated = true
166
+ return unless @browser
142
167
 
143
- #Ensure the page is empty and trigger an UnhandledAlertError for any modals that appear during unload
144
- until find_xpath("/html/body/*").empty? do
145
- raise Capybara::ExpectationNotMet.new('Timed out waiting for Selenium session reset') if (Capybara::Helpers.monotonic_time - start_time) >= 10
146
- sleep 0.05
147
- end
148
- rescue Selenium::WebDriver::Error::UnhandledAlertError, Selenium::WebDriver::Error::UnexpectedAlertOpenError
149
- # This error is thrown if an unhandled alert is on the page
150
- # Firefox appears to automatically dismiss this alert, chrome does not
151
- # We'll try to accept it
152
- begin
153
- @browser.switch_to.alert.accept
154
- sleep 0.25 # allow time for the modal to be handled
155
- rescue modal_error
156
- # The alert is now gone - nothing to do
157
- end
158
- # try cleaning up the browser again
159
- retry
160
- end
168
+ navigated = false
169
+ timer = Capybara::Helpers.timer(expire_in: 10)
170
+ begin
171
+ # Only trigger a navigation if we haven't done it already, otherwise it
172
+ # can trigger an endless series of unload modals
173
+ reset_browser_state unless navigated
174
+ navigated = true
175
+ # Ensure the page is empty and trigger an UnhandledAlertError for any modals that appear during unload
176
+ wait_for_empty_page(timer)
177
+ rescue *unhandled_alert_errors
178
+ # This error is thrown if an unhandled alert is on the page
179
+ # Firefox appears to automatically dismiss this alert, chrome does not
180
+ # We'll try to accept it
181
+ accept_unhandled_reset_alert
182
+ # try cleaning up the browser again
183
+ retry
184
+ end
185
+ end
186
+
187
+ def frame_obscured_at?(x:, y:)
188
+ frame = @frame_handles[current_window_handle].last
189
+ return false unless frame
190
+
191
+ switch_to_frame(:parent)
192
+ begin
193
+ frame.base.obscured?(x: x, y: y)
194
+ ensure
195
+ switch_to_frame(frame)
161
196
  end
162
197
  end
163
198
 
164
199
  def switch_to_frame(frame)
200
+ handles = @frame_handles[current_window_handle]
165
201
  case frame
166
202
  when :top
167
- @frame_handles[browser.window_handle] = []
203
+ handles.clear
168
204
  browser.switch_to.default_content
169
205
  when :parent
170
- # would love to use browser.switch_to.parent_frame here
171
- # but it has an issue if the current frame is removed from within it
172
- @frame_handles[browser.window_handle].pop
173
- browser.switch_to.default_content
174
- @frame_handles[browser.window_handle].each { |fh| browser.switch_to.frame(fh) }
206
+ handles.pop
207
+ browser.switch_to.parent_frame
175
208
  else
176
- @frame_handles[browser.window_handle] ||= []
177
- @frame_handles[browser.window_handle] << frame.native
209
+ handles << frame
178
210
  browser.switch_to.frame(frame.native)
179
211
  end
180
212
  end
@@ -192,12 +224,7 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
192
224
 
193
225
  def resize_window_to(handle, width, height)
194
226
  within_given_window(handle) do
195
- # Don't set the size if already set - See https://github.com/mozilla/geckodriver/issues/643
196
- if marionette? && (window_size(handle) == [width, height])
197
- {}
198
- else
199
- browser.manage.window.resize_to(width, height)
200
- end
227
+ browser.manage.window.resize_to(width, height)
201
228
  end
202
229
  end
203
230
 
@@ -208,7 +235,15 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
208
235
  sleep 0.1 # work around for https://code.google.com/p/selenium/issues/detail?id=7405
209
236
  end
210
237
 
238
+ def fullscreen_window(handle)
239
+ within_given_window(handle) do
240
+ browser.manage.window.full_screen
241
+ end
242
+ end
243
+
211
244
  def close_window(handle)
245
+ raise ArgumentError, 'Not allowed to close the primary window' if handle == window_handles.first
246
+
212
247
  within_given_window(handle) do
213
248
  browser.close
214
249
  end
@@ -218,7 +253,16 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
218
253
  browser.window_handles
219
254
  end
220
255
 
221
- def open_new_window
256
+ def open_new_window(kind = :tab)
257
+ if browser.switch_to.respond_to?(:new_window)
258
+ handle = current_window_handle
259
+ browser.switch_to.new_window(kind)
260
+ switch_to_window(handle)
261
+ else
262
+ browser.manage.new_window(kind)
263
+ end
264
+ rescue NoMethodError, Selenium::WebDriver::Error::WebDriverError
265
+ # If not supported by the driver or browser default to using JS
222
266
  browser.execute_script('window.open();')
223
267
  end
224
268
 
@@ -226,14 +270,9 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
226
270
  browser.switch_to.window handle
227
271
  end
228
272
 
229
- def within_window(locator)
230
- handle = find_window(locator)
231
- browser.switch_to.window(handle) { yield }
232
- end
233
-
234
- def accept_modal(_type, options={})
273
+ def accept_modal(_type, **options)
235
274
  yield if block_given?
236
- modal = find_modal(options)
275
+ modal = find_modal(**options)
237
276
 
238
277
  modal.send_keys options[:with] if options[:with]
239
278
 
@@ -242,17 +281,18 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
242
281
  message
243
282
  end
244
283
 
245
- def dismiss_modal(_type, options={})
284
+ def dismiss_modal(_type, **options)
246
285
  yield if block_given?
247
- modal = find_modal(options)
286
+ modal = find_modal(**options)
248
287
  message = modal.text
249
288
  modal.dismiss
250
289
  message
251
290
  end
252
291
 
253
292
  def quit
254
- @browser.quit if @browser
255
- rescue Selenium::WebDriver::Error::SessionNotCreatedError, Errno::ECONNREFUSED
293
+ @browser&.quit
294
+ rescue Selenium::WebDriver::Error::SessionNotCreatedError, Errno::ECONNREFUSED,
295
+ Selenium::WebDriver::Error::InvalidSessionIdError
256
296
  # Browser must have already gone
257
297
  rescue Selenium::WebDriver::Error::UnknownError => e
258
298
  unless silenced_unknown_error_message?(e.message) # Most likely already gone
@@ -264,129 +304,122 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
264
304
  end
265
305
 
266
306
  def invalid_element_errors
267
- [::Selenium::WebDriver::Error::StaleElementReferenceError,
268
- ::Selenium::WebDriver::Error::UnhandledError,
269
- ::Selenium::WebDriver::Error::ElementNotVisibleError,
270
- ::Selenium::WebDriver::Error::InvalidSelectorError, # Work around a race condition that can occur with chromedriver and #go_back/#go_forward
271
- ::Selenium::WebDriver::Error::ElementNotInteractableError,
272
- ::Selenium::WebDriver::Error::ElementClickInterceptedError,
273
- ::Selenium::WebDriver::Error::InvalidElementStateError,
274
- ::Selenium::WebDriver::Error::ElementNotSelectableError,
275
- ]
307
+ @invalid_element_errors ||=
308
+ [
309
+ ::Selenium::WebDriver::Error::StaleElementReferenceError,
310
+ ::Selenium::WebDriver::Error::ElementNotInteractableError,
311
+ ::Selenium::WebDriver::Error::InvalidSelectorError, # Work around chromedriver go_back/go_forward race condition
312
+ ::Selenium::WebDriver::Error::ElementClickInterceptedError,
313
+ ::Selenium::WebDriver::Error::NoSuchElementError, # IE
314
+ ::Selenium::WebDriver::Error::InvalidArgumentError # IE
315
+ ].tap do |errors|
316
+ unless selenium_4?
317
+ ::Selenium::WebDriver.logger.suppress_deprecations do
318
+ errors.concat [
319
+ ::Selenium::WebDriver::Error::UnhandledError,
320
+ ::Selenium::WebDriver::Error::ElementNotVisibleError,
321
+ ::Selenium::WebDriver::Error::InvalidElementStateError,
322
+ ::Selenium::WebDriver::Error::ElementNotSelectableError
323
+ ]
324
+ end
325
+ end
326
+ if defined?(::Selenium::WebDriver::Error::DetachedShadowRootError)
327
+ errors.concat([::Selenium::WebDriver::Error::DetachedShadowRootError])
328
+ end
329
+ end
276
330
  end
277
331
 
278
332
  def no_such_window_error
279
333
  Selenium::WebDriver::Error::NoSuchWindowError
280
334
  end
281
335
 
282
- # @api private
283
- def marionette?
284
- firefox? && browser && @w3c
336
+ private
337
+
338
+ def selenium_4?
339
+ defined?(Selenium::WebDriver::VERSION) && (Selenium::WebDriver::VERSION.to_f >= 4)
285
340
  end
286
341
 
287
- # @api private
288
- def firefox?
289
- browser_name == "firefox"
342
+ def native_args(args)
343
+ args.map { |arg| arg.is_a?(Capybara::Selenium::Node) ? arg.native : arg }
290
344
  end
291
345
 
292
- # @api private
293
- def chrome?
294
- browser_name == "chrome"
346
+ def native_active_element
347
+ browser.switch_to.active_element
295
348
  end
296
349
 
297
- # @deprecated This method is being removed
298
- def browser_initialized?
299
- super && !@browser.nil?
350
+ def clear_browser_state
351
+ delete_all_cookies
352
+ clear_storage
353
+ rescue *clear_browser_state_errors
354
+ # delete_all_cookies fails when we've previously gone
355
+ # to about:blank, so we rescue this error and do nothing
356
+ # instead.
300
357
  end
301
358
 
302
- private
359
+ def clear_browser_state_errors
360
+ @clear_browser_state_errors ||= [Selenium::WebDriver::Error::UnknownError]
361
+ end
303
362
 
304
- # @api private
305
- def browser_name
306
- options[:browser].to_s
363
+ def unhandled_alert_errors
364
+ @unhandled_alert_errors ||= with_legacy_error(
365
+ [Selenium::WebDriver::Error::UnexpectedAlertOpenError],
366
+ 'UnhandledAlertError'
367
+ )
307
368
  end
308
369
 
309
- def modal_error
310
- if defined?(Selenium::WebDriver::Error::NoSuchAlertError)
311
- Selenium::WebDriver::Error::NoSuchAlertError
312
- else
313
- Selenium::WebDriver::Error::NoAlertPresentError
314
- end
370
+ def delete_all_cookies
371
+ @browser.manage.delete_all_cookies
315
372
  end
316
373
 
317
- def find_window(locator)
318
- handles = browser.window_handles
319
- return locator if handles.include? locator
374
+ def clear_storage
375
+ clear_session_storage unless options[:clear_session_storage] == false
376
+ clear_local_storage unless options[:clear_local_storage] == false
377
+ rescue Selenium::WebDriver::Error::JavascriptError
378
+ # session/local storage may not be available if on non-http pages (e.g. about:blank)
379
+ end
320
380
 
321
- original_handle = browser.window_handle
322
- handles.each do |handle|
323
- switch_to_window(handle)
324
- if (locator == browser.execute_script("return window.name") ||
325
- browser.title.include?(locator) ||
326
- browser.current_url.include?(locator))
327
- switch_to_window(original_handle)
328
- return handle
381
+ def clear_session_storage
382
+ if @browser.respond_to? :session_storage
383
+ @browser.session_storage.clear
384
+ else
385
+ begin
386
+ @browser&.execute_script('window.sessionStorage.clear()')
387
+ rescue # rubocop:disable Style/RescueStandardError
388
+ unless options[:clear_session_storage].nil?
389
+ warn 'sessionStorage clear requested but is not supported by this driver'
390
+ end
329
391
  end
330
392
  end
331
- raise Capybara::ElementNotFound, "Could not find a window identified by #{locator}"
332
393
  end
333
394
 
334
- def insert_modal_handlers(accept, response_text)
335
- prompt_response = if accept
336
- if response_text.nil?
337
- "default_text"
338
- else
339
- "'#{response_text.gsub("\\", "\\\\\\").gsub("'", "\\\\'")}'"
340
- end
395
+ def clear_local_storage
396
+ if @browser.respond_to? :local_storage
397
+ @browser.local_storage.clear
341
398
  else
342
- 'null'
399
+ begin
400
+ @browser&.execute_script('window.localStorage.clear()')
401
+ rescue # rubocop:disable Style/RescueStandardError
402
+ unless options[:clear_local_storage].nil?
403
+ warn 'localStorage clear requested but is not supported by this driver'
404
+ end
405
+ end
343
406
  end
407
+ end
408
+
409
+ def navigate_with_accept(url)
410
+ @browser.navigate.to(url)
411
+ sleep 0.1 # slight wait for alert
412
+ @browser.switch_to.alert.accept
413
+ rescue modal_error
414
+ # alert now gone, should mean navigation happened
415
+ end
344
416
 
345
- script = <<-JS
346
- if (typeof window.capybara === 'undefined') {
347
- window.capybara = {
348
- modal_handlers: [],
349
- current_modal_status: function() {
350
- return [this.modal_handlers[0].called, this.modal_handlers[0].modal_text];
351
- },
352
- add_handler: function(handler) {
353
- this.modal_handlers.unshift(handler);
354
- },
355
- remove_handler: function(handler) {
356
- window.alert = handler.alert;
357
- window.confirm = handler.confirm;
358
- window.prompt = handler.prompt;
359
- },
360
- handler_called: function(handler, str) {
361
- handler.called = true;
362
- handler.modal_text = str;
363
- this.remove_handler(handler);
364
- }
365
- };
366
- };
367
-
368
- var modal_handler = {
369
- prompt: window.prompt,
370
- confirm: window.confirm,
371
- alert: window.alert,
372
- called: false
373
- }
374
- window.capybara.add_handler(modal_handler);
375
-
376
- window.alert = window.confirm = function(str = "") {
377
- window.capybara.handler_called(modal_handler, str.toString());
378
- return #{accept ? 'true' : 'false'};
379
- }
380
- window.prompt = function(str = "", default_text = "") {
381
- window.capybara.handler_called(modal_handler, str.toString());
382
- return #{prompt_response};
383
- }
384
- JS
385
- execute_script script
417
+ def modal_error
418
+ Selenium::WebDriver::Error::NoSuchAlertError
386
419
  end
387
420
 
388
421
  def within_given_window(handle)
389
- original_handle = self.current_window_handle
422
+ original_handle = current_window_handle
390
423
  if handle == original_handle
391
424
  yield
392
425
  else
@@ -397,90 +430,128 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
397
430
  end
398
431
  end
399
432
 
400
- def find_modal(options={})
433
+ def find_modal(text: nil, **options)
401
434
  # Selenium has its own built in wait (2 seconds)for a modal to show up, so this wait is really the minimum time
402
435
  # Actual wait time may be longer than specified
403
436
  wait = Selenium::WebDriver::Wait.new(
404
- timeout: options.fetch(:wait, session_options.default_max_wait_time) || 0 ,
405
- ignore: modal_error)
437
+ timeout: options.fetch(:wait, session_options.default_max_wait_time) || 0,
438
+ ignore: modal_error
439
+ )
406
440
  begin
407
441
  wait.until do
408
442
  alert = @browser.switch_to.alert
409
- regexp = options[:text].is_a?(Regexp) ? options[:text] : Regexp.escape(options[:text].to_s)
410
- alert.text.match(regexp) ? alert : nil
443
+ regexp = text.is_a?(Regexp) ? text : Regexp.new(Regexp.escape(text.to_s))
444
+ matched = alert.text.match?(regexp)
445
+ unless matched
446
+ raise Capybara::ModalNotFound, "Unable to find modal dialog with #{text} - found '#{alert.text}' instead."
447
+ end
448
+
449
+ alert
411
450
  end
412
- rescue Selenium::WebDriver::Error::TimeOutError
413
- raise Capybara::ModalNotFound.new("Unable to find modal dialog#{" with #{options[:text]}" if options[:text]}")
451
+ rescue *find_modal_errors
452
+ raise Capybara::ModalNotFound, "Unable to find modal dialog#{" with #{text}" if text}"
414
453
  end
415
454
  end
416
455
 
417
- def find_headless_modal(options={})
418
- # Selenium has its own built in wait (2 seconds)for a modal to show up, so this wait is really the minimum time
419
- # Actual wait time may be longer than specified
420
- wait = Selenium::WebDriver::Wait.new(
421
- timeout: options.fetch(:wait, session_options.default_max_wait_time) || 0 ,
422
- ignore: modal_error)
423
- begin
424
- wait.until do
425
- called, alert_text = evaluate_script('window.capybara && window.capybara.current_modal_status()')
426
- if called
427
- execute_script('window.capybara && window.capybara.modal_handlers.shift()')
428
- regexp = options[:text].is_a?(Regexp) ? options[:text] : Regexp.escape(options[:text].to_s)
429
- if alert_text.match(regexp)
430
- alert_text
431
- else
432
- raise Capybara::ModalNotFound.new("Unable to find modal dialog#{" with #{options[:text]}" if options[:text]}")
433
- end
434
- elsif called.nil?
435
- # page changed so modal_handler data has gone away
436
- warn "Can't verify modal text when page change occurs - ignoring" if options[:text]
437
- ""
438
- else
439
- nil
456
+ def find_modal_errors
457
+ @find_modal_errors ||= with_legacy_error([Selenium::WebDriver::Error::TimeoutError], 'TimeOutError')
458
+ end
459
+
460
+ def with_legacy_error(errors, legacy_error)
461
+ errors.tap do |errs|
462
+ unless selenium_4?
463
+ ::Selenium::WebDriver.logger.suppress_deprecations do
464
+ errs << Selenium::WebDriver::Error.const_get(legacy_error)
440
465
  end
441
466
  end
442
- rescue Selenium::WebDriver::Error::TimeOutError
443
- raise Capybara::ModalNotFound.new("Unable to find modal dialog#{" with #{options[:text]}" if options[:text]}")
444
467
  end
445
468
  end
446
469
 
447
470
  def silenced_unknown_error_message?(msg)
448
- silenced_unknown_error_messages.any? { |r| msg =~ r }
471
+ silenced_unknown_error_messages.any? { |regex| msg.match? regex }
449
472
  end
450
473
 
451
474
  def silenced_unknown_error_messages
452
- [ /Error communicating with the remote browser/ ]
475
+ [/Error communicating with the remote browser/]
453
476
  end
454
477
 
455
478
  def unwrap_script_result(arg)
479
+ # TODO: move into the case when we drop support for Selenium < 4.1
480
+ element_types = [Selenium::WebDriver::Element]
481
+ element_types.push(Selenium::WebDriver::ShadowRoot) if defined?(Selenium::WebDriver::ShadowRoot)
482
+
456
483
  case arg
457
484
  when Array
458
- arg.map { |e| unwrap_script_result(e) }
485
+ arg.map { |arr| unwrap_script_result(arr) }
459
486
  when Hash
460
- arg.each { |k, v| arg[k] = unwrap_script_result(v) }
461
- when Selenium::WebDriver::Element
462
- Capybara::Selenium::Node.new(self, arg)
487
+ arg.transform_values! { |value| unwrap_script_result(value) }
488
+ when *element_types
489
+ build_node(arg)
463
490
  else
464
491
  arg
465
492
  end
466
493
  end
467
494
 
468
- def load_selenium
469
- begin
470
- require 'selenium-webdriver'
471
- # Fix for selenium-webdriver 3.4.0 which misnamed these
472
- if !defined?(::Selenium::WebDriver::Error::ElementNotInteractableError)
473
- ::Selenium::WebDriver::Error.const_set('ElementNotInteractableError', Class.new(::Selenium::WebDriver::Error::WebDriverError))
474
- end
475
- if !defined?(::Selenium::WebDriver::Error::ElementClickInterceptedError)
476
- ::Selenium::WebDriver::Error.const_set('ElementClickInterceptedError', Class.new(::Selenium::WebDriver::Error::WebDriverError))
477
- end
478
- rescue LoadError => e
479
- if e.message =~ /selenium-webdriver/
480
- raise LoadError, "Capybara's selenium driver is unable to load `selenium-webdriver`, please install the gem and add `gem 'selenium-webdriver'` to your Gemfile if you are using bundler."
481
- else
482
- raise e
483
- end
495
+ def find_context
496
+ browser
497
+ end
498
+
499
+ def build_node(native_node, initial_cache = {})
500
+ ::Capybara::Selenium::Node.new(self, native_node, initial_cache)
501
+ end
502
+
503
+ def bridge
504
+ browser.send(:bridge)
505
+ end
506
+
507
+ def specialize_driver
508
+ browser_type = browser.browser
509
+ Capybara::Selenium::Driver.specializations.select { |k, _v| k === browser_type }.each_value do |specialization| # rubocop:disable Style/CaseEquality
510
+ extend specialization
511
+ end
512
+ end
513
+
514
+ def setup_exit_handler
515
+ main = Process.pid
516
+ at_exit do
517
+ # Store the exit status of the test run since it goes away after calling the at_exit proc...
518
+ @exit_status = $ERROR_INFO.status if $ERROR_INFO.is_a?(SystemExit)
519
+ quit if Process.pid == main
520
+ exit @exit_status if @exit_status # Force exit with stored status
521
+ end
522
+ end
523
+
524
+ def reset_browser_state
525
+ clear_browser_state
526
+ @browser.navigate.to('about:blank')
527
+ end
528
+
529
+ def wait_for_empty_page(timer)
530
+ until find_xpath('/html/body/*').empty?
531
+ raise Capybara::ExpectationNotMet, 'Timed out waiting for Selenium session reset' if timer.expired?
532
+
533
+ sleep 0.01
534
+
535
+ # It has been observed that it is possible that asynchronous JS code in
536
+ # the application under test can navigate the browser away from about:blank
537
+ # if the timing is just right. Ensure we are still at about:blank...
538
+ @browser.navigate.to('about:blank') unless current_url == 'about:blank'
484
539
  end
485
540
  end
541
+
542
+ def accept_unhandled_reset_alert
543
+ @browser.switch_to.alert.accept
544
+ sleep 0.25 # allow time for the modal to be handled
545
+ rescue modal_error
546
+ # The alert is now gone.
547
+ # If navigation has not occurred attempt again and accept alert
548
+ # since FF may have dismissed the alert at first attempt.
549
+ navigate_with_accept('about:blank') if current_url != 'about:blank'
550
+ end
486
551
  end
552
+
553
+ require 'capybara/selenium/driver_specializations/chrome_driver'
554
+ require 'capybara/selenium/driver_specializations/firefox_driver'
555
+ require 'capybara/selenium/driver_specializations/internet_explorer_driver'
556
+ require 'capybara/selenium/driver_specializations/safari_driver'
557
+ require 'capybara/selenium/driver_specializations/edge_driver'