rspec-expectations 3.8.6

Sign up to get free protection for your applications and to get access to all the features.
Files changed (58) hide show
  1. checksums.yaml +7 -0
  2. checksums.yaml.gz.sig +0 -0
  3. data.tar.gz.sig +5 -0
  4. data/.document +5 -0
  5. data/.yardopts +6 -0
  6. data/Changelog.md +1156 -0
  7. data/LICENSE.md +25 -0
  8. data/README.md +305 -0
  9. data/lib/rspec/expectations.rb +82 -0
  10. data/lib/rspec/expectations/block_snippet_extractor.rb +253 -0
  11. data/lib/rspec/expectations/configuration.rb +215 -0
  12. data/lib/rspec/expectations/expectation_target.rb +127 -0
  13. data/lib/rspec/expectations/fail_with.rb +39 -0
  14. data/lib/rspec/expectations/failure_aggregator.rb +194 -0
  15. data/lib/rspec/expectations/handler.rb +170 -0
  16. data/lib/rspec/expectations/minitest_integration.rb +58 -0
  17. data/lib/rspec/expectations/syntax.rb +132 -0
  18. data/lib/rspec/expectations/version.rb +8 -0
  19. data/lib/rspec/matchers.rb +1034 -0
  20. data/lib/rspec/matchers/aliased_matcher.rb +116 -0
  21. data/lib/rspec/matchers/built_in.rb +52 -0
  22. data/lib/rspec/matchers/built_in/all.rb +86 -0
  23. data/lib/rspec/matchers/built_in/base_matcher.rb +193 -0
  24. data/lib/rspec/matchers/built_in/be.rb +288 -0
  25. data/lib/rspec/matchers/built_in/be_between.rb +77 -0
  26. data/lib/rspec/matchers/built_in/be_instance_of.rb +26 -0
  27. data/lib/rspec/matchers/built_in/be_kind_of.rb +20 -0
  28. data/lib/rspec/matchers/built_in/be_within.rb +72 -0
  29. data/lib/rspec/matchers/built_in/change.rb +428 -0
  30. data/lib/rspec/matchers/built_in/compound.rb +271 -0
  31. data/lib/rspec/matchers/built_in/contain_exactly.rb +302 -0
  32. data/lib/rspec/matchers/built_in/cover.rb +24 -0
  33. data/lib/rspec/matchers/built_in/eq.rb +40 -0
  34. data/lib/rspec/matchers/built_in/eql.rb +34 -0
  35. data/lib/rspec/matchers/built_in/equal.rb +81 -0
  36. data/lib/rspec/matchers/built_in/exist.rb +90 -0
  37. data/lib/rspec/matchers/built_in/has.rb +103 -0
  38. data/lib/rspec/matchers/built_in/have_attributes.rb +114 -0
  39. data/lib/rspec/matchers/built_in/include.rb +149 -0
  40. data/lib/rspec/matchers/built_in/match.rb +106 -0
  41. data/lib/rspec/matchers/built_in/operators.rb +128 -0
  42. data/lib/rspec/matchers/built_in/output.rb +200 -0
  43. data/lib/rspec/matchers/built_in/raise_error.rb +230 -0
  44. data/lib/rspec/matchers/built_in/respond_to.rb +165 -0
  45. data/lib/rspec/matchers/built_in/satisfy.rb +60 -0
  46. data/lib/rspec/matchers/built_in/start_or_end_with.rb +94 -0
  47. data/lib/rspec/matchers/built_in/throw_symbol.rb +132 -0
  48. data/lib/rspec/matchers/built_in/yield.rb +432 -0
  49. data/lib/rspec/matchers/composable.rb +171 -0
  50. data/lib/rspec/matchers/dsl.rb +527 -0
  51. data/lib/rspec/matchers/english_phrasing.rb +58 -0
  52. data/lib/rspec/matchers/expecteds_for_multiple_diffs.rb +73 -0
  53. data/lib/rspec/matchers/fail_matchers.rb +42 -0
  54. data/lib/rspec/matchers/generated_descriptions.rb +41 -0
  55. data/lib/rspec/matchers/matcher_delegator.rb +35 -0
  56. data/lib/rspec/matchers/matcher_protocol.rb +99 -0
  57. metadata +215 -0
  58. 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