capybara 2.15.1 → 3.36.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (311) hide show
  1. checksums.yaml +5 -5
  2. data/.yardopts +1 -1
  3. data/History.md +902 -10
  4. data/License.txt +1 -1
  5. data/README.md +103 -75
  6. data/lib/capybara/config.rb +29 -56
  7. data/lib/capybara/cucumber.rb +2 -3
  8. data/lib/capybara/driver/base.rb +39 -18
  9. data/lib/capybara/driver/node.rb +36 -10
  10. data/lib/capybara/dsl.rb +15 -6
  11. data/lib/capybara/helpers.rb +65 -30
  12. data/lib/capybara/minitest/spec.rb +173 -81
  13. data/lib/capybara/minitest.rb +220 -111
  14. data/lib/capybara/node/actions.rb +274 -171
  15. data/lib/capybara/node/base.rb +41 -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 +353 -137
  19. data/lib/capybara/node/finders.rb +144 -138
  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 +13 -10
  24. data/lib/capybara/queries/base_query.rb +39 -28
  25. data/lib/capybara/queries/current_path_query.rb +22 -25
  26. data/lib/capybara/queries/match_query.rb +14 -7
  27. data/lib/capybara/queries/selector_query.rb +646 -145
  28. data/lib/capybara/queries/sibling_query.rb +12 -10
  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 +57 -41
  33. data/lib/capybara/rack_test/css_handlers.rb +6 -4
  34. data/lib/capybara/rack_test/driver.rb +18 -13
  35. data/lib/capybara/rack_test/errors.rb +6 -0
  36. data/lib/capybara/rack_test/form.rb +73 -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 +44 -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 +45 -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 +144 -264
  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 +279 -0
  86. data/lib/capybara/selector/filter.rb +2 -17
  87. data/lib/capybara/selector/filter_set.rb +81 -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 +214 -0
  93. data/lib/capybara/selector/selector.rb +89 -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 +343 -276
  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 +228 -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 +40 -0
  112. data/lib/capybara/selenium/node.rb +508 -124
  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 +69 -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 +79 -59
  129. data/lib/capybara/session/matchers.rb +41 -25
  130. data/lib/capybara/session.rb +371 -357
  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 +159 -13
  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 +34 -6
  137. data/lib/capybara/spec/session/active_element_spec.rb +31 -0
  138. data/lib/capybara/spec/session/all_spec.rb +161 -55
  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.rb → assert_text_spec.rb} +91 -59
  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 +138 -69
  147. data/lib/capybara/spec/session/body_spec.rb +12 -13
  148. data/lib/capybara/spec/session/check_spec.rb +116 -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 +11 -10
  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 +23 -0
  162. data/lib/capybara/spec/session/evaluate_script_spec.rb +30 -9
  163. data/lib/capybara/spec/session/execute_script_spec.rb +10 -8
  164. data/lib/capybara/spec/session/fill_in_spec.rb +128 -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 +245 -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 +69 -0
  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 +94 -13
  181. data/lib/capybara/spec/session/has_css_spec.rb +272 -137
  182. data/lib/capybara/spec/session/has_current_path_spec.rb +87 -45
  183. data/lib/capybara/spec/session/has_field_spec.rb +139 -59
  184. data/lib/capybara/spec/session/has_link_spec.rb +34 -9
  185. data/lib/capybara/spec/session/has_none_selectors_spec.rb +78 -0
  186. data/lib/capybara/spec/session/has_select_spec.rb +103 -74
  187. data/lib/capybara/spec/session/has_selector_spec.rb +105 -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 +113 -61
  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 +950 -152
  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 +69 -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 -80
  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 +99 -32
  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 +70 -44
  224. data/lib/capybara/spec/spec_helper.rb +49 -43
  225. data/lib/capybara/spec/test_app.rb +89 -42
  226. data/lib/capybara/spec/views/animated.erb +49 -0
  227. data/lib/capybara/spec/views/form.erb +141 -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 +11 -0
  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 +18 -0
  247. data/lib/capybara/spec/views/with_hover.erb +8 -2
  248. data/lib/capybara/spec/views/with_hover1.erb +10 -0
  249. data/lib/capybara/spec/views/with_html.erb +70 -11
  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 +39 -3
  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_other.erb +6 -0
  256. data/lib/capybara/spec/views/with_slow_unload.erb +2 -1
  257. data/lib/capybara/spec/views/with_sortable_js.erb +21 -0
  258. data/lib/capybara/spec/views/with_unload_alert.erb +1 -0
  259. data/lib/capybara/spec/views/with_windows.erb +1 -1
  260. data/lib/capybara/spec/views/within_frames.erb +5 -2
  261. data/lib/capybara/version.rb +2 -1
  262. data/lib/capybara/window.rb +35 -33
  263. data/lib/capybara.rb +126 -103
  264. data/spec/basic_node_spec.rb +60 -34
  265. data/spec/capybara_spec.rb +53 -104
  266. data/spec/css_builder_spec.rb +101 -0
  267. data/spec/css_splitter_spec.rb +38 -0
  268. data/spec/dsl_spec.rb +81 -62
  269. data/spec/filter_set_spec.rb +27 -9
  270. data/spec/fixtures/certificate.pem +25 -0
  271. data/spec/fixtures/key.pem +27 -0
  272. data/spec/fixtures/selenium_driver_rspec_failure.rb +5 -4
  273. data/spec/fixtures/selenium_driver_rspec_success.rb +5 -4
  274. data/spec/minitest_spec.rb +49 -7
  275. data/spec/minitest_spec_spec.rb +94 -59
  276. data/spec/per_session_config_spec.rb +14 -13
  277. data/spec/rack_test_spec.rb +180 -125
  278. data/spec/regexp_dissassembler_spec.rb +250 -0
  279. data/spec/result_spec.rb +99 -45
  280. data/spec/rspec/features_spec.rb +37 -31
  281. data/spec/rspec/scenarios_spec.rb +9 -7
  282. data/spec/rspec/shared_spec_matchers.rb +448 -421
  283. data/spec/rspec/views_spec.rb +5 -3
  284. data/spec/rspec_matchers_spec.rb +27 -11
  285. data/spec/rspec_spec.rb +109 -89
  286. data/spec/sauce_spec_chrome.rb +43 -0
  287. data/spec/selector_spec.rb +396 -67
  288. data/spec/selenium_spec_chrome.rb +183 -35
  289. data/spec/selenium_spec_chrome_remote.rb +101 -0
  290. data/spec/selenium_spec_edge.rb +47 -0
  291. data/spec/selenium_spec_firefox.rb +184 -41
  292. data/spec/selenium_spec_firefox_remote.rb +80 -0
  293. data/spec/selenium_spec_ie.rb +147 -0
  294. data/spec/selenium_spec_safari.rb +156 -0
  295. data/spec/server_spec.rb +198 -99
  296. data/spec/session_spec.rb +53 -16
  297. data/spec/shared_selenium_node.rb +79 -0
  298. data/spec/shared_selenium_session.rb +489 -97
  299. data/spec/spec_helper.rb +93 -7
  300. data/spec/xpath_builder_spec.rb +93 -0
  301. metadata +365 -70
  302. data/.yard/templates_custom/default/class/html/selectors.erb +0 -38
  303. data/.yard/templates_custom/default/class/html/setup.rb +0 -17
  304. data/.yard/yard_extensions.rb +0 -78
  305. data/lib/capybara/query.rb +0 -7
  306. data/lib/capybara/rspec/compound.rb +0 -95
  307. data/lib/capybara/spec/session/assert_current_path.rb +0 -72
  308. data/lib/capybara/spec/session/assert_selector.rb +0 -148
  309. data/lib/capybara/spec/session/source_spec.rb +0 -0
  310. data/lib/capybara/spec/views/with_title.erb +0 -5
  311. data/spec/selenium_spec_marionette.rb +0 -117
@@ -1,66 +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
 
15
- def browser
16
- unless @browser
17
- if firefox?
18
- options[:desired_capabilities] ||= {}
19
- options[:desired_capabilities].merge!({ unexpectedAlertBehaviour: "ignore" })
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."
20
48
  end
21
49
 
22
- @processed_options = options.reject { |key,_val| SPECIAL_OPTIONS.include?(key) }
23
- @browser = Selenium::WebDriver.for(options[:browser], @processed_options)
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
24
56
 
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)))
57
+ attr_reader :specializations
27
58
 
28
- main = Process.pid
29
- at_exit do
30
- # Store the exit status of the test run since it goes away after calling the at_exit proc...
31
- @exit_status = $!.status if $!.is_a?(SystemExit)
32
- quit if Process.pid == main
33
- exit @exit_status if @exit_status # Force exit with stored status
34
- end
59
+ def register_specialization(browser_name, specialization)
60
+ @specializations ||= {}
61
+ @specializations[browser_name] = specialization
35
62
  end
36
- @browser
37
63
  end
38
64
 
39
- def initialize(app, options={})
40
- @session = nil
41
- begin
42
- require 'selenium-webdriver'
43
- # Fix for selenium-webdriver 3.4.0 which misnamed these
44
- if !defined?(::Selenium::WebDriver::Error::ElementNotInteractableError)
45
- ::Selenium::WebDriver::Error.const_set('ElementNotInteractableError', Class.new(::Selenium::WebDriver::Error::WebDriverError))
46
- end
47
- if !defined?(::Selenium::WebDriver::Error::ElementClickInterceptedError)
48
- ::Selenium::WebDriver::Error.const_set('ElementClickInterceptedError', Class.new(::Selenium::WebDriver::Error::WebDriverError))
65
+ def browser
66
+ unless @browser
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
49
74
  end
50
- rescue LoadError => e
51
- if e.message =~ /selenium-webdriver/
52
- 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."
75
+ processed_options = options.reject { |key, _val| SPECIAL_OPTIONS.include?(key) }
76
+
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)
53
82
  else
54
- raise e
83
+ Selenium::WebDriver.for(options[:browser], processed_options)
55
84
  end
56
- end
57
85
 
86
+ specialize_driver
87
+ setup_exit_handler
88
+ end
89
+ @browser
90
+ end
58
91
 
92
+ def initialize(app, **options)
93
+ super()
94
+ self.class.load_selenium
59
95
  @app = app
60
96
  @browser = nil
61
97
  @exit_status = nil
62
- @frame_handles = {}
98
+ @frame_handles = Hash.new { |hash, handle| hash[handle] = [] }
63
99
  @options = DEFAULT_OPTIONS.merge(options)
100
+ @node_class = ::Capybara::Selenium::Node
64
101
  end
65
102
 
66
103
  def visit(path)
@@ -68,10 +105,7 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
68
105
  end
69
106
 
70
107
  def refresh
71
- accept_modal(nil, wait: 0.1) do
72
- browser.navigate.refresh
73
- end
74
- rescue Capybara::ModalNotFound
108
+ browser.navigate.refresh
75
109
  end
76
110
 
77
111
  def go_back
@@ -84,6 +118,8 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
84
118
 
85
119
  def html
86
120
  browser.page_source
121
+ rescue Selenium::WebDriver::Error::JavascriptError => e
122
+ raise unless e.message.include?('documentElement is null')
87
123
  end
88
124
 
89
125
  def title
@@ -94,19 +130,11 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
94
130
  browser.current_url
95
131
  end
96
132
 
97
- def find_xpath(selector)
98
- browser.find_elements(:xpath, selector).map { |node| Capybara::Selenium::Node.new(self, node) }
99
- end
100
-
101
- def find_css(selector)
102
- browser.find_elements(:css, selector).map { |node| Capybara::Selenium::Node.new(self, node) }
103
- end
104
-
105
133
  def wait?; true; end
106
134
  def needs_server?; true; end
107
135
 
108
136
  def execute_script(script, *args)
109
- browser.execute_script(script, *args.map { |arg| arg.is_a?(Capybara::Selenium::Node) ? arg.native : arg} )
137
+ browser.execute_script(script, *native_args(args))
110
138
  end
111
139
 
112
140
  def evaluate_script(script, *args)
@@ -114,79 +142,71 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
114
142
  unwrap_script_result(result)
115
143
  end
116
144
 
117
- def save_screenshot(path, _options={})
145
+ def evaluate_async_script(script, *args)
146
+ browser.manage.timeouts.script_timeout = Capybara.default_max_wait_time
147
+ result = browser.execute_async_script(script, *native_args(args))
148
+ unwrap_script_result(result)
149
+ end
150
+
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)
118
161
  browser.save_screenshot(path)
119
162
  end
120
163
 
121
164
  def reset!
122
165
  # Use instance variable directly so we avoid starting the browser just to reset the session
123
- if @browser
124
- navigated = false
125
- start_time = Capybara::Helpers.monotonic_time
126
- begin
127
- if !navigated
128
- # Only trigger a navigation if we haven't done it already, otherwise it
129
- # can trigger an endless series of unload modals
130
- begin
131
- @browser.manage.delete_all_cookies
132
- if options[:clear_session_storage]
133
- if @browser.respond_to? :session_storage
134
- @browser.session_storage.clear
135
- else
136
- warn "sessionStorage clear requested but is not available for this driver"
137
- end
138
- end
139
- if options[:clear_local_storage]
140
- if @browser.respond_to? :local_storage
141
- @browser.local_storage.clear
142
- else
143
- warn "localStorage clear requested but is not available for this driver"
144
- end
145
- end
146
- rescue Selenium::WebDriver::Error::UnhandledError
147
- # delete_all_cookies fails when we've previously gone
148
- # to about:blank, so we rescue this error and do nothing
149
- # instead.
150
- end
151
- @browser.navigate.to("about:blank")
152
- end
153
- navigated = true
166
+ return unless @browser
154
167
 
155
- #Ensure the page is empty and trigger an UnhandledAlertError for any modals that appear during unload
156
- until find_xpath("/html/body/*").empty? do
157
- raise Capybara::ExpectationNotMet.new('Timed out waiting for Selenium session reset') if (Capybara::Helpers.monotonic_time - start_time) >= 10
158
- sleep 0.05
159
- end
160
- rescue Selenium::WebDriver::Error::UnhandledAlertError, Selenium::WebDriver::Error::UnexpectedAlertOpenError
161
- # This error is thrown if an unhandled alert is on the page
162
- # Firefox appears to automatically dismiss this alert, chrome does not
163
- # We'll try to accept it
164
- begin
165
- @browser.switch_to.alert.accept
166
- sleep 0.25 # allow time for the modal to be handled
167
- rescue Selenium::WebDriver::Error::NoAlertPresentError
168
- # The alert is now gone - nothing to do
169
- end
170
- # try cleaning up the browser again
171
- retry
172
- 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)
173
196
  end
174
197
  end
175
198
 
176
199
  def switch_to_frame(frame)
200
+ handles = @frame_handles[current_window_handle]
177
201
  case frame
178
202
  when :top
179
- @frame_handles[browser.window_handle] = []
203
+ handles.clear
180
204
  browser.switch_to.default_content
181
205
  when :parent
182
- # would love to use browser.switch_to.parent_frame here
183
- # but it has an issue if the current frame is removed from within it
184
- @frame_handles[browser.window_handle].pop
185
- browser.switch_to.default_content
186
- @frame_handles[browser.window_handle].each { |fh| browser.switch_to.frame(fh) }
206
+ handles.pop
207
+ browser.switch_to.parent_frame
187
208
  else
188
- @frame_handles[browser.window_handle] ||= []
189
- @frame_handles[browser.window_handle] << frame.native
209
+ handles << frame
190
210
  browser.switch_to.frame(frame.native)
191
211
  end
192
212
  end
@@ -204,12 +224,7 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
204
224
 
205
225
  def resize_window_to(handle, width, height)
206
226
  within_given_window(handle) do
207
- # Don't set the size if already set - See https://github.com/mozilla/geckodriver/issues/643
208
- if marionette? && (window_size(handle) == [width, height])
209
- {}
210
- else
211
- browser.manage.window.resize_to(width, height)
212
- end
227
+ browser.manage.window.resize_to(width, height)
213
228
  end
214
229
  end
215
230
 
@@ -220,7 +235,15 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
220
235
  sleep 0.1 # work around for https://code.google.com/p/selenium/issues/detail?id=7405
221
236
  end
222
237
 
238
+ def fullscreen_window(handle)
239
+ within_given_window(handle) do
240
+ browser.manage.window.full_screen
241
+ end
242
+ end
243
+
223
244
  def close_window(handle)
245
+ raise ArgumentError, 'Not allowed to close the primary window' if handle == window_handles.first
246
+
224
247
  within_given_window(handle) do
225
248
  browser.close
226
249
  end
@@ -230,7 +253,16 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
230
253
  browser.window_handles
231
254
  end
232
255
 
233
- 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
234
266
  browser.execute_script('window.open();')
235
267
  end
236
268
 
@@ -238,43 +270,29 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
238
270
  browser.switch_to.window handle
239
271
  end
240
272
 
241
- def within_window(locator)
242
- handle = find_window(locator)
243
- browser.switch_to.window(handle) { yield }
244
- end
273
+ def accept_modal(_type, **options)
274
+ yield if block_given?
275
+ modal = find_modal(**options)
245
276
 
246
- def accept_modal(_type, options={})
247
- if headless_chrome?
248
- insert_modal_handlers(true, options[:with], options[:text])
249
- yield if block_given?
250
- find_headless_modal(options)
251
- else
252
- yield if block_given?
253
- modal = find_modal(options)
254
- modal.send_keys options[:with] if options[:with]
255
- message = modal.text
256
- modal.accept
257
- message
258
- end
277
+ modal.send_keys options[:with] if options[:with]
278
+
279
+ message = modal.text
280
+ modal.accept
281
+ message
259
282
  end
260
283
 
261
- def dismiss_modal(_type, options={})
262
- if headless_chrome?
263
- insert_modal_handlers(false, options[:with], options[:text])
264
- yield if block_given?
265
- find_headless_modal(options)
266
- else
267
- yield if block_given?
268
- modal = find_modal(options)
269
- message = modal.text
270
- modal.dismiss
271
- message
272
- end
284
+ def dismiss_modal(_type, **options)
285
+ yield if block_given?
286
+ modal = find_modal(**options)
287
+ message = modal.text
288
+ modal.dismiss
289
+ message
273
290
  end
274
291
 
275
292
  def quit
276
- @browser.quit if @browser
277
- rescue Errno::ECONNREFUSED
293
+ @browser&.quit
294
+ rescue Selenium::WebDriver::Error::SessionNotCreatedError, Errno::ECONNREFUSED,
295
+ Selenium::WebDriver::Error::InvalidSessionIdError
278
296
  # Browser must have already gone
279
297
  rescue Selenium::WebDriver::Error::UnknownError => e
280
298
  unless silenced_unknown_error_message?(e.message) # Most likely already gone
@@ -286,123 +304,119 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
286
304
  end
287
305
 
288
306
  def invalid_element_errors
289
- [::Selenium::WebDriver::Error::StaleElementReferenceError,
290
- ::Selenium::WebDriver::Error::UnhandledError,
291
- ::Selenium::WebDriver::Error::ElementNotVisibleError,
292
- ::Selenium::WebDriver::Error::InvalidSelectorError, # Work around a race condition that can occur with chromedriver and #go_back/#go_forward
293
- ::Selenium::WebDriver::Error::ElementNotInteractableError,
294
- ::Selenium::WebDriver::Error::ElementClickInterceptedError,
295
- ::Selenium::WebDriver::Error::InvalidElementStateError,
296
- ::Selenium::WebDriver::Error::ElementNotSelectableError,
297
- ]
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
+ end
298
327
  end
299
328
 
300
329
  def no_such_window_error
301
330
  Selenium::WebDriver::Error::NoSuchWindowError
302
331
  end
303
332
 
304
- # @api private
305
- def marionette?
306
- firefox? && browser && @w3c
333
+ private
334
+
335
+ def selenium_4?
336
+ defined?(Selenium::WebDriver::VERSION) && (Selenium::WebDriver::VERSION.to_f >= 4)
307
337
  end
308
338
 
309
- # @api private
310
- def firefox?
311
- browser_name == "firefox"
339
+ def native_args(args)
340
+ args.map { |arg| arg.is_a?(Capybara::Selenium::Node) ? arg.native : arg }
312
341
  end
313
342
 
314
- # @api private
315
- def chrome?
316
- browser_name == "chrome"
343
+ def native_active_element
344
+ browser.switch_to.active_element
317
345
  end
318
346
 
319
- # @api private
320
- def headless_chrome?
321
- if chrome?
322
- caps = @processed_options[:desired_capabilities]
323
- chrome_options = caps[:chrome_options] || caps[:chromeOptions] || {}
324
- args = chrome_options['args'] || chrome_options[:args] || []
325
- return args.include?("--headless") || args.include?("headless")
326
- end
327
- return false
347
+ def clear_browser_state
348
+ delete_all_cookies
349
+ clear_storage
350
+ rescue *clear_browser_state_errors
351
+ # delete_all_cookies fails when we've previously gone
352
+ # to about:blank, so we rescue this error and do nothing
353
+ # instead.
328
354
  end
329
355
 
356
+ def clear_browser_state_errors
357
+ @clear_browser_state_errors ||= [Selenium::WebDriver::Error::UnknownError]
358
+ end
330
359
 
331
- # @deprecated This method is being removed
332
- def browser_initialized?
333
- super && !@browser.nil?
360
+ def unhandled_alert_errors
361
+ @unhandled_alert_errors ||= with_legacy_error(
362
+ [Selenium::WebDriver::Error::UnexpectedAlertOpenError],
363
+ 'UnhandledAlertError'
364
+ )
334
365
  end
335
366
 
336
- private
367
+ def delete_all_cookies
368
+ @browser.manage.delete_all_cookies
369
+ end
337
370
 
338
- # @api private
339
- def browser_name
340
- options[:browser].to_s
371
+ def clear_storage
372
+ clear_session_storage unless options[:clear_session_storage] == false
373
+ clear_local_storage unless options[:clear_local_storage] == false
374
+ rescue Selenium::WebDriver::Error::JavascriptError
375
+ # session/local storage may not be available if on non-http pages (e.g. about:blank)
341
376
  end
342
377
 
343
- def find_window(locator)
344
- handles = browser.window_handles
345
- return locator if handles.include? locator
378
+ def clear_session_storage
379
+ if @browser.respond_to? :session_storage
380
+ @browser.session_storage.clear
381
+ else
382
+ begin
383
+ @browser&.execute_script('window.sessionStorage.clear()')
384
+ rescue # rubocop:disable Style/RescueStandardError
385
+ unless options[:clear_session_storage].nil?
386
+ warn 'sessionStorage clear requested but is not supported by this driver'
387
+ end
388
+ end
389
+ end
390
+ end
346
391
 
347
- original_handle = browser.window_handle
348
- handles.each do |handle|
349
- switch_to_window(handle)
350
- if (locator == browser.execute_script("return window.name") ||
351
- browser.title.include?(locator) ||
352
- browser.current_url.include?(locator))
353
- switch_to_window(original_handle)
354
- return handle
392
+ def clear_local_storage
393
+ if @browser.respond_to? :local_storage
394
+ @browser.local_storage.clear
395
+ else
396
+ begin
397
+ @browser&.execute_script('window.localStorage.clear()')
398
+ rescue # rubocop:disable Style/RescueStandardError
399
+ unless options[:clear_local_storage].nil?
400
+ warn 'localStorage clear requested but is not supported by this driver'
401
+ end
355
402
  end
356
403
  end
357
- raise Capybara::ElementNotFound, "Could not find a window identified by #{locator}"
358
- end
359
-
360
- def insert_modal_handlers(accept, response_text, expected_text=nil)
361
- script = <<-JS
362
- if (typeof window.capybara === 'undefined') {
363
- window.capybara = {
364
- modal_handlers: [],
365
- current_modal_status: function() {
366
- return [this.modal_handlers[0].called, this.modal_handlers[0].modal_text];
367
- },
368
- add_handler: function(handler) {
369
- this.modal_handlers.unshift(handler);
370
- },
371
- remove_handler: function(handler) {
372
- window.alert = handler.alert;
373
- window.confirm = handler.confirm;
374
- window.prompt = handler.prompt;
375
- },
376
- handler_called: function(handler, str) {
377
- handler.called = true;
378
- handler.modal_text = str;
379
- this.remove_handler(handler);
380
- }
381
- };
382
- };
383
-
384
- var modal_handler = {
385
- prompt: window.prompt,
386
- confirm: window.confirm,
387
- alert: window.alert,
388
- called: false
389
- }
390
- window.capybara.add_handler(modal_handler);
391
-
392
- window.alert = window.confirm = function(str) {
393
- window.capybara.handler_called(modal_handler, str);
394
- return #{accept ? 'true' : 'false'};
395
- };
396
- window.prompt = function(str) {
397
- window.capybara.handler_called(modal_handler, str);
398
- return #{accept ? "'#{response_text}'" : 'null'};
399
- }
400
- JS
401
- execute_script script
404
+ end
405
+
406
+ def navigate_with_accept(url)
407
+ @browser.navigate.to(url)
408
+ sleep 0.1 # slight wait for alert
409
+ @browser.switch_to.alert.accept
410
+ rescue modal_error
411
+ # alert now gone, should mean navigation happened
412
+ end
413
+
414
+ def modal_error
415
+ Selenium::WebDriver::Error::NoSuchAlertError
402
416
  end
403
417
 
404
418
  def within_given_window(handle)
405
- original_handle = self.current_window_handle
419
+ original_handle = current_window_handle
406
420
  if handle == original_handle
407
421
  yield
408
422
  else
@@ -413,71 +427,124 @@ class Capybara::Selenium::Driver < Capybara::Driver::Base
413
427
  end
414
428
  end
415
429
 
416
- def find_modal(options={})
430
+ def find_modal(text: nil, **options)
417
431
  # Selenium has its own built in wait (2 seconds)for a modal to show up, so this wait is really the minimum time
418
432
  # Actual wait time may be longer than specified
419
433
  wait = Selenium::WebDriver::Wait.new(
420
- timeout: options.fetch(:wait, session_options.default_max_wait_time) || 0 ,
421
- ignore: Selenium::WebDriver::Error::NoAlertPresentError)
434
+ timeout: options.fetch(:wait, session_options.default_max_wait_time) || 0,
435
+ ignore: modal_error
436
+ )
422
437
  begin
423
438
  wait.until do
424
439
  alert = @browser.switch_to.alert
425
- regexp = options[:text].is_a?(Regexp) ? options[:text] : Regexp.escape(options[:text].to_s)
426
- alert.text.match(regexp) ? alert : nil
440
+ regexp = text.is_a?(Regexp) ? text : Regexp.new(Regexp.escape(text.to_s))
441
+ matched = alert.text.match?(regexp)
442
+ unless matched
443
+ raise Capybara::ModalNotFound, "Unable to find modal dialog with #{text} - found '#{alert.text}' instead."
444
+ end
445
+
446
+ alert
427
447
  end
428
- rescue Selenium::WebDriver::Error::TimeOutError
429
- raise Capybara::ModalNotFound.new("Unable to find modal dialog#{" with #{options[:text]}" if options[:text]}")
448
+ rescue *find_modal_errors
449
+ raise Capybara::ModalNotFound, "Unable to find modal dialog#{" with #{text}" if text}"
430
450
  end
431
451
  end
432
452
 
433
- def find_headless_modal(options={})
434
- # Selenium has its own built in wait (2 seconds)for a modal to show up, so this wait is really the minimum time
435
- # Actual wait time may be longer than specified
436
- wait = Selenium::WebDriver::Wait.new(
437
- timeout: options.fetch(:wait, session_options.default_max_wait_time) || 0 ,
438
- ignore: Selenium::WebDriver::Error::NoAlertPresentError)
439
- begin
440
- wait.until do
441
- called, alert_text = evaluate_script('window.capybara && window.capybara.current_modal_status()')
442
- if called
443
- execute_script('window.capybara && window.capybara.modal_handlers.shift()')
444
- regexp = options[:text].is_a?(Regexp) ? options[:text] : Regexp.escape(options[:text].to_s)
445
- if alert_text.match(regexp)
446
- alert_text
447
- else
448
- raise Capybara::ModalNotFound.new("Unable to find modal dialog#{" with #{options[:text]}" if options[:text]}")
449
- end
450
- elsif called.nil?
451
- # page changed so modal_handler data has gone away
452
- warn "Can't verify modal text when page change occurs - ignoring" if options[:text]
453
- ""
454
- else
455
- nil
453
+ def find_modal_errors
454
+ @find_modal_errors ||= with_legacy_error([Selenium::WebDriver::Error::TimeoutError], 'TimeOutError')
455
+ end
456
+
457
+ def with_legacy_error(errors, legacy_error)
458
+ errors.tap do |errs|
459
+ unless selenium_4?
460
+ ::Selenium::WebDriver.logger.suppress_deprecations do
461
+ errs << Selenium::WebDriver::Error.const_get(legacy_error)
456
462
  end
457
463
  end
458
- rescue Selenium::WebDriver::Error::TimeOutError
459
- raise Capybara::ModalNotFound.new("Unable to find modal dialog#{" with #{options[:text]}" if options[:text]}")
460
464
  end
461
465
  end
462
466
 
463
467
  def silenced_unknown_error_message?(msg)
464
- silenced_unknown_error_messages.any? { |r| msg =~ r }
468
+ silenced_unknown_error_messages.any? { |regex| msg.match? regex }
465
469
  end
466
470
 
467
471
  def silenced_unknown_error_messages
468
- [ /Error communicating with the remote browser/ ]
472
+ [/Error communicating with the remote browser/]
469
473
  end
470
474
 
471
475
  def unwrap_script_result(arg)
472
476
  case arg
473
477
  when Array
474
- arg.map { |e| unwrap_script_result(e) }
478
+ arg.map { |arr| unwrap_script_result(arr) }
475
479
  when Hash
476
- arg.each { |k, v| arg[k] = unwrap_script_result(v) }
480
+ arg.transform_values! { |value| unwrap_script_result(value) }
477
481
  when Selenium::WebDriver::Element
478
- Capybara::Selenium::Node.new(self, arg)
482
+ build_node(arg)
479
483
  else
480
484
  arg
481
485
  end
482
486
  end
487
+
488
+ def find_context
489
+ browser
490
+ end
491
+
492
+ def build_node(native_node, initial_cache = {})
493
+ ::Capybara::Selenium::Node.new(self, native_node, initial_cache)
494
+ end
495
+
496
+ def bridge
497
+ browser.send(:bridge)
498
+ end
499
+
500
+ def specialize_driver
501
+ browser_type = browser.browser
502
+ Capybara::Selenium::Driver.specializations.select { |k, _v| k === browser_type }.each_value do |specialization| # rubocop:disable Style/CaseEquality
503
+ extend specialization
504
+ end
505
+ end
506
+
507
+ def setup_exit_handler
508
+ main = Process.pid
509
+ at_exit do
510
+ # Store the exit status of the test run since it goes away after calling the at_exit proc...
511
+ @exit_status = $ERROR_INFO.status if $ERROR_INFO.is_a?(SystemExit)
512
+ quit if Process.pid == main
513
+ exit @exit_status if @exit_status # Force exit with stored status
514
+ end
515
+ end
516
+
517
+ def reset_browser_state
518
+ clear_browser_state
519
+ @browser.navigate.to('about:blank')
520
+ end
521
+
522
+ def wait_for_empty_page(timer)
523
+ until find_xpath('/html/body/*').empty?
524
+ raise Capybara::ExpectationNotMet, 'Timed out waiting for Selenium session reset' if timer.expired?
525
+
526
+ sleep 0.01
527
+
528
+ # It has been observed that it is possible that asynchronous JS code in
529
+ # the application under test can navigate the browser away from about:blank
530
+ # if the timing is just right. Ensure we are still at about:blank...
531
+ @browser.navigate.to('about:blank') unless current_url == 'about:blank'
532
+ end
533
+ end
534
+
535
+ def accept_unhandled_reset_alert
536
+ @browser.switch_to.alert.accept
537
+ sleep 0.25 # allow time for the modal to be handled
538
+ rescue modal_error
539
+ # The alert is now gone.
540
+ # If navigation has not occurred attempt again and accept alert
541
+ # since FF may have dismissed the alert at first attempt.
542
+ navigate_with_accept('about:blank') if current_url != 'about:blank'
543
+ end
483
544
  end
545
+
546
+ require 'capybara/selenium/driver_specializations/chrome_driver'
547
+ require 'capybara/selenium/driver_specializations/firefox_driver'
548
+ require 'capybara/selenium/driver_specializations/internet_explorer_driver'
549
+ require 'capybara/selenium/driver_specializations/safari_driver'
550
+ require 'capybara/selenium/driver_specializations/edge_driver'