module Capybara::SpecHelper
def be_an_invalid_element_error(session)
def be_an_invalid_element_error(session) satisfy { |error| session.driver.invalid_element_errors.any? { |e| error.is_a? e } } end
def configure(config)
def configure(config) config.filter_run_excluding requires: method(:filter).to_proc config.before { Capybara::SpecHelper.reset! } config.after { Capybara::SpecHelper.reset! } config.shared_context_metadata_behavior = :apply_to_host_groups end
def extract_content_type(session)
def extract_content_type(session) expect(session).to have_xpath("//pre[@id='content_type']") Capybara::HTML(session.body).xpath("//pre[@id='content_type']").first.text end
def extract_results(session)
def extract_results(session) expect(session).to have_xpath("//pre[@id='results']") perms = [(::Sinatra::IndifferentHash if defined? ::Sinatra::IndifferentHash)].compact results = Capybara::HTML(session.body).xpath("//pre[@id='results']").first.inner_html.lstrip YAML.safe_load results, permitted_classes: perms end
def filter(requires, metadata)
def filter(requires, metadata) if requires && metadata[:capybara_skip] requires.any? do |require| metadata[:capybara_skip].include?(require) end else false end end
def quietly(&block)
def quietly(&block) silence_stream($stdout) do silence_stream($stderr, &block) end end
def reset!
def reset! Capybara.app = TestApp Capybara.app_host = nil Capybara.default_selector = :xpath Capybara.default_max_wait_time = 1 Capybara.default_retry_interval = 0.01 Capybara.ignore_hidden_elements = true Capybara.exact = false Capybara.raise_server_errors = true Capybara.visible_text_only = false Capybara.match = :smart Capybara.enable_aria_label = false Capybara.enable_aria_role = false Capybara.default_set_options = {} Capybara.disable_animation = false Capybara.test_id = nil Capybara.predicates_wait = true Capybara.default_normalize_ws = false Capybara.use_html5_parsing = !ENV['HTML5_PARSING'].nil? Capybara.w3c_click_offset = true reset_threadsafe end
def reset_threadsafe(bool: false, session: nil)
def reset_threadsafe(bool: false, session: nil) # Work around limit on when threadsafe can be changed Capybara::Session.class_variable_set(:@@instance_created, false) # rubocop:disable Style/ClassVars Capybara.threadsafe = bool session = session.current_session if session.respond_to?(:current_session) session&.instance_variable_set(:@config, nil) end
def run_specs(session, name, **options, &filter_block)
def run_specs(session, name, **options, &filter_block) specs = @specs RSpec.describe Capybara::Session, name, options do include Capybara::SpecHelper include Capybara::RSpecMatchers before do |example| @session = session instance_exec(example, &filter_block) if filter_block end after do session.reset_session! end before :each, :psc do SpecHelper.reset_threadsafe(bool: true, session: session) end after psc: true do SpecHelper.reset_threadsafe(session: session) end before :each, :exact_false do Capybara.exact = false end specs.each do |spec_name, spec_options, block| describe spec_name, *spec_options do class_eval(&block) end end end end
def silence_stream(stream)
def silence_stream(stream) old_stream = stream.dup stream.reopen(RbConfig::CONFIG['host_os'].match?(/rmswin|mingw/) ? 'NUL:' : '/dev/null') stream.sync = true yield ensure stream.reopen(old_stream) end
def spec(name, *options, &block)
def spec(name, *options, &block) @specs ||= [] @specs << [name, options, block] end
def with_os_path_separators(path)
def with_os_path_separators(path) Gem.win_platform? ? path.to_s.tr('/', '\\') : path.to_s end