capybara 2.15.1 → 3.35.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (298) hide show
  1. checksums.yaml +5 -5
  2. data/.yardopts +1 -1
  3. data/History.md +871 -9
  4. data/License.txt +1 -1
  5. data/README.md +99 -75
  6. data/lib/capybara/config.rb +20 -59
  7. data/lib/capybara/cucumber.rb +2 -3
  8. data/lib/capybara/driver/base.rb +35 -18
  9. data/lib/capybara/driver/node.rb +35 -9
  10. data/lib/capybara/dsl.rb +15 -6
  11. data/lib/capybara/helpers.rb +72 -28
  12. data/lib/capybara/minitest/spec.rb +173 -81
  13. data/lib/capybara/minitest.rb +220 -111
  14. data/lib/capybara/node/actions.rb +270 -172
  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 +55 -26
  22. data/lib/capybara/queries/ancestor_query.rb +11 -9
  23. data/lib/capybara/queries/base_query.rb +39 -28
  24. data/lib/capybara/queries/current_path_query.rb +22 -25
  25. data/lib/capybara/queries/match_query.rb +14 -7
  26. data/lib/capybara/queries/selector_query.rb +633 -145
  27. data/lib/capybara/queries/sibling_query.rb +10 -9
  28. data/lib/capybara/queries/style_query.rb +45 -0
  29. data/lib/capybara/queries/text_query.rb +56 -38
  30. data/lib/capybara/queries/title_query.rb +8 -11
  31. data/lib/capybara/rack_test/browser.rb +57 -41
  32. data/lib/capybara/rack_test/css_handlers.rb +6 -4
  33. data/lib/capybara/rack_test/driver.rb +18 -13
  34. data/lib/capybara/rack_test/errors.rb +6 -0
  35. data/lib/capybara/rack_test/form.rb +73 -58
  36. data/lib/capybara/rack_test/node.rb +182 -78
  37. data/lib/capybara/rails.rb +3 -7
  38. data/lib/capybara/registration_container.rb +44 -0
  39. data/lib/capybara/registrations/drivers.rb +42 -0
  40. data/lib/capybara/registrations/patches/puma_ssl.rb +29 -0
  41. data/lib/capybara/registrations/servers.rb +45 -0
  42. data/lib/capybara/result.rb +96 -62
  43. data/lib/capybara/rspec/features.rb +17 -50
  44. data/lib/capybara/rspec/matcher_proxies.rb +51 -14
  45. data/lib/capybara/rspec/matchers/base.rb +111 -0
  46. data/lib/capybara/rspec/matchers/become_closed.rb +33 -0
  47. data/lib/capybara/rspec/matchers/compound.rb +88 -0
  48. data/lib/capybara/rspec/matchers/count_sugar.rb +37 -0
  49. data/lib/capybara/rspec/matchers/have_ancestor.rb +28 -0
  50. data/lib/capybara/rspec/matchers/have_current_path.rb +29 -0
  51. data/lib/capybara/rspec/matchers/have_selector.rb +77 -0
  52. data/lib/capybara/rspec/matchers/have_sibling.rb +27 -0
  53. data/lib/capybara/rspec/matchers/have_text.rb +33 -0
  54. data/lib/capybara/rspec/matchers/have_title.rb +29 -0
  55. data/lib/capybara/rspec/matchers/match_selector.rb +27 -0
  56. data/lib/capybara/rspec/matchers/match_style.rb +43 -0
  57. data/lib/capybara/rspec/matchers/spatial_sugar.rb +39 -0
  58. data/lib/capybara/rspec/matchers.rb +144 -264
  59. data/lib/capybara/rspec.rb +7 -11
  60. data/lib/capybara/selector/builders/css_builder.rb +84 -0
  61. data/lib/capybara/selector/builders/xpath_builder.rb +71 -0
  62. data/lib/capybara/selector/css.rb +89 -17
  63. data/lib/capybara/selector/definition/button.rb +63 -0
  64. data/lib/capybara/selector/definition/checkbox.rb +26 -0
  65. data/lib/capybara/selector/definition/css.rb +10 -0
  66. data/lib/capybara/selector/definition/datalist_input.rb +35 -0
  67. data/lib/capybara/selector/definition/datalist_option.rb +25 -0
  68. data/lib/capybara/selector/definition/element.rb +28 -0
  69. data/lib/capybara/selector/definition/field.rb +40 -0
  70. data/lib/capybara/selector/definition/fieldset.rb +14 -0
  71. data/lib/capybara/selector/definition/file_field.rb +13 -0
  72. data/lib/capybara/selector/definition/fillable_field.rb +33 -0
  73. data/lib/capybara/selector/definition/frame.rb +17 -0
  74. data/lib/capybara/selector/definition/id.rb +6 -0
  75. data/lib/capybara/selector/definition/label.rb +62 -0
  76. data/lib/capybara/selector/definition/link.rb +54 -0
  77. data/lib/capybara/selector/definition/link_or_button.rb +16 -0
  78. data/lib/capybara/selector/definition/option.rb +27 -0
  79. data/lib/capybara/selector/definition/radio_button.rb +27 -0
  80. data/lib/capybara/selector/definition/select.rb +81 -0
  81. data/lib/capybara/selector/definition/table.rb +109 -0
  82. data/lib/capybara/selector/definition/table_row.rb +21 -0
  83. data/lib/capybara/selector/definition/xpath.rb +5 -0
  84. data/lib/capybara/selector/definition.rb +278 -0
  85. data/lib/capybara/selector/filter.rb +2 -17
  86. data/lib/capybara/selector/filter_set.rb +83 -33
  87. data/lib/capybara/selector/filters/base.rb +50 -6
  88. data/lib/capybara/selector/filters/expression_filter.rb +8 -26
  89. data/lib/capybara/selector/filters/locator_filter.rb +29 -0
  90. data/lib/capybara/selector/filters/node_filter.rb +16 -12
  91. data/lib/capybara/selector/regexp_disassembler.rb +214 -0
  92. data/lib/capybara/selector/selector.rb +89 -210
  93. data/lib/capybara/selector/xpath_extensions.rb +17 -0
  94. data/lib/capybara/selector.rb +226 -526
  95. data/lib/capybara/selenium/atoms/getAttribute.min.js +1 -0
  96. data/lib/capybara/selenium/atoms/isDisplayed.min.js +1 -0
  97. data/lib/capybara/selenium/atoms/src/getAttribute.js +161 -0
  98. data/lib/capybara/selenium/atoms/src/isDisplayed.js +454 -0
  99. data/lib/capybara/selenium/driver.rb +334 -277
  100. data/lib/capybara/selenium/driver_specializations/chrome_driver.rb +117 -0
  101. data/lib/capybara/selenium/driver_specializations/edge_driver.rb +124 -0
  102. data/lib/capybara/selenium/driver_specializations/firefox_driver.rb +89 -0
  103. data/lib/capybara/selenium/driver_specializations/internet_explorer_driver.rb +26 -0
  104. data/lib/capybara/selenium/driver_specializations/safari_driver.rb +24 -0
  105. data/lib/capybara/selenium/extensions/file_input_click_emulation.rb +34 -0
  106. data/lib/capybara/selenium/extensions/find.rb +110 -0
  107. data/lib/capybara/selenium/extensions/html5_drag.rb +228 -0
  108. data/lib/capybara/selenium/extensions/modifier_keys_stack.rb +28 -0
  109. data/lib/capybara/selenium/extensions/scroll.rb +76 -0
  110. data/lib/capybara/selenium/logger_suppressor.rb +40 -0
  111. data/lib/capybara/selenium/node.rb +506 -124
  112. data/lib/capybara/selenium/nodes/chrome_node.rb +137 -0
  113. data/lib/capybara/selenium/nodes/edge_node.rb +104 -0
  114. data/lib/capybara/selenium/nodes/firefox_node.rb +136 -0
  115. data/lib/capybara/selenium/nodes/ie_node.rb +22 -0
  116. data/lib/capybara/selenium/nodes/safari_node.rb +118 -0
  117. data/lib/capybara/selenium/patches/action_pauser.rb +26 -0
  118. data/lib/capybara/selenium/patches/atoms.rb +18 -0
  119. data/lib/capybara/selenium/patches/is_displayed.rb +16 -0
  120. data/lib/capybara/selenium/patches/logs.rb +45 -0
  121. data/lib/capybara/selenium/patches/pause_duration_fix.rb +9 -0
  122. data/lib/capybara/selenium/patches/persistent_client.rb +20 -0
  123. data/lib/capybara/server/animation_disabler.rb +63 -0
  124. data/lib/capybara/server/checker.rb +44 -0
  125. data/lib/capybara/server/middleware.rb +71 -0
  126. data/lib/capybara/server.rb +59 -67
  127. data/lib/capybara/session/config.rb +79 -59
  128. data/lib/capybara/session/matchers.rb +41 -25
  129. data/lib/capybara/session.rb +360 -356
  130. data/lib/capybara/spec/public/jquery.js +5 -5
  131. data/lib/capybara/spec/public/offset.js +6 -0
  132. data/lib/capybara/spec/public/test.js +159 -13
  133. data/lib/capybara/spec/session/accept_alert_spec.rb +12 -11
  134. data/lib/capybara/spec/session/accept_confirm_spec.rb +6 -5
  135. data/lib/capybara/spec/session/accept_prompt_spec.rb +34 -6
  136. data/lib/capybara/spec/session/all_spec.rb +163 -55
  137. data/lib/capybara/spec/session/ancestor_spec.rb +27 -24
  138. data/lib/capybara/spec/session/assert_all_of_selectors_spec.rb +68 -38
  139. data/lib/capybara/spec/session/assert_current_path_spec.rb +75 -0
  140. data/lib/capybara/spec/session/assert_selector_spec.rb +143 -0
  141. data/lib/capybara/spec/session/assert_style_spec.rb +26 -0
  142. data/lib/capybara/spec/session/{assert_text.rb → assert_text_spec.rb} +91 -59
  143. data/lib/capybara/spec/session/{assert_title.rb → assert_title_spec.rb} +22 -12
  144. data/lib/capybara/spec/session/attach_file_spec.rb +138 -69
  145. data/lib/capybara/spec/session/body_spec.rb +12 -13
  146. data/lib/capybara/spec/session/check_spec.rb +107 -55
  147. data/lib/capybara/spec/session/choose_spec.rb +58 -31
  148. data/lib/capybara/spec/session/click_button_spec.rb +231 -173
  149. data/lib/capybara/spec/session/click_link_or_button_spec.rb +55 -35
  150. data/lib/capybara/spec/session/click_link_spec.rb +82 -58
  151. data/lib/capybara/spec/session/current_scope_spec.rb +11 -10
  152. data/lib/capybara/spec/session/current_url_spec.rb +57 -39
  153. data/lib/capybara/spec/session/dismiss_confirm_spec.rb +4 -4
  154. data/lib/capybara/spec/session/dismiss_prompt_spec.rb +3 -2
  155. data/lib/capybara/spec/session/element/{assert_match_selector.rb → assert_match_selector_spec.rb} +11 -9
  156. data/lib/capybara/spec/session/element/match_css_spec.rb +18 -10
  157. data/lib/capybara/spec/session/element/match_xpath_spec.rb +9 -7
  158. data/lib/capybara/spec/session/element/matches_selector_spec.rb +71 -57
  159. data/lib/capybara/spec/session/evaluate_async_script_spec.rb +23 -0
  160. data/lib/capybara/spec/session/evaluate_script_spec.rb +30 -9
  161. data/lib/capybara/spec/session/execute_script_spec.rb +10 -8
  162. data/lib/capybara/spec/session/fill_in_spec.rb +128 -43
  163. data/lib/capybara/spec/session/find_button_spec.rb +25 -24
  164. data/lib/capybara/spec/session/find_by_id_spec.rb +10 -9
  165. data/lib/capybara/spec/session/find_field_spec.rb +37 -41
  166. data/lib/capybara/spec/session/find_link_spec.rb +36 -17
  167. data/lib/capybara/spec/session/find_spec.rb +245 -144
  168. data/lib/capybara/spec/session/first_spec.rb +79 -51
  169. data/lib/capybara/spec/session/frame/frame_title_spec.rb +23 -0
  170. data/lib/capybara/spec/session/frame/frame_url_spec.rb +23 -0
  171. data/lib/capybara/spec/session/frame/switch_to_frame_spec.rb +33 -20
  172. data/lib/capybara/spec/session/frame/within_frame_spec.rb +50 -32
  173. data/lib/capybara/spec/session/go_back_spec.rb +2 -1
  174. data/lib/capybara/spec/session/go_forward_spec.rb +2 -1
  175. data/lib/capybara/spec/session/has_all_selectors_spec.rb +69 -0
  176. data/lib/capybara/spec/session/has_ancestor_spec.rb +46 -0
  177. data/lib/capybara/spec/session/has_any_selectors_spec.rb +25 -0
  178. data/lib/capybara/spec/session/has_button_spec.rb +70 -13
  179. data/lib/capybara/spec/session/has_css_spec.rb +272 -137
  180. data/lib/capybara/spec/session/has_current_path_spec.rb +87 -45
  181. data/lib/capybara/spec/session/has_field_spec.rb +115 -59
  182. data/lib/capybara/spec/session/has_link_spec.rb +10 -9
  183. data/lib/capybara/spec/session/has_none_selectors_spec.rb +78 -0
  184. data/lib/capybara/spec/session/has_select_spec.rb +103 -74
  185. data/lib/capybara/spec/session/has_selector_spec.rb +105 -71
  186. data/lib/capybara/spec/session/has_sibling_spec.rb +50 -0
  187. data/lib/capybara/spec/session/has_table_spec.rb +172 -5
  188. data/lib/capybara/spec/session/has_text_spec.rb +113 -61
  189. data/lib/capybara/spec/session/has_title_spec.rb +20 -14
  190. data/lib/capybara/spec/session/has_xpath_spec.rb +57 -38
  191. data/lib/capybara/spec/session/{headers.rb → headers_spec.rb} +3 -2
  192. data/lib/capybara/spec/session/html_spec.rb +14 -6
  193. data/lib/capybara/spec/session/matches_style_spec.rb +35 -0
  194. data/lib/capybara/spec/session/node_spec.rb +950 -152
  195. data/lib/capybara/spec/session/node_wrapper_spec.rb +39 -0
  196. data/lib/capybara/spec/session/refresh_spec.rb +12 -6
  197. data/lib/capybara/spec/session/reset_session_spec.rb +69 -35
  198. data/lib/capybara/spec/session/{response_code.rb → response_code_spec.rb} +2 -1
  199. data/lib/capybara/spec/session/save_and_open_page_spec.rb +3 -2
  200. data/lib/capybara/spec/session/save_and_open_screenshot_spec.rb +8 -12
  201. data/lib/capybara/spec/session/save_page_spec.rb +42 -55
  202. data/lib/capybara/spec/session/save_screenshot_spec.rb +16 -14
  203. data/lib/capybara/spec/session/screenshot_spec.rb +2 -2
  204. data/lib/capybara/spec/session/scroll_spec.rb +117 -0
  205. data/lib/capybara/spec/session/select_spec.rb +107 -80
  206. data/lib/capybara/spec/session/selectors_spec.rb +52 -19
  207. data/lib/capybara/spec/session/sibling_spec.rb +10 -10
  208. data/lib/capybara/spec/session/text_spec.rb +37 -21
  209. data/lib/capybara/spec/session/title_spec.rb +17 -5
  210. data/lib/capybara/spec/session/uncheck_spec.rb +42 -22
  211. data/lib/capybara/spec/session/unselect_spec.rb +39 -38
  212. data/lib/capybara/spec/session/visit_spec.rb +99 -32
  213. data/lib/capybara/spec/session/window/become_closed_spec.rb +24 -20
  214. data/lib/capybara/spec/session/window/current_window_spec.rb +5 -3
  215. data/lib/capybara/spec/session/window/open_new_window_spec.rb +5 -3
  216. data/lib/capybara/spec/session/window/switch_to_window_spec.rb +27 -22
  217. data/lib/capybara/spec/session/window/window_opened_by_spec.rb +12 -6
  218. data/lib/capybara/spec/session/window/window_spec.rb +97 -63
  219. data/lib/capybara/spec/session/window/windows_spec.rb +12 -10
  220. data/lib/capybara/spec/session/window/within_window_spec.rb +31 -86
  221. data/lib/capybara/spec/session/within_spec.rb +70 -44
  222. data/lib/capybara/spec/spec_helper.rb +48 -43
  223. data/lib/capybara/spec/test_app.rb +78 -40
  224. data/lib/capybara/spec/views/animated.erb +49 -0
  225. data/lib/capybara/spec/views/form.erb +130 -39
  226. data/lib/capybara/spec/views/frame_child.erb +3 -2
  227. data/lib/capybara/spec/views/frame_one.erb +1 -0
  228. data/lib/capybara/spec/views/initial_alert.erb +10 -0
  229. data/lib/capybara/spec/views/obscured.erb +47 -0
  230. data/lib/capybara/spec/views/offset.erb +32 -0
  231. data/lib/capybara/spec/views/react.erb +45 -0
  232. data/lib/capybara/spec/views/scroll.erb +20 -0
  233. data/lib/capybara/spec/views/spatial.erb +31 -0
  234. data/lib/capybara/spec/views/tables.erb +68 -1
  235. data/lib/capybara/spec/views/with_animation.erb +82 -0
  236. data/lib/capybara/spec/views/with_dragula.erb +24 -0
  237. data/lib/capybara/spec/views/with_fixed_header_footer.erb +17 -0
  238. data/lib/capybara/spec/views/with_hover.erb +6 -0
  239. data/lib/capybara/spec/views/with_hover1.erb +10 -0
  240. data/lib/capybara/spec/views/with_html.erb +69 -10
  241. data/lib/capybara/spec/views/with_html5_svg.erb +20 -0
  242. data/lib/capybara/spec/views/with_jquery_animation.erb +24 -0
  243. data/lib/capybara/spec/views/with_js.erb +37 -0
  244. data/lib/capybara/spec/views/with_jstree.erb +26 -0
  245. data/lib/capybara/spec/views/with_namespace.erb +20 -0
  246. data/lib/capybara/spec/views/with_scope_other.erb +6 -0
  247. data/lib/capybara/spec/views/with_sortable_js.erb +21 -0
  248. data/lib/capybara/spec/views/within_frames.erb +4 -1
  249. data/lib/capybara/version.rb +2 -1
  250. data/lib/capybara/window.rb +35 -33
  251. data/lib/capybara.rb +131 -104
  252. data/spec/basic_node_spec.rb +47 -34
  253. data/spec/capybara_spec.rb +53 -104
  254. data/spec/css_builder_spec.rb +101 -0
  255. data/spec/css_splitter_spec.rb +38 -0
  256. data/spec/dsl_spec.rb +81 -62
  257. data/spec/filter_set_spec.rb +27 -9
  258. data/spec/fixtures/certificate.pem +25 -0
  259. data/spec/fixtures/key.pem +27 -0
  260. data/spec/fixtures/selenium_driver_rspec_failure.rb +5 -4
  261. data/spec/fixtures/selenium_driver_rspec_success.rb +5 -4
  262. data/spec/minitest_spec.rb +49 -7
  263. data/spec/minitest_spec_spec.rb +94 -59
  264. data/spec/per_session_config_spec.rb +14 -13
  265. data/spec/rack_test_spec.rb +176 -125
  266. data/spec/regexp_dissassembler_spec.rb +250 -0
  267. data/spec/result_spec.rb +101 -46
  268. data/spec/rspec/features_spec.rb +37 -31
  269. data/spec/rspec/scenarios_spec.rb +9 -7
  270. data/spec/rspec/shared_spec_matchers.rb +448 -421
  271. data/spec/rspec/views_spec.rb +5 -3
  272. data/spec/rspec_matchers_spec.rb +27 -11
  273. data/spec/rspec_spec.rb +109 -89
  274. data/spec/sauce_spec_chrome.rb +43 -0
  275. data/spec/selector_spec.rb +396 -67
  276. data/spec/selenium_spec_chrome.rb +184 -35
  277. data/spec/selenium_spec_chrome_remote.rb +100 -0
  278. data/spec/selenium_spec_edge.rb +47 -0
  279. data/spec/selenium_spec_firefox.rb +183 -41
  280. data/spec/selenium_spec_firefox_remote.rb +80 -0
  281. data/spec/selenium_spec_ie.rb +150 -0
  282. data/spec/selenium_spec_safari.rb +148 -0
  283. data/spec/server_spec.rb +198 -99
  284. data/spec/session_spec.rb +53 -16
  285. data/spec/shared_selenium_node.rb +83 -0
  286. data/spec/shared_selenium_session.rb +486 -97
  287. data/spec/spec_helper.rb +93 -7
  288. data/spec/xpath_builder_spec.rb +93 -0
  289. metadata +338 -64
  290. data/.yard/templates_custom/default/class/html/selectors.erb +0 -38
  291. data/.yard/templates_custom/default/class/html/setup.rb +0 -17
  292. data/.yard/yard_extensions.rb +0 -78
  293. data/lib/capybara/query.rb +0 -7
  294. data/lib/capybara/rspec/compound.rb +0 -95
  295. data/lib/capybara/spec/session/assert_current_path.rb +0 -72
  296. data/lib/capybara/spec/session/assert_selector.rb +0 -148
  297. data/lib/capybara/spec/session/source_spec.rb +0 -0
  298. data/spec/selenium_spec_marionette.rb +0 -117
@@ -1,17 +1,19 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  module Capybara
3
4
  class Selector
4
5
  module Filters
5
6
  class Base
6
- def initialize(name, block, options={})
7
+ def initialize(name, matcher, block, **options)
7
8
  @name = name
9
+ @matcher = matcher
8
10
  @block = block
9
11
  @options = options
10
- @options[:valid_values] = [true,false] if options[:boolean]
12
+ @options[:valid_values] = [true, false] if options[:boolean]
11
13
  end
12
14
 
13
15
  def default?
14
- @options.has_key?(:default)
16
+ @options.key?(:default)
15
17
  end
16
18
 
17
19
  def default
@@ -19,13 +21,55 @@ module Capybara
19
21
  end
20
22
 
21
23
  def skip?(value)
22
- @options.has_key?(:skip_if) && value == @options[:skip_if]
24
+ @options.key?(:skip_if) && value == @options[:skip_if]
25
+ end
26
+
27
+ def format
28
+ @options[:format]
29
+ end
30
+
31
+ def matcher?
32
+ !@matcher.nil?
33
+ end
34
+
35
+ def boolean?
36
+ !!@options[:boolean]
37
+ end
38
+
39
+ def handles_option?(option_name)
40
+ if matcher?
41
+ @matcher.match? option_name
42
+ else
43
+ @name == option_name
44
+ end
45
+ end
46
+
47
+ private
48
+
49
+ def apply(subject, name, value, skip_value, ctx)
50
+ return skip_value if skip?(value)
51
+
52
+ unless valid_value?(value)
53
+ raise ArgumentError,
54
+ "Invalid value #{value.inspect} passed to #{self.class.name.split('::').last} #{name}" \
55
+ "#{" : #{name}" if @name.is_a?(Regexp)}"
56
+ end
57
+
58
+ if @block.arity == 2
59
+ filter_context(ctx).instance_exec(subject, value, &@block)
60
+ else
61
+ filter_context(ctx).instance_exec(subject, name, value, &@block)
62
+ end
23
63
  end
24
64
 
25
- private
65
+ def filter_context(context)
66
+ context || @block.binding.receiver
67
+ end
26
68
 
27
69
  def valid_value?(value)
28
- !@options.has_key?(:valid_values) || Array(@options[:valid_values]).include?(value)
70
+ return true unless @options.key?(:valid_values)
71
+
72
+ Array(@options[:valid_values]).any? { |valid| valid === value } # rubocop:disable Style/CaseEquality
29
73
  end
30
74
  end
31
75
  end
@@ -1,40 +1,22 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require 'capybara/selector/filters/base'
3
4
 
4
5
  module Capybara
5
6
  class Selector
6
7
  module Filters
7
8
  class ExpressionFilter < Base
8
- def apply_filter(expr, value)
9
- return expr if skip?(value)
10
-
11
- if !valid_value?(value)
12
- msg = "Invalid value #{value.inspect} passed to expression filter #{@name} - "
13
- if default?
14
- warn msg + "defaulting to #{default}"
15
- value = default
16
- else
17
- warn msg + "skipping"
18
- return expr
19
- end
20
- end
21
-
22
- @block.call(expr, value)
9
+ def apply_filter(expr, name, value, selector)
10
+ apply(expr, name, value, expr, selector)
23
11
  end
24
12
  end
25
13
 
26
14
  class IdentityExpressionFilter < ExpressionFilter
27
- def initialize
28
- end
29
-
30
- def default?
31
- false
32
- end
33
-
34
- def apply_filter(expr, _value)
35
- return expr
36
- end
15
+ def initialize(name); super(name, nil, nil); end
16
+ def default?; false; end
17
+ def matcher?; false; end
18
+ def apply_filter(expr, _name, _value, _ctx); expr; end
37
19
  end
38
20
  end
39
21
  end
40
- end
22
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'capybara/selector/filters/base'
4
+
5
+ module Capybara
6
+ class Selector
7
+ module Filters
8
+ class LocatorFilter < NodeFilter
9
+ def initialize(block, **options)
10
+ super(nil, nil, block, **options)
11
+ end
12
+
13
+ def matches?(node, value, context = nil, exact:)
14
+ apply(node, value, true, context, exact: exact, format: context&.default_format)
15
+ rescue Capybara::ElementNotFound
16
+ false
17
+ end
18
+
19
+ private
20
+
21
+ def apply(subject, value, skip_value, ctx, **options)
22
+ return skip_value if skip?(value)
23
+
24
+ filter_context(ctx).instance_exec(subject, value, **options, &@block)
25
+ end
26
+ end
27
+ end
28
+ end
29
+ end
@@ -1,25 +1,29 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require 'capybara/selector/filters/base'
3
4
 
4
5
  module Capybara
5
6
  class Selector
6
7
  module Filters
7
8
  class NodeFilter < Base
8
- def matches?(node, value)
9
- return true if skip?(value)
10
-
11
- if !valid_value?(value)
12
- msg = "Invalid value #{value.inspect} passed to filter #{@name} - "
13
- if default?
14
- warn msg + "defaulting to #{default}"
15
- value = default
16
- else
17
- warn msg + "skipping"
18
- return true
9
+ def initialize(name, matcher, block, **options)
10
+ super
11
+ @block = if boolean?
12
+ proc do |node, value|
13
+ error_cnt = errors.size
14
+ block.call(node, value).tap do |res|
15
+ add_error("Expected #{name} #{value} but it wasn't") if !res && error_cnt == errors.size
16
+ end
19
17
  end
18
+ else
19
+ block
20
20
  end
21
+ end
21
22
 
22
- @block.call(node, value)
23
+ def matches?(node, name, value, context = nil)
24
+ apply(node, name, value, true, context)
25
+ rescue Capybara::ElementNotFound
26
+ false
23
27
  end
24
28
  end
25
29
  end
@@ -0,0 +1,214 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'regexp_parser'
4
+
5
+ module Capybara
6
+ class Selector
7
+ # @api private
8
+ class RegexpDisassembler
9
+ def initialize(regexp)
10
+ @regexp = regexp
11
+ end
12
+
13
+ def alternated_substrings
14
+ @alternated_substrings ||= begin
15
+ or_strings = process(alternation: true)
16
+ remove_or_covered(or_strings)
17
+ or_strings.any?(&:empty?) ? [] : or_strings
18
+ end
19
+ end
20
+
21
+ def substrings
22
+ @substrings ||= begin
23
+ strs = process(alternation: false).first
24
+ remove_and_covered(strs)
25
+ end
26
+ end
27
+
28
+ private
29
+
30
+ def remove_and_covered(strings)
31
+ # delete_if is documented to modify the array after every block iteration - this doesn't appear to be true
32
+ # uniq the strings to prevent identical strings from removing each other
33
+ strings.uniq!
34
+
35
+ # If we have "ab" and "abcd" required - only need to check for "abcd"
36
+ strings.delete_if do |sub_string|
37
+ strings.any? do |cover_string|
38
+ next if sub_string.equal? cover_string
39
+
40
+ cover_string.include?(sub_string)
41
+ end
42
+ end
43
+ end
44
+
45
+ def remove_or_covered(or_series)
46
+ # If we are going to match `("a" and "b") or ("ade" and "bce")` it only makes sense to match ("a" and "b")
47
+
48
+ # Ensure minimum sets of strings are being or'd
49
+ or_series.each { |strs| remove_and_covered(strs) }
50
+
51
+ # Remove any of the alternated string series that fully contain any other string series
52
+ or_series.delete_if do |and_strs|
53
+ or_series.any? do |and_strs2|
54
+ next if and_strs.equal? and_strs2
55
+
56
+ remove_and_covered(and_strs + and_strs2) == and_strs
57
+ end
58
+ end
59
+ end
60
+
61
+ def process(alternation:)
62
+ strs = extract_strings(Regexp::Parser.parse(@regexp), alternation: alternation)
63
+ strs = collapse(combine(strs).map(&:flatten))
64
+ strs.each { |str| str.map!(&:upcase) } if @regexp.casefold?
65
+ strs
66
+ end
67
+
68
+ def combine(strs)
69
+ suffixes = [[]]
70
+ strs.reverse_each do |str|
71
+ if str.is_a? Set
72
+ prefixes = str.each_with_object([]) { |s, memo| memo.concat combine(s) }
73
+
74
+ result = []
75
+ prefixes.product(suffixes) { |pair| result << pair.flatten(1) }
76
+ suffixes = result
77
+ else
78
+ suffixes.each { |arr| arr.unshift str }
79
+ end
80
+ end
81
+ suffixes
82
+ end
83
+
84
+ def collapse(strs)
85
+ strs.map do |substrings|
86
+ substrings.slice_before(&:nil?).map(&:join).reject(&:empty?).uniq
87
+ end
88
+ end
89
+
90
+ def extract_strings(expression, alternation: false)
91
+ Expression.new(expression).extract_strings(alternation)
92
+ end
93
+
94
+ # @api private
95
+ class Expression
96
+ def initialize(exp)
97
+ @exp = exp
98
+ end
99
+
100
+ def extract_strings(process_alternatives)
101
+ strings = []
102
+ each do |exp|
103
+ next if exp.ignore?
104
+
105
+ next strings.push(nil) if exp.optional? && !process_alternatives
106
+
107
+ next strings.push(exp.alternative_strings) if exp.alternation? && process_alternatives
108
+
109
+ strings.concat(exp.strings(process_alternatives))
110
+ end
111
+ strings
112
+ end
113
+
114
+ protected
115
+
116
+ def alternation?
117
+ (type == :meta) && !terminal?
118
+ end
119
+
120
+ def optional?
121
+ min_repeat.zero?
122
+ end
123
+
124
+ def terminal?
125
+ @exp.terminal?
126
+ end
127
+
128
+ def strings(process_alternatives)
129
+ if indeterminate?
130
+ [nil]
131
+ elsif terminal?
132
+ terminal_strings
133
+ elsif optional?
134
+ optional_strings
135
+ else
136
+ repeated_strings(process_alternatives)
137
+ end
138
+ end
139
+
140
+ def terminal_strings
141
+ text = case @exp.type
142
+ when :literal then @exp.text
143
+ when :escape then @exp.char
144
+ else
145
+ return [nil]
146
+ end
147
+
148
+ optional? ? options_set(text) : repeat_set(text)
149
+ end
150
+
151
+ def optional_strings
152
+ options_set(extract_strings(true))
153
+ end
154
+
155
+ def repeated_strings(process_alternatives)
156
+ repeat_set extract_strings(process_alternatives)
157
+ end
158
+
159
+ def alternative_strings
160
+ alts = alternatives.map { |sub_exp| sub_exp.extract_strings(alternation: true) }
161
+ alts.all?(&:any?) ? Set.new(alts) : nil
162
+ end
163
+
164
+ def ignore?
165
+ [Regexp::Expression::Assertion::NegativeLookahead,
166
+ Regexp::Expression::Assertion::NegativeLookbehind].any? { |klass| @exp.is_a? klass }
167
+ end
168
+
169
+ private
170
+
171
+ def indeterminate?
172
+ %i[meta set].include?(type)
173
+ end
174
+
175
+ def min_repeat
176
+ @exp.repetitions.begin
177
+ end
178
+
179
+ def max_repeat
180
+ @exp.repetitions.end
181
+ end
182
+
183
+ def fixed_repeat?
184
+ min_repeat == max_repeat
185
+ end
186
+
187
+ def type
188
+ @exp.type
189
+ end
190
+
191
+ def repeat_set(str)
192
+ strs = Array(str * min_repeat)
193
+ strs.push(nil) unless fixed_repeat?
194
+ strs
195
+ end
196
+
197
+ def options_set(strs)
198
+ strs = [Set.new([[''], Array(strs)])]
199
+ strs.push(nil) unless max_repeat == 1
200
+ strs
201
+ end
202
+
203
+ def alternatives
204
+ @exp.alternatives.map { |exp| Expression.new(exp) }
205
+ end
206
+
207
+ def each
208
+ @exp.each { |exp| yield Expression.new(exp) }
209
+ end
210
+ end
211
+ private_constant :Expression
212
+ end
213
+ end
214
+ end
@@ -1,257 +1,142 @@
1
1
  # frozen_string_literal: true
2
- require 'capybara/selector/filter_set'
3
- require 'capybara/selector/css'
4
- require 'xpath'
5
-
6
- #Patch XPath to allow a nil condition in where
7
- module XPath
8
- class Renderer
9
- undef :where if method_defined?(:where)
10
- def where(on, condition)
11
- condition = condition.to_s
12
- if !condition.empty?
13
- "#{on}[#{condition}]"
14
- else
15
- "#{on}"
16
- end
17
- end
18
- end
19
- end
20
2
 
21
3
  module Capybara
22
- class Selector
23
-
24
- attr_reader :name, :format
25
-
4
+ class Selector < SimpleDelegator
26
5
  class << self
27
6
  def all
28
- @selectors ||= {}
7
+ @definitions ||= {} # rubocop:disable Naming/MemoizedInstanceVariableName
29
8
  end
30
9
 
31
- def add(name, &block)
32
- all[name.to_sym] = Capybara::Selector.new(name.to_sym, &block)
10
+ def [](name)
11
+ all.fetch(name.to_sym) { |sel_type| raise ArgumentError, "Unknown selector type (:#{sel_type})" }
12
+ end
13
+
14
+ def add(name, **options, &block)
15
+ all[name.to_sym] = Definition.new(name.to_sym, **options, &block)
33
16
  end
34
17
 
35
18
  def update(name, &block)
36
- all[name.to_sym].instance_eval(&block)
19
+ self[name].instance_eval(&block)
37
20
  end
38
21
 
39
22
  def remove(name)
40
23
  all.delete(name.to_sym)
41
24
  end
42
- end
43
25
 
44
- def initialize(name, &block)
45
- @name = name
46
- @filter_set = FilterSet.add(name){}
47
- @match = nil
48
- @label = nil
49
- @failure_message = nil
50
- @description = nil
51
- @format = nil
52
- @expression = nil
53
- @expression_filters = {}
54
- @default_visibility = nil
55
- instance_eval(&block)
26
+ def for(locator)
27
+ all.values.find { |sel| sel.match?(locator) }
28
+ end
56
29
  end
57
30
 
58
- def custom_filters
59
- @filter_set.filters
60
- end
31
+ attr_reader :errors
61
32
 
62
- def node_filters
63
- @filter_set.node_filters
33
+ def initialize(definition, config:, format:)
34
+ definition = self.class[definition] unless definition.is_a? Definition
35
+ super(definition)
36
+ @definition = definition
37
+ @config = config
38
+ @format = format
39
+ @errors = []
64
40
  end
65
41
 
66
- def expression_filters
67
- @filter_set.expression_filters
42
+ def format
43
+ @format || @definition.default_format
68
44
  end
45
+ alias_method :current_format, :format
69
46
 
70
- ##
71
- #
72
- # Define a selector by an xpath expression
73
- #
74
- # @overload xpath(*expression_filters, &block)
75
- # @param [Array<Symbol>] expression_filters ([]) Names of filters that can be implemented via this expression
76
- # @yield [locator, options] The block to use to generate the XPath expression
77
- # @yieldparam [String] locator The locator string passed to the query
78
- # @yieldparam [Hash] options The options hash passed to the query
79
- # @yieldreturn [#to_xpath, #to_s] An object that can produce an xpath expression
80
- #
81
- # @overload xpath()
82
- # @return [#call] The block that will be called to generate the XPath expression
83
- #
84
- def xpath(*expression_filters, &block)
85
- if block
86
- @format, @expression = :xpath, block
87
- expression_filters.flatten.each { |ef| custom_filters[ef] = Filters::IdentityExpressionFilter.new }
88
- end
89
- format == :xpath ? @expression : nil
47
+ def enable_aria_label
48
+ @config[:enable_aria_label]
90
49
  end
91
50
 
92
- ##
93
- #
94
- # Define a selector by a CSS selector
95
- #
96
- # @overload css(*expression_filters, &block)
97
- # @param [Array<Symbol>] expression_filters ([]) Names of filters that can be implemented via this CSS selector
98
- # @yield [locator, options] The block to use to generate the CSS selector
99
- # @yieldparam [String] locator The locator string passed to the query
100
- # @yieldparam [Hash] options The options hash passed to the query
101
- # @yieldreturn [#to_s] An object that can produce a CSS selector
102
- #
103
- # @overload css()
104
- # @return [#call] The block that will be called to generate the CSS selector
105
- #
106
- def css(*expression_filters, &block)
107
- if block
108
- @format, @expression = :css, block
109
- expression_filters.flatten.each { |ef| custom_filters[ef] = nil }
110
- end
111
- format == :css ? @expression : nil
51
+ def enable_aria_role
52
+ @config[:enable_aria_role]
112
53
  end
113
54
 
114
- ##
115
- #
116
- # Automatic selector detection
117
- #
118
- # @yield [locator] This block takes the passed in locator string and returns whether or not it matches the selector
119
- # @yieldparam [String], locator The locator string used to determin if it matches the selector
120
- # @yieldreturn [Boolean] Whether this selector matches the locator string
121
- # @return [#call] The block that will be used to detect selector match
122
- #
123
- def match(&block)
124
- @match = block if block
125
- @match
55
+ def test_id
56
+ @config[:test_id]
126
57
  end
127
58
 
128
- ##
129
- #
130
- # Set/get a descriptive label for the selector
131
- #
132
- # @overload label(label)
133
- # @param [String] label A descriptive label for this selector - used in error messages
134
- # @overload label()
135
- # @return [String] The currently set label
136
- #
137
- def label(label=nil)
138
- @label = label if label
139
- @label
140
- end
141
-
142
- ##
143
- #
144
- # Description of the selector
145
- #
146
- # @param [Hash] options The options of the query used to generate the description
147
- # @return [String] Description of the selector when used with the options passed
148
- #
149
- def description(options={})
150
- @filter_set.description(options)
151
- end
152
-
153
- def call(locator, options={})
59
+ def call(locator, **options)
154
60
  if format
155
- # @expression.call(locator, options.select {|k,v| @expression_filters.include?(k)})
156
- @expression.call(locator, options)
61
+ raise ArgumentError, "Selector #{@name} does not support #{format}" unless expressions.key?(format)
62
+
63
+ instance_exec(locator, **options, &expressions[format])
157
64
  else
158
- warn "Selector has no format"
65
+ warn 'Selector has no format'
66
+ end
67
+ ensure
68
+ unless locator_valid?(locator)
69
+ warn "Locator #{locator.class}:#{locator.inspect} for selector #{name.inspect} must #{locator_description}. This will raise an error in a future version of Capybara."
159
70
  end
160
71
  end
161
72
 
162
- ##
163
- #
164
- # Should this selector be used for the passed in locator
165
- #
166
- # This is used by the automatic selector selection mechanism when no selector type is passed to a selector query
167
- #
168
- # @param [String] locator The locator passed to the query
169
- # @return [Boolean] Whether or not to use this selector
170
- #
171
- def match?(locator)
172
- @match and @match.call(locator)
73
+ def add_error(error_msg)
74
+ errors << error_msg
173
75
  end
174
76
 
175
- ##
176
- #
177
- # Define a non-expression filter for use with this selector
178
- #
179
- # @overload filter(name, *types, options={}, &block)
180
- # @param [Symbol] name The filter name
181
- # @param [Array<Symbol>] types The types of the filter - currently valid types are [:boolean]
182
- # @param [Hash] options ({}) Options of the filter
183
- # @option options [Array<>] :valid_values Valid values for this filter
184
- # @option options :default The default value of the filter (if any)
185
- # @option options :skip_if Value of the filter that will cause it to be skipped
186
- #
187
- def filter(name, *types_and_options, &block)
188
- options = types_and_options.last.is_a?(Hash) ? types_and_options.pop.dup : {}
189
- types_and_options.each { |k| options[k] = true }
190
- custom_filters[name] = Filters::NodeFilter.new(name, block, options)
77
+ def expression_for(name, locator, config: @config, format: current_format, **options)
78
+ Selector.new(name, config: config, format: format).call(locator, **options)
191
79
  end
192
80
 
193
- def expression_filter(name, *types_and_options, &block)
194
- options = types_and_options.last.is_a?(Hash) ? types_and_options.pop.dup : {}
195
- types_and_options.each { |k| options[k] = true }
196
- custom_filters[name] = Filters::ExpressionFilter.new(name, block, options)
81
+ # @api private
82
+ def with_filter_errors(errors)
83
+ old_errors = @errors
84
+ @errors = errors
85
+ yield
86
+ ensure
87
+ @errors = old_errors
197
88
  end
198
89
 
199
- def filter_set(name, filters_to_use = nil)
200
- f_set = FilterSet.all[name]
201
- f_set.filters.each do |n, filter|
202
- custom_filters[n] = filter if filters_to_use.nil? || filters_to_use.include?(n)
203
- end
204
-
205
- f_set.descriptions.each { |desc| @filter_set.describe(&desc) }
90
+ # @api private
91
+ def builder(expr = nil)
92
+ case format
93
+ when :css
94
+ Capybara::Selector::CSSBuilder
95
+ when :xpath
96
+ Capybara::Selector::XPathBuilder
97
+ else
98
+ raise NotImplementedError, "No builder exists for selector of type #{default_format}"
99
+ end.new(expr)
206
100
  end
207
101
 
208
- def describe &block
209
- @filter_set.describe(&block)
210
- end
102
+ private
211
103
 
212
- ##
213
- #
214
- # Set the default visibility mode that shouble be used if no visibile option is passed when using the selector.
215
- # If not specified will default to the behavior indicated by Capybara.ignore_hidden_elements
216
- #
217
- # @param [Symbol] default_visibility Only find elements with the specified visibility:
218
- # * :all - finds visible and invisible elements.
219
- # * :hidden - only finds invisible elements.
220
- # * :visible - only finds visible elements.
221
- def visible(default_visibility)
222
- @default_visibility = default_visibility
104
+ def locator_description
105
+ locator_types.group_by { |lt| lt.is_a? Symbol }.map do |symbol, types_or_methods|
106
+ if symbol
107
+ "respond to #{types_or_methods.join(' or ')}"
108
+ else
109
+ "be an instance of #{types_or_methods.join(' or ')}"
110
+ end
111
+ end.join(' or ')
223
112
  end
224
113
 
225
- def default_visibility(fallback = Capybara.ignore_hidden_elements)
226
- if @default_visibility.nil?
227
- fallback
228
- else
229
- @default_visibility
114
+ def locator_valid?(locator)
115
+ return true unless locator && locator_types
116
+
117
+ locator_types&.any? do |type_or_method|
118
+ type_or_method.is_a?(Symbol) ? locator.respond_to?(type_or_method) : type_or_method === locator # rubocop:disable Style/CaseEquality
230
119
  end
231
120
  end
232
121
 
233
- private
122
+ def locate_field(xpath, locator, **_options)
123
+ return xpath if locator.nil?
234
124
 
235
- def locate_field(xpath, locator, options={})
236
- locate_xpath = xpath #need to save original xpath for the label wrap
237
- if locator
238
- locator = locator.to_s
239
- attr_matchers = XPath.attr(:id).equals(locator).or(
240
- XPath.attr(:name).equals(locator)).or(
241
- XPath.attr(:placeholder).equals(locator)).or(
242
- XPath.attr(:id).equals(XPath.anywhere(:label)[XPath.string.n.is(locator)].attr(:for)))
243
- attr_matchers = attr_matchers.or XPath.attr(:'aria-label').is(locator) if options[:enable_aria_label]
125
+ locate_xpath = xpath # Need to save original xpath for the label wrap
126
+ locator = locator.to_s
127
+ attr_matchers = [XPath.attr(:id) == locator,
128
+ XPath.attr(:name) == locator,
129
+ XPath.attr(:placeholder) == locator,
130
+ XPath.attr(:id) == XPath.anywhere(:label)[XPath.string.n.is(locator)].attr(:for)].reduce(:|)
131
+ attr_matchers |= XPath.attr(:'aria-label').is(locator) if enable_aria_label
132
+ attr_matchers |= XPath.attr(test_id) == locator if test_id
244
133
 
245
- locate_xpath = locate_xpath[attr_matchers]
246
- locate_xpath = locate_xpath.union(XPath.descendant(:label)[XPath.string.n.is(locator)].descendant(xpath))
247
- end
248
-
249
- # locate_xpath = [:name, :placeholder].inject(locate_xpath) { |memo, ef| memo[find_by_attr(ef, options[ef])] }
250
- locate_xpath
134
+ locate_xpath = locate_xpath[attr_matchers]
135
+ locate_xpath + locate_label(locator).descendant(xpath)
251
136
  end
252
137
 
253
- def describe_all_expression_filters(opts={})
254
- expression_filters.map { |ef| " with #{ef} #{opts[ef]}" if opts.has_key?(ef) }.join
138
+ def locate_label(locator)
139
+ XPath.descendant(:label)[XPath.string.n.is(locator)]
255
140
  end
256
141
 
257
142
  def find_by_attr(attribute, value)
@@ -259,18 +144,12 @@ module Capybara
259
144
  if respond_to?(finder_name, true)
260
145
  send(finder_name, value)
261
146
  else
262
- value ? XPath.attr(attribute).equals(value) : nil
147
+ value ? XPath.attr(attribute) == value : nil
263
148
  end
264
149
  end
265
150
 
266
151
  def find_by_class_attr(classes)
267
- if classes
268
- Array(classes).map do |klass|
269
- "contains(concat(' ',normalize-space(@class),' '),' #{klass} ')"
270
- end.join(" and ").to_sym
271
- else
272
- nil
273
- end
152
+ Array(classes).map { |klass| XPath.attr(:class).contains_word(klass) }.reduce(:&)
274
153
  end
275
154
  end
276
155
  end