rspec-expectations 2.14.0 → 3.13.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- checksums.yaml.gz.sig +0 -0
- data/.document +1 -1
- data/.yardopts +1 -1
- data/Changelog.md +976 -25
- data/{License.txt → LICENSE.md} +5 -3
- data/README.md +162 -26
- data/lib/rspec/expectations/block_snippet_extractor.rb +253 -0
- data/lib/rspec/expectations/configuration.rb +230 -0
- data/lib/rspec/expectations/expectation_target.rb +127 -51
- data/lib/rspec/expectations/fail_with.rb +17 -57
- data/lib/rspec/expectations/failure_aggregator.rb +229 -0
- data/lib/rspec/expectations/handler.rb +146 -32
- data/lib/rspec/expectations/minitest_integration.rb +58 -0
- data/lib/rspec/expectations/syntax.rb +68 -100
- data/lib/rspec/expectations/version.rb +1 -1
- data/lib/rspec/expectations.rb +58 -23
- data/lib/rspec/matchers/aliased_matcher.rb +116 -0
- data/lib/rspec/matchers/built_in/all.rb +86 -0
- data/lib/rspec/matchers/built_in/base_matcher.rb +191 -20
- data/lib/rspec/matchers/built_in/be.rb +114 -114
- data/lib/rspec/matchers/built_in/be_between.rb +77 -0
- data/lib/rspec/matchers/built_in/be_instance_of.rb +15 -4
- data/lib/rspec/matchers/built_in/be_kind_of.rb +10 -1
- data/lib/rspec/matchers/built_in/be_within.rb +35 -18
- data/lib/rspec/matchers/built_in/change.rb +389 -80
- data/lib/rspec/matchers/built_in/compound.rb +290 -0
- data/lib/rspec/matchers/built_in/contain_exactly.rb +310 -0
- data/lib/rspec/matchers/built_in/count_expectation.rb +169 -0
- data/lib/rspec/matchers/built_in/cover.rb +3 -0
- data/lib/rspec/matchers/built_in/eq.rb +30 -8
- data/lib/rspec/matchers/built_in/eql.rb +23 -8
- data/lib/rspec/matchers/built_in/equal.rb +55 -22
- data/lib/rspec/matchers/built_in/exist.rb +74 -10
- data/lib/rspec/matchers/built_in/has.rb +141 -22
- data/lib/rspec/matchers/built_in/have_attributes.rb +114 -0
- data/lib/rspec/matchers/built_in/include.rb +184 -32
- data/lib/rspec/matchers/built_in/match.rb +95 -1
- data/lib/rspec/matchers/built_in/operators.rb +128 -0
- data/lib/rspec/matchers/built_in/output.rb +207 -0
- data/lib/rspec/matchers/built_in/raise_error.rb +192 -44
- data/lib/rspec/matchers/built_in/respond_to.rb +154 -28
- data/lib/rspec/matchers/built_in/satisfy.rb +39 -9
- data/lib/rspec/matchers/built_in/start_or_end_with.rb +94 -0
- data/lib/rspec/matchers/built_in/throw_symbol.rb +58 -14
- data/lib/rspec/matchers/built_in/yield.rb +240 -161
- data/lib/rspec/matchers/built_in.rb +47 -33
- data/lib/rspec/matchers/composable.rb +171 -0
- data/lib/rspec/matchers/dsl.rb +531 -10
- data/lib/rspec/matchers/english_phrasing.rb +58 -0
- data/lib/rspec/matchers/fail_matchers.rb +42 -0
- data/lib/rspec/matchers/generated_descriptions.rb +14 -8
- data/lib/rspec/matchers/matcher_delegator.rb +61 -0
- data/lib/rspec/matchers/matcher_protocol.rb +105 -0
- data/lib/rspec/matchers/multi_matcher_diff.rb +82 -0
- data/lib/rspec/matchers.rb +520 -173
- data.tar.gz.sig +0 -0
- metadata +141 -242
- metadata.gz.sig +2 -0
- data/features/README.md +0 -48
- data/features/Upgrade.md +0 -53
- data/features/built_in_matchers/README.md +0 -90
- data/features/built_in_matchers/be.feature +0 -175
- data/features/built_in_matchers/be_within.feature +0 -48
- data/features/built_in_matchers/cover.feature +0 -47
- data/features/built_in_matchers/end_with.feature +0 -48
- data/features/built_in_matchers/equality.feature +0 -139
- data/features/built_in_matchers/exist.feature +0 -45
- data/features/built_in_matchers/expect_change.feature +0 -59
- data/features/built_in_matchers/expect_error.feature +0 -144
- data/features/built_in_matchers/have.feature +0 -109
- data/features/built_in_matchers/include.feature +0 -174
- data/features/built_in_matchers/match.feature +0 -52
- data/features/built_in_matchers/operators.feature +0 -227
- data/features/built_in_matchers/predicates.feature +0 -137
- data/features/built_in_matchers/respond_to.feature +0 -84
- data/features/built_in_matchers/satisfy.feature +0 -33
- data/features/built_in_matchers/start_with.feature +0 -48
- data/features/built_in_matchers/throw_symbol.feature +0 -91
- data/features/built_in_matchers/types.feature +0 -116
- data/features/built_in_matchers/yield.feature +0 -161
- data/features/custom_matchers/access_running_example.feature +0 -53
- data/features/custom_matchers/define_diffable_matcher.feature +0 -27
- data/features/custom_matchers/define_matcher.feature +0 -368
- data/features/custom_matchers/define_matcher_outside_rspec.feature +0 -38
- data/features/custom_matchers/define_matcher_with_fluent_interface.feature +0 -24
- data/features/customized_message.feature +0 -22
- data/features/diffing.feature +0 -85
- data/features/implicit_docstrings.feature +0 -52
- data/features/step_definitions/additional_cli_steps.rb +0 -22
- data/features/support/env.rb +0 -14
- data/features/syntax_configuration.feature +0 -71
- data/features/test_frameworks/test_unit.feature +0 -44
- data/lib/rspec/expectations/deprecation.rb +0 -17
- data/lib/rspec/expectations/differ.rb +0 -133
- data/lib/rspec/expectations/errors.rb +0 -9
- data/lib/rspec/expectations/extensions/array.rb +0 -9
- data/lib/rspec/expectations/extensions/object.rb +0 -29
- data/lib/rspec/expectations/extensions.rb +0 -2
- data/lib/rspec/matchers/be_close.rb +0 -9
- data/lib/rspec/matchers/built_in/have.rb +0 -124
- data/lib/rspec/matchers/built_in/match_array.rb +0 -51
- data/lib/rspec/matchers/built_in/start_and_end_with.rb +0 -48
- data/lib/rspec/matchers/compatibility.rb +0 -14
- data/lib/rspec/matchers/configuration.rb +0 -108
- data/lib/rspec/matchers/extensions/instance_eval_with_args.rb +0 -39
- data/lib/rspec/matchers/matcher.rb +0 -300
- data/lib/rspec/matchers/method_missing.rb +0 -12
- data/lib/rspec/matchers/operator_matcher.rb +0 -109
- data/lib/rspec/matchers/pretty.rb +0 -70
- data/lib/rspec/matchers/test_unit_integration.rb +0 -11
- data/lib/rspec-expectations.rb +0 -1
- data/spec/rspec/expectations/differ_spec.rb +0 -192
- data/spec/rspec/expectations/expectation_target_spec.rb +0 -82
- data/spec/rspec/expectations/extensions/kernel_spec.rb +0 -67
- data/spec/rspec/expectations/fail_with_spec.rb +0 -114
- data/spec/rspec/expectations/handler_spec.rb +0 -227
- data/spec/rspec/expectations/syntax_spec.rb +0 -139
- data/spec/rspec/matchers/base_matcher_spec.rb +0 -62
- data/spec/rspec/matchers/be_close_spec.rb +0 -22
- data/spec/rspec/matchers/be_instance_of_spec.rb +0 -63
- data/spec/rspec/matchers/be_kind_of_spec.rb +0 -41
- data/spec/rspec/matchers/be_spec.rb +0 -516
- data/spec/rspec/matchers/be_within_spec.rb +0 -137
- data/spec/rspec/matchers/change_spec.rb +0 -553
- data/spec/rspec/matchers/configuration_spec.rb +0 -206
- data/spec/rspec/matchers/cover_spec.rb +0 -69
- data/spec/rspec/matchers/description_generation_spec.rb +0 -190
- data/spec/rspec/matchers/dsl_spec.rb +0 -57
- data/spec/rspec/matchers/eq_spec.rb +0 -60
- data/spec/rspec/matchers/eql_spec.rb +0 -41
- data/spec/rspec/matchers/equal_spec.rb +0 -78
- data/spec/rspec/matchers/exist_spec.rb +0 -124
- data/spec/rspec/matchers/has_spec.rb +0 -122
- data/spec/rspec/matchers/have_spec.rb +0 -455
- data/spec/rspec/matchers/include_matcher_integration_spec.rb +0 -30
- data/spec/rspec/matchers/include_spec.rb +0 -531
- data/spec/rspec/matchers/match_array_spec.rb +0 -194
- data/spec/rspec/matchers/match_spec.rb +0 -61
- data/spec/rspec/matchers/matcher_spec.rb +0 -471
- data/spec/rspec/matchers/matchers_spec.rb +0 -37
- data/spec/rspec/matchers/method_missing_spec.rb +0 -28
- data/spec/rspec/matchers/operator_matcher_spec.rb +0 -223
- data/spec/rspec/matchers/raise_error_spec.rb +0 -485
- data/spec/rspec/matchers/respond_to_spec.rb +0 -292
- data/spec/rspec/matchers/satisfy_spec.rb +0 -44
- data/spec/rspec/matchers/start_with_end_with_spec.rb +0 -186
- data/spec/rspec/matchers/throw_symbol_spec.rb +0 -116
- data/spec/rspec/matchers/yield_spec.rb +0 -514
- data/spec/spec_helper.rb +0 -54
- data/spec/support/classes.rb +0 -56
- data/spec/support/in_sub_process.rb +0 -38
- data/spec/support/matchers.rb +0 -22
- data/spec/support/ruby_version.rb +0 -10
- data/spec/support/shared_examples.rb +0 -13
@@ -0,0 +1,171 @@
|
|
1
|
+
RSpec::Support.require_rspec_support "fuzzy_matcher"
|
2
|
+
|
3
|
+
module RSpec
|
4
|
+
module Matchers
|
5
|
+
# Mixin designed to support the composable matcher features
|
6
|
+
# of RSpec 3+. Mix it into your custom matcher classes to
|
7
|
+
# allow them to be used in a composable fashion.
|
8
|
+
#
|
9
|
+
# @api public
|
10
|
+
module Composable
|
11
|
+
# Creates a compound `and` expectation. The matcher will
|
12
|
+
# only pass if both sub-matchers pass.
|
13
|
+
# This can be chained together to form an arbitrarily long
|
14
|
+
# chain of matchers.
|
15
|
+
#
|
16
|
+
# @example
|
17
|
+
# expect(alphabet).to start_with("a").and end_with("z")
|
18
|
+
# expect(alphabet).to start_with("a") & end_with("z")
|
19
|
+
#
|
20
|
+
# @note The negative form (`expect(...).not_to matcher.and other`)
|
21
|
+
# is not supported at this time.
|
22
|
+
def and(matcher)
|
23
|
+
BuiltIn::Compound::And.new self, matcher
|
24
|
+
end
|
25
|
+
alias & and
|
26
|
+
|
27
|
+
# Creates a compound `or` expectation. The matcher will
|
28
|
+
# pass if either sub-matcher passes.
|
29
|
+
# This can be chained together to form an arbitrarily long
|
30
|
+
# chain of matchers.
|
31
|
+
#
|
32
|
+
# @example
|
33
|
+
# expect(stoplight.color).to eq("red").or eq("green").or eq("yellow")
|
34
|
+
# expect(stoplight.color).to eq("red") | eq("green") | eq("yellow")
|
35
|
+
#
|
36
|
+
# @note The negative form (`expect(...).not_to matcher.or other`)
|
37
|
+
# is not supported at this time.
|
38
|
+
def or(matcher)
|
39
|
+
BuiltIn::Compound::Or.new self, matcher
|
40
|
+
end
|
41
|
+
alias | or
|
42
|
+
|
43
|
+
# Delegates to `#matches?`. Allows matchers to be used in composable
|
44
|
+
# fashion and also supports using matchers in case statements.
|
45
|
+
def ===(value)
|
46
|
+
matches?(value)
|
47
|
+
end
|
48
|
+
|
49
|
+
private
|
50
|
+
|
51
|
+
# This provides a generic way to fuzzy-match an expected value against
|
52
|
+
# an actual value. It understands nested data structures (e.g. hashes
|
53
|
+
# and arrays) and is able to match against a matcher being used as
|
54
|
+
# the expected value or within the expected value at any level of
|
55
|
+
# nesting.
|
56
|
+
#
|
57
|
+
# Within a custom matcher you are encouraged to use this whenever your
|
58
|
+
# matcher needs to match two values, unless it needs more precise semantics.
|
59
|
+
# For example, the `eq` matcher _does not_ use this as it is meant to
|
60
|
+
# use `==` (and only `==`) for matching.
|
61
|
+
#
|
62
|
+
# @param expected [Object] what is expected
|
63
|
+
# @param actual [Object] the actual value
|
64
|
+
#
|
65
|
+
# @!visibility public
|
66
|
+
def values_match?(expected, actual)
|
67
|
+
expected = with_matchers_cloned(expected)
|
68
|
+
Support::FuzzyMatcher.values_match?(expected, actual)
|
69
|
+
end
|
70
|
+
|
71
|
+
# Returns the description of the given object in a way that is
|
72
|
+
# aware of composed matchers. If the object is a matcher with
|
73
|
+
# a `description` method, returns the description; otherwise
|
74
|
+
# returns `object.inspect`.
|
75
|
+
#
|
76
|
+
# You are encouraged to use this in your custom matcher's
|
77
|
+
# `description`, `failure_message` or
|
78
|
+
# `failure_message_when_negated` implementation if you are
|
79
|
+
# supporting matcher arguments.
|
80
|
+
#
|
81
|
+
# @!visibility public
|
82
|
+
def description_of(object)
|
83
|
+
RSpec::Support::ObjectFormatter.format(object)
|
84
|
+
end
|
85
|
+
|
86
|
+
# Transforms the given data structure (typically a hash or array)
|
87
|
+
# into a new data structure that, when `#inspect` is called on it,
|
88
|
+
# will provide descriptions of any contained matchers rather than
|
89
|
+
# the normal `#inspect` output.
|
90
|
+
#
|
91
|
+
# You are encouraged to use this in your custom matcher's
|
92
|
+
# `description`, `failure_message` or
|
93
|
+
# `failure_message_when_negated` implementation if you are
|
94
|
+
# supporting any arguments which may be a data structure
|
95
|
+
# containing matchers.
|
96
|
+
#
|
97
|
+
# @!visibility public
|
98
|
+
def surface_descriptions_in(item)
|
99
|
+
if Matchers.is_a_describable_matcher?(item)
|
100
|
+
DescribableItem.new(item)
|
101
|
+
elsif Hash === item
|
102
|
+
Hash[surface_descriptions_in(item.to_a)]
|
103
|
+
elsif Struct === item || unreadable_io?(item)
|
104
|
+
RSpec::Support::ObjectFormatter.format(item)
|
105
|
+
elsif should_enumerate?(item)
|
106
|
+
item.map { |subitem| surface_descriptions_in(subitem) }
|
107
|
+
else
|
108
|
+
item
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
# @private
|
113
|
+
# Historically, a single matcher instance was only checked
|
114
|
+
# against a single value. Given that the matcher was only
|
115
|
+
# used once, it's been common to memoize some intermediate
|
116
|
+
# calculation that is derived from the `actual` value in
|
117
|
+
# order to reuse that intermediate result in the failure
|
118
|
+
# message.
|
119
|
+
#
|
120
|
+
# This can cause a problem when using such a matcher as an
|
121
|
+
# argument to another matcher in a composed matcher expression,
|
122
|
+
# since the matcher instance may be checked against multiple
|
123
|
+
# values and produce invalid results due to the memoization.
|
124
|
+
#
|
125
|
+
# To deal with this, we clone any matchers in `expected` via
|
126
|
+
# this method when using `values_match?`, so that any memoization
|
127
|
+
# does not "leak" between checks.
|
128
|
+
def with_matchers_cloned(object)
|
129
|
+
if Matchers.is_a_matcher?(object)
|
130
|
+
object.clone
|
131
|
+
elsif Hash === object
|
132
|
+
Hash[with_matchers_cloned(object.to_a)]
|
133
|
+
elsif should_enumerate?(object)
|
134
|
+
object.map { |subobject| with_matchers_cloned(subobject) }
|
135
|
+
else
|
136
|
+
object
|
137
|
+
end
|
138
|
+
end
|
139
|
+
|
140
|
+
# @api private
|
141
|
+
# We should enumerate arrays as long as they are not recursive.
|
142
|
+
def should_enumerate?(item)
|
143
|
+
Array === item && item.none? { |subitem| subitem.equal?(item) }
|
144
|
+
end
|
145
|
+
|
146
|
+
# @api private
|
147
|
+
def unreadable_io?(object)
|
148
|
+
return false unless IO === object
|
149
|
+
object.each {} # STDOUT is enumerable but raises an error
|
150
|
+
false
|
151
|
+
rescue IOError
|
152
|
+
true
|
153
|
+
end
|
154
|
+
module_function :surface_descriptions_in, :should_enumerate?, :unreadable_io?
|
155
|
+
|
156
|
+
# Wraps an item in order to surface its `description` via `inspect`.
|
157
|
+
# @api private
|
158
|
+
DescribableItem = Struct.new(:item) do
|
159
|
+
# Inspectable version of the item description
|
160
|
+
def inspect
|
161
|
+
"(#{item.description})"
|
162
|
+
end
|
163
|
+
|
164
|
+
# A pretty printed version of the item description.
|
165
|
+
def pretty_print(pp)
|
166
|
+
pp.text "(#{item.description})"
|
167
|
+
end
|
168
|
+
end
|
169
|
+
end
|
170
|
+
end
|
171
|
+
end
|