2016-03-07 19:52:19 -05:00
# frozen_string_literal: true
2012-12-05 13:46:00 -05:00
2012-08-01 07:24:43 -04:00
module Capybara
2013-03-04 16:35:11 -05:00
# @api private
2012-08-01 07:24:43 -04:00
module Helpers
2019-05-30 18:07:27 -04:00
module_function
2012-08-01 07:24:43 -04:00
2013-03-17 10:48:04 -04:00
##
2018-03-12 13:23:32 -04:00
# @deprecated
2013-03-17 10:48:04 -04:00
# Normalizes whitespace space by stripping leading and trailing
# whitespace and replacing sequences of whitespace characters
# with a single space.
#
# @param [String] text Text to normalize
# @return [String] Normalized text
#
def normalize_whitespace ( text )
2020-09-06 15:21:53 -04:00
Capybara :: Helpers . warn 'DEPRECATED: Capybara::Helpers::normalize_whitespace is deprecated, please update your driver'
2013-03-17 10:48:04 -04:00
text . to_s . gsub ( / [[:space:]]+ / , ' ' ) . strip
end
2013-02-19 17:57:34 -05:00
2013-03-17 10:48:04 -04:00
##
#
# Escapes any characters that would have special meaning in a regexp
# if text is not a regexp
#
# @param [String] text Text to escape
2018-02-27 18:46:13 -05:00
# @param [Boolean] exact (false) Whether or not this should be an exact text match
# @param [Fixnum, Boolean, nil] options Options passed to Regexp.new when creating the Regexp
# @return [Regexp] Regexp to match the passed in text and options
2013-03-17 10:48:04 -04:00
#
2018-03-05 17:57:33 -05:00
def to_regexp ( text , exact : false , all_whitespace : false , options : nil )
2018-01-13 16:06:03 -05:00
return text if text . is_a? ( Regexp )
2018-03-05 17:57:33 -05:00
escaped = Regexp . escape ( text )
2018-07-10 17:18:39 -04:00
escaped = escaped . gsub ( '\\ ' , '[[:blank:]]' ) if all_whitespace
2018-01-13 16:06:03 -05:00
escaped = " \\ A #{ escaped } \\ z " if exact
2018-02-27 18:46:13 -05:00
Regexp . new ( escaped , options )
2013-03-17 10:48:04 -04:00
end
##
#
# Injects a `<base>` tag into the given HTML code, pointing to
2019-05-21 20:48:31 -04:00
# {Capybara.configure asset_host}.
2013-03-17 10:48:04 -04:00
#
# @param [String] html HTML code to inject into
2018-02-27 18:46:13 -05:00
# @param [URL] host (Capybara.asset_host) The host from which assets should be loaded
2013-03-28 22:15:07 -04:00
# @return [String] The modified HTML code
2013-03-17 10:48:04 -04:00
#
2018-02-27 18:46:13 -05:00
def inject_asset_host ( html , host : Capybara . asset_host )
2018-07-10 17:18:39 -04:00
if host && Nokogiri :: HTML ( html ) . css ( 'base' ) . empty?
2018-10-16 19:06:50 -04:00
html . match ( / <head[^<]*?> / ) do | m |
return html . clone . insert m . end ( 0 ) , " <base href=' #{ host } ' /> "
end
2013-02-19 17:57:34 -05:00
end
2015-05-03 22:13:35 -04:00
html
2012-08-01 07:24:43 -04:00
end
2013-03-03 18:04:23 -05:00
2013-03-17 10:48:04 -04:00
##
#
# A poor man's `pluralize`. Given two declensions, one singular and one
# plural, as well as a count, this will pick the correct declension. This
2014-03-19 19:28:26 -04:00
# way we can generate grammatically correct error message.
2013-03-17 10:48:04 -04:00
#
# @param [String] singular The singular form of the word
# @param [String] plural The plural form of the word
# @param [Integer] count The number of items
#
def declension ( singular , plural , count )
2018-01-13 16:06:03 -05:00
count == 1 ? singular : plural
2013-03-03 18:04:23 -05:00
end
2015-04-14 17:41:01 -04:00
2020-07-05 16:18:37 -04:00
def filter_backtrace ( trace )
return 'No backtrace' unless trace
filter = %r{ lib/capybara/|lib/rspec/|lib/minitest/ }
new_trace = trace . take_while { | line | line !~ filter }
new_trace = trace . reject { | line | line =~ filter } if new_trace . empty?
new_trace = trace . dup if new_trace . empty?
new_trace . first . split ( / :in / , 2 ) . first
end
2020-09-06 15:21:53 -04:00
def warn ( message , uplevel : 1 )
2020-09-06 15:31:16 -04:00
return Kernel . warn ( message , uplevel : uplevel ) if RUBY_VERSION > = '2.6'
# TODO: Remove when we drop support for Ruby 2.5
2020-09-06 16:22:09 -04:00
# Workaround for emulating `warn '...', uplevel: n` in Ruby 2.5 or lower.
2020-09-06 15:21:53 -04:00
if ( match = / ^(?<file>.+?):(?<line> \ d+)(?::in `.*')? / . match ( caller [ uplevel ] ) )
location = [ match [ :file ] , match [ :line ] ] . join ( ':' )
Kernel . warn " #{ location } : #{ message } "
else
Kernel . warn message
end
end
2015-04-14 17:41:01 -04:00
if defined? ( Process :: CLOCK_MONOTONIC )
2018-02-27 18:46:13 -05:00
def monotonic_time ; Process . clock_gettime Process :: CLOCK_MONOTONIC ; end
2015-04-14 17:41:01 -04:00
else
2018-02-27 18:46:13 -05:00
def monotonic_time ; Time . now . to_f ; end
2015-04-14 17:41:01 -04:00
end
2018-06-06 13:51:25 -04:00
def timer ( expire_in : )
Timer . new ( expire_in )
end
class Timer
def initialize ( expire_in )
@start = current
@expire_in = expire_in
end
def expired?
2019-10-15 21:02:35 -04:00
if stalled?
raise Capybara :: FrozenInTime , 'Time appears to be frozen. Capybara does not work with libraries which freeze time, consider using time travelling instead'
end
2018-09-24 12:43:46 -04:00
2018-06-06 13:51:25 -04:00
current - @start > = @expire_in
end
def stalled?
@start == current
end
private
def current
Capybara :: Helpers . monotonic_time
end
end
2013-03-03 18:04:23 -05:00
end
2012-08-01 07:24:43 -04:00
end