rspec-expectations 3.8.6
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- checksums.yaml.gz.sig +0 -0
- data.tar.gz.sig +5 -0
- data/.document +5 -0
- data/.yardopts +6 -0
- data/Changelog.md +1156 -0
- data/LICENSE.md +25 -0
- data/README.md +305 -0
- data/lib/rspec/expectations.rb +82 -0
- data/lib/rspec/expectations/block_snippet_extractor.rb +253 -0
- data/lib/rspec/expectations/configuration.rb +215 -0
- data/lib/rspec/expectations/expectation_target.rb +127 -0
- data/lib/rspec/expectations/fail_with.rb +39 -0
- data/lib/rspec/expectations/failure_aggregator.rb +194 -0
- data/lib/rspec/expectations/handler.rb +170 -0
- data/lib/rspec/expectations/minitest_integration.rb +58 -0
- data/lib/rspec/expectations/syntax.rb +132 -0
- data/lib/rspec/expectations/version.rb +8 -0
- data/lib/rspec/matchers.rb +1034 -0
- data/lib/rspec/matchers/aliased_matcher.rb +116 -0
- data/lib/rspec/matchers/built_in.rb +52 -0
- data/lib/rspec/matchers/built_in/all.rb +86 -0
- data/lib/rspec/matchers/built_in/base_matcher.rb +193 -0
- data/lib/rspec/matchers/built_in/be.rb +288 -0
- data/lib/rspec/matchers/built_in/be_between.rb +77 -0
- data/lib/rspec/matchers/built_in/be_instance_of.rb +26 -0
- data/lib/rspec/matchers/built_in/be_kind_of.rb +20 -0
- data/lib/rspec/matchers/built_in/be_within.rb +72 -0
- data/lib/rspec/matchers/built_in/change.rb +428 -0
- data/lib/rspec/matchers/built_in/compound.rb +271 -0
- data/lib/rspec/matchers/built_in/contain_exactly.rb +302 -0
- data/lib/rspec/matchers/built_in/cover.rb +24 -0
- data/lib/rspec/matchers/built_in/eq.rb +40 -0
- data/lib/rspec/matchers/built_in/eql.rb +34 -0
- data/lib/rspec/matchers/built_in/equal.rb +81 -0
- data/lib/rspec/matchers/built_in/exist.rb +90 -0
- data/lib/rspec/matchers/built_in/has.rb +103 -0
- data/lib/rspec/matchers/built_in/have_attributes.rb +114 -0
- data/lib/rspec/matchers/built_in/include.rb +149 -0
- data/lib/rspec/matchers/built_in/match.rb +106 -0
- data/lib/rspec/matchers/built_in/operators.rb +128 -0
- data/lib/rspec/matchers/built_in/output.rb +200 -0
- data/lib/rspec/matchers/built_in/raise_error.rb +230 -0
- data/lib/rspec/matchers/built_in/respond_to.rb +165 -0
- data/lib/rspec/matchers/built_in/satisfy.rb +60 -0
- data/lib/rspec/matchers/built_in/start_or_end_with.rb +94 -0
- data/lib/rspec/matchers/built_in/throw_symbol.rb +132 -0
- data/lib/rspec/matchers/built_in/yield.rb +432 -0
- data/lib/rspec/matchers/composable.rb +171 -0
- data/lib/rspec/matchers/dsl.rb +527 -0
- data/lib/rspec/matchers/english_phrasing.rb +58 -0
- data/lib/rspec/matchers/expecteds_for_multiple_diffs.rb +73 -0
- data/lib/rspec/matchers/fail_matchers.rb +42 -0
- data/lib/rspec/matchers/generated_descriptions.rb +41 -0
- data/lib/rspec/matchers/matcher_delegator.rb +35 -0
- data/lib/rspec/matchers/matcher_protocol.rb +99 -0
- metadata +215 -0
- metadata.gz.sig +0 -0
@@ -0,0 +1,116 @@
|
|
1
|
+
module RSpec
|
2
|
+
module Matchers
|
3
|
+
# Decorator that wraps a matcher and overrides `description`
|
4
|
+
# using the provided block in order to support an alias
|
5
|
+
# of a matcher. This is intended for use when composing
|
6
|
+
# matchers, so that you can use an expression like
|
7
|
+
# `include( a_value_within(0.1).of(3) )` rather than
|
8
|
+
# `include( be_within(0.1).of(3) )`, and have the corresponding
|
9
|
+
# description read naturally.
|
10
|
+
#
|
11
|
+
# @api private
|
12
|
+
class AliasedMatcher < MatcherDelegator
|
13
|
+
def initialize(base_matcher, description_block)
|
14
|
+
@description_block = description_block
|
15
|
+
super(base_matcher)
|
16
|
+
end
|
17
|
+
|
18
|
+
# Forward messages on to the wrapped matcher.
|
19
|
+
# Since many matchers provide a fluent interface
|
20
|
+
# (e.g. `a_value_within(0.1).of(3)`), we need to wrap
|
21
|
+
# the returned value if it responds to `description`,
|
22
|
+
# so that our override can be applied when it is eventually
|
23
|
+
# used.
|
24
|
+
def method_missing(*)
|
25
|
+
return_val = super
|
26
|
+
return return_val unless RSpec::Matchers.is_a_matcher?(return_val)
|
27
|
+
self.class.new(return_val, @description_block)
|
28
|
+
end
|
29
|
+
|
30
|
+
# Provides the description of the aliased matcher. Aliased matchers
|
31
|
+
# are designed to behave identically to the original matcher except
|
32
|
+
# for the description and failure messages. The description is different
|
33
|
+
# to reflect the aliased name.
|
34
|
+
#
|
35
|
+
# @api private
|
36
|
+
def description
|
37
|
+
@description_block.call(super)
|
38
|
+
end
|
39
|
+
|
40
|
+
# Provides the failure_message of the aliased matcher. Aliased matchers
|
41
|
+
# are designed to behave identically to the original matcher except
|
42
|
+
# for the description and failure messages. The failure_message is different
|
43
|
+
# to reflect the aliased name.
|
44
|
+
#
|
45
|
+
# @api private
|
46
|
+
def failure_message
|
47
|
+
@description_block.call(super)
|
48
|
+
end
|
49
|
+
|
50
|
+
# Provides the failure_message_when_negated of the aliased matcher. Aliased matchers
|
51
|
+
# are designed to behave identically to the original matcher except
|
52
|
+
# for the description and failure messages. The failure_message_when_negated is different
|
53
|
+
# to reflect the aliased name.
|
54
|
+
#
|
55
|
+
# @api private
|
56
|
+
def failure_message_when_negated
|
57
|
+
@description_block.call(super)
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
# Decorator used for matchers that have special implementations of
|
62
|
+
# operators like `==` and `===`.
|
63
|
+
# @private
|
64
|
+
class AliasedMatcherWithOperatorSupport < AliasedMatcher
|
65
|
+
# We undef these so that they get delegated via `method_missing`.
|
66
|
+
undef ==
|
67
|
+
undef ===
|
68
|
+
end
|
69
|
+
|
70
|
+
# @private
|
71
|
+
class AliasedNegatedMatcher < AliasedMatcher
|
72
|
+
def matches?(*args, &block)
|
73
|
+
if @base_matcher.respond_to?(:does_not_match?)
|
74
|
+
@base_matcher.does_not_match?(*args, &block)
|
75
|
+
else
|
76
|
+
!super
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
def does_not_match?(*args, &block)
|
81
|
+
@base_matcher.matches?(*args, &block)
|
82
|
+
end
|
83
|
+
|
84
|
+
def failure_message
|
85
|
+
optimal_failure_message(__method__, :failure_message_when_negated)
|
86
|
+
end
|
87
|
+
|
88
|
+
def failure_message_when_negated
|
89
|
+
optimal_failure_message(__method__, :failure_message)
|
90
|
+
end
|
91
|
+
|
92
|
+
private
|
93
|
+
|
94
|
+
DefaultFailureMessages = BuiltIn::BaseMatcher::DefaultFailureMessages
|
95
|
+
|
96
|
+
# For a matcher that uses the default failure messages, we prefer to
|
97
|
+
# use the override provided by the `description_block`, because it
|
98
|
+
# includes the phrasing that the user has expressed a preference for
|
99
|
+
# by going through the effort of defining a negated matcher.
|
100
|
+
#
|
101
|
+
# However, if the override didn't actually change anything, then we
|
102
|
+
# should return the opposite failure message instead -- the overriden
|
103
|
+
# message is going to be confusing if we return it as-is, as it represents
|
104
|
+
# the non-negated failure message for a negated match (or vice versa).
|
105
|
+
def optimal_failure_message(same, inverted)
|
106
|
+
if DefaultFailureMessages.has_default_failure_messages?(@base_matcher)
|
107
|
+
base_message = @base_matcher.__send__(same)
|
108
|
+
overriden = @description_block.call(base_message)
|
109
|
+
return overriden if overriden != base_message
|
110
|
+
end
|
111
|
+
|
112
|
+
@base_matcher.__send__(inverted)
|
113
|
+
end
|
114
|
+
end
|
115
|
+
end
|
116
|
+
end
|
@@ -0,0 +1,52 @@
|
|
1
|
+
RSpec::Support.require_rspec_matchers "built_in/base_matcher"
|
2
|
+
|
3
|
+
module RSpec
|
4
|
+
module Matchers
|
5
|
+
# Container module for all built-in matchers. The matcher classes are here
|
6
|
+
# (rather than directly under `RSpec::Matchers`) in order to prevent name
|
7
|
+
# collisions, since `RSpec::Matchers` gets included into the user's namespace.
|
8
|
+
#
|
9
|
+
# Autoloading is used to delay when the matcher classes get loaded, allowing
|
10
|
+
# rspec-matchers to boot faster, and avoiding loading matchers the user is
|
11
|
+
# not using.
|
12
|
+
module BuiltIn
|
13
|
+
autoload :BeAKindOf, 'rspec/matchers/built_in/be_kind_of'
|
14
|
+
autoload :BeAnInstanceOf, 'rspec/matchers/built_in/be_instance_of'
|
15
|
+
autoload :BeBetween, 'rspec/matchers/built_in/be_between'
|
16
|
+
autoload :Be, 'rspec/matchers/built_in/be'
|
17
|
+
autoload :BeComparedTo, 'rspec/matchers/built_in/be'
|
18
|
+
autoload :BeFalsey, 'rspec/matchers/built_in/be'
|
19
|
+
autoload :BeNil, 'rspec/matchers/built_in/be'
|
20
|
+
autoload :BePredicate, 'rspec/matchers/built_in/be'
|
21
|
+
autoload :BeTruthy, 'rspec/matchers/built_in/be'
|
22
|
+
autoload :BeWithin, 'rspec/matchers/built_in/be_within'
|
23
|
+
autoload :Change, 'rspec/matchers/built_in/change'
|
24
|
+
autoload :Compound, 'rspec/matchers/built_in/compound'
|
25
|
+
autoload :ContainExactly, 'rspec/matchers/built_in/contain_exactly'
|
26
|
+
autoload :Cover, 'rspec/matchers/built_in/cover'
|
27
|
+
autoload :EndWith, 'rspec/matchers/built_in/start_or_end_with'
|
28
|
+
autoload :Eq, 'rspec/matchers/built_in/eq'
|
29
|
+
autoload :Eql, 'rspec/matchers/built_in/eql'
|
30
|
+
autoload :Equal, 'rspec/matchers/built_in/equal'
|
31
|
+
autoload :Exist, 'rspec/matchers/built_in/exist'
|
32
|
+
autoload :Has, 'rspec/matchers/built_in/has'
|
33
|
+
autoload :HaveAttributes, 'rspec/matchers/built_in/have_attributes'
|
34
|
+
autoload :Include, 'rspec/matchers/built_in/include'
|
35
|
+
autoload :All, 'rspec/matchers/built_in/all'
|
36
|
+
autoload :Match, 'rspec/matchers/built_in/match'
|
37
|
+
autoload :NegativeOperatorMatcher, 'rspec/matchers/built_in/operators'
|
38
|
+
autoload :OperatorMatcher, 'rspec/matchers/built_in/operators'
|
39
|
+
autoload :Output, 'rspec/matchers/built_in/output'
|
40
|
+
autoload :PositiveOperatorMatcher, 'rspec/matchers/built_in/operators'
|
41
|
+
autoload :RaiseError, 'rspec/matchers/built_in/raise_error'
|
42
|
+
autoload :RespondTo, 'rspec/matchers/built_in/respond_to'
|
43
|
+
autoload :Satisfy, 'rspec/matchers/built_in/satisfy'
|
44
|
+
autoload :StartWith, 'rspec/matchers/built_in/start_or_end_with'
|
45
|
+
autoload :ThrowSymbol, 'rspec/matchers/built_in/throw_symbol'
|
46
|
+
autoload :YieldControl, 'rspec/matchers/built_in/yield'
|
47
|
+
autoload :YieldSuccessiveArgs, 'rspec/matchers/built_in/yield'
|
48
|
+
autoload :YieldWithArgs, 'rspec/matchers/built_in/yield'
|
49
|
+
autoload :YieldWithNoArgs, 'rspec/matchers/built_in/yield'
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
@@ -0,0 +1,86 @@
|
|
1
|
+
module RSpec
|
2
|
+
module Matchers
|
3
|
+
module BuiltIn
|
4
|
+
# @api private
|
5
|
+
# Provides the implementation for `all`.
|
6
|
+
# Not intended to be instantiated directly.
|
7
|
+
class All < BaseMatcher
|
8
|
+
# @private
|
9
|
+
attr_reader :matcher, :failed_objects
|
10
|
+
|
11
|
+
def initialize(matcher)
|
12
|
+
@matcher = matcher
|
13
|
+
@failed_objects = {}
|
14
|
+
end
|
15
|
+
|
16
|
+
# @private
|
17
|
+
def does_not_match?(_actual)
|
18
|
+
raise NotImplementedError, '`expect().not_to all( matcher )` is not supported.'
|
19
|
+
end
|
20
|
+
|
21
|
+
# @api private
|
22
|
+
# @return [String]
|
23
|
+
def failure_message
|
24
|
+
unless iterable?
|
25
|
+
return "#{improve_hash_formatting(super)}, but was not iterable"
|
26
|
+
end
|
27
|
+
|
28
|
+
all_messages = [improve_hash_formatting(super)]
|
29
|
+
failed_objects.each do |index, matcher_failure_message|
|
30
|
+
all_messages << failure_message_for_item(index, matcher_failure_message)
|
31
|
+
end
|
32
|
+
all_messages.join("\n\n")
|
33
|
+
end
|
34
|
+
|
35
|
+
# @api private
|
36
|
+
# @return [String]
|
37
|
+
def description
|
38
|
+
improve_hash_formatting "all #{description_of matcher}"
|
39
|
+
end
|
40
|
+
|
41
|
+
private
|
42
|
+
|
43
|
+
def match(_expected, _actual)
|
44
|
+
return false unless iterable?
|
45
|
+
|
46
|
+
index_failed_objects
|
47
|
+
failed_objects.empty?
|
48
|
+
end
|
49
|
+
|
50
|
+
def index_failed_objects
|
51
|
+
actual.each_with_index do |actual_item, index|
|
52
|
+
cloned_matcher = matcher.clone
|
53
|
+
matches = cloned_matcher.matches?(actual_item)
|
54
|
+
failed_objects[index] = cloned_matcher.failure_message unless matches
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
def failure_message_for_item(index, failure_message)
|
59
|
+
failure_message = indent_multiline_message(add_new_line_if_needed(failure_message))
|
60
|
+
indent_multiline_message("object at index #{index} failed to match:#{failure_message}")
|
61
|
+
end
|
62
|
+
|
63
|
+
def add_new_line_if_needed(message)
|
64
|
+
message.start_with?("\n") ? message : "\n#{message}"
|
65
|
+
end
|
66
|
+
|
67
|
+
def indent_multiline_message(message)
|
68
|
+
message = message.sub(/\n+\z/, '')
|
69
|
+
message.lines.map do |line|
|
70
|
+
line =~ /\S/ ? ' ' + line : line
|
71
|
+
end.join
|
72
|
+
end
|
73
|
+
|
74
|
+
def initialize_copy(other)
|
75
|
+
@matcher = @matcher.clone
|
76
|
+
@failed_objects = @failed_objects.clone
|
77
|
+
super
|
78
|
+
end
|
79
|
+
|
80
|
+
def iterable?
|
81
|
+
@actual.respond_to?(:each_with_index)
|
82
|
+
end
|
83
|
+
end
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
@@ -0,0 +1,193 @@
|
|
1
|
+
module RSpec
|
2
|
+
module Matchers
|
3
|
+
module BuiltIn
|
4
|
+
# @api private
|
5
|
+
#
|
6
|
+
# Used _internally_ as a base class for matchers that ship with
|
7
|
+
# rspec-expectations and rspec-rails.
|
8
|
+
#
|
9
|
+
# ### Warning:
|
10
|
+
#
|
11
|
+
# This class is for internal use, and subject to change without notice.
|
12
|
+
# We strongly recommend that you do not base your custom matchers on this
|
13
|
+
# class. If/when this changes, we will announce it and remove this warning.
|
14
|
+
class BaseMatcher
|
15
|
+
include RSpec::Matchers::Composable
|
16
|
+
|
17
|
+
# @api private
|
18
|
+
# Used to detect when no arg is passed to `initialize`.
|
19
|
+
# `nil` cannot be used because it's a valid value to pass.
|
20
|
+
UNDEFINED = Object.new.freeze
|
21
|
+
|
22
|
+
# @private
|
23
|
+
attr_reader :actual, :expected, :rescued_exception
|
24
|
+
|
25
|
+
# @private
|
26
|
+
attr_writer :matcher_name
|
27
|
+
|
28
|
+
def initialize(expected=UNDEFINED)
|
29
|
+
@expected = expected unless UNDEFINED.equal?(expected)
|
30
|
+
end
|
31
|
+
|
32
|
+
# @api private
|
33
|
+
# Indicates if the match is successful. Delegates to `match`, which
|
34
|
+
# should be defined on a subclass. Takes care of consistently
|
35
|
+
# initializing the `actual` attribute.
|
36
|
+
def matches?(actual)
|
37
|
+
@actual = actual
|
38
|
+
match(expected, actual)
|
39
|
+
end
|
40
|
+
|
41
|
+
# @api private
|
42
|
+
# Used to wrap a block of code that will indicate failure by
|
43
|
+
# raising one of the named exceptions.
|
44
|
+
#
|
45
|
+
# This is used by rspec-rails for some of its matchers that
|
46
|
+
# wrap rails' assertions.
|
47
|
+
def match_unless_raises(*exceptions)
|
48
|
+
exceptions.unshift Exception if exceptions.empty?
|
49
|
+
begin
|
50
|
+
yield
|
51
|
+
true
|
52
|
+
rescue *exceptions => @rescued_exception
|
53
|
+
false
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
# @api private
|
58
|
+
# Generates a description using {EnglishPhrasing}.
|
59
|
+
# @return [String]
|
60
|
+
def description
|
61
|
+
desc = EnglishPhrasing.split_words(self.class.matcher_name)
|
62
|
+
desc << EnglishPhrasing.list(@expected) if defined?(@expected)
|
63
|
+
desc
|
64
|
+
end
|
65
|
+
|
66
|
+
# @api private
|
67
|
+
# Matchers are not diffable by default. Override this to make your
|
68
|
+
# subclass diffable.
|
69
|
+
def diffable?
|
70
|
+
false
|
71
|
+
end
|
72
|
+
|
73
|
+
# @api private
|
74
|
+
# Most matchers are value matchers (i.e. meant to work with `expect(value)`)
|
75
|
+
# rather than block matchers (i.e. meant to work with `expect { }`), so
|
76
|
+
# this defaults to false. Block matchers must override this to return true.
|
77
|
+
def supports_block_expectations?
|
78
|
+
false
|
79
|
+
end
|
80
|
+
|
81
|
+
# @api private
|
82
|
+
def expects_call_stack_jump?
|
83
|
+
false
|
84
|
+
end
|
85
|
+
|
86
|
+
# @private
|
87
|
+
def expected_formatted
|
88
|
+
RSpec::Support::ObjectFormatter.format(@expected)
|
89
|
+
end
|
90
|
+
|
91
|
+
# @private
|
92
|
+
def actual_formatted
|
93
|
+
RSpec::Support::ObjectFormatter.format(@actual)
|
94
|
+
end
|
95
|
+
|
96
|
+
# @private
|
97
|
+
def self.matcher_name
|
98
|
+
@matcher_name ||= underscore(name.split('::').last)
|
99
|
+
end
|
100
|
+
|
101
|
+
# @private
|
102
|
+
def matcher_name
|
103
|
+
if defined?(@matcher_name)
|
104
|
+
@matcher_name
|
105
|
+
else
|
106
|
+
self.class.matcher_name
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
# @private
|
111
|
+
# Borrowed from ActiveSupport.
|
112
|
+
def self.underscore(camel_cased_word)
|
113
|
+
word = camel_cased_word.to_s.dup
|
114
|
+
word.gsub!(/([A-Z]+)([A-Z][a-z])/, '\1_\2')
|
115
|
+
word.gsub!(/([a-z\d])([A-Z])/, '\1_\2')
|
116
|
+
word.tr!('-', '_')
|
117
|
+
word.downcase!
|
118
|
+
word
|
119
|
+
end
|
120
|
+
private_class_method :underscore
|
121
|
+
|
122
|
+
private
|
123
|
+
|
124
|
+
def assert_ivars(*expected_ivars)
|
125
|
+
return unless (expected_ivars - present_ivars).any?
|
126
|
+
ivar_list = EnglishPhrasing.list(expected_ivars)
|
127
|
+
raise "#{self.class.name} needs to supply#{ivar_list}"
|
128
|
+
end
|
129
|
+
|
130
|
+
if RUBY_VERSION.to_f < 1.9
|
131
|
+
# :nocov:
|
132
|
+
def present_ivars
|
133
|
+
instance_variables.map(&:to_sym)
|
134
|
+
end
|
135
|
+
# :nocov:
|
136
|
+
else
|
137
|
+
alias present_ivars instance_variables
|
138
|
+
end
|
139
|
+
|
140
|
+
# @private
|
141
|
+
module HashFormatting
|
142
|
+
# `{ :a => 5, :b => 2 }.inspect` produces:
|
143
|
+
#
|
144
|
+
# {:a=>5, :b=>2}
|
145
|
+
#
|
146
|
+
# ...but it looks much better as:
|
147
|
+
#
|
148
|
+
# {:a => 5, :b => 2}
|
149
|
+
#
|
150
|
+
# This is idempotent and safe to run on a string multiple times.
|
151
|
+
def improve_hash_formatting(inspect_string)
|
152
|
+
inspect_string.gsub(/(\S)=>(\S)/, '\1 => \2')
|
153
|
+
end
|
154
|
+
module_function :improve_hash_formatting
|
155
|
+
end
|
156
|
+
|
157
|
+
include HashFormatting
|
158
|
+
|
159
|
+
# @api private
|
160
|
+
# Provides default implementations of failure messages, based on the `description`.
|
161
|
+
module DefaultFailureMessages
|
162
|
+
# @api private
|
163
|
+
# Provides a good generic failure message. Based on `description`.
|
164
|
+
# When subclassing, if you are not satisfied with this failure message
|
165
|
+
# you often only need to override `description`.
|
166
|
+
# @return [String]
|
167
|
+
def failure_message
|
168
|
+
"expected #{description_of @actual} to #{description}".dup
|
169
|
+
end
|
170
|
+
|
171
|
+
# @api private
|
172
|
+
# Provides a good generic negative failure message. Based on `description`.
|
173
|
+
# When subclassing, if you are not satisfied with this failure message
|
174
|
+
# you often only need to override `description`.
|
175
|
+
# @return [String]
|
176
|
+
def failure_message_when_negated
|
177
|
+
"expected #{description_of @actual} not to #{description}".dup
|
178
|
+
end
|
179
|
+
|
180
|
+
# @private
|
181
|
+
def self.has_default_failure_messages?(matcher)
|
182
|
+
matcher.method(:failure_message).owner == self &&
|
183
|
+
matcher.method(:failure_message_when_negated).owner == self
|
184
|
+
rescue NameError
|
185
|
+
false
|
186
|
+
end
|
187
|
+
end
|
188
|
+
|
189
|
+
include DefaultFailureMessages
|
190
|
+
end
|
191
|
+
end
|
192
|
+
end
|
193
|
+
end
|