mocha 1.9.0 → 1.11.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (215) hide show
  1. checksums.yaml +4 -4
  2. data/.github/FUNDING.yml +1 -0
  3. data/.rubocop.yml +3 -0
  4. data/.rubocop_todo.yml +7 -30
  5. data/.yardopts +1 -0
  6. data/README.md +30 -11
  7. data/RELEASE.md +93 -0
  8. data/Rakefile +10 -7
  9. data/docs/CNAME +1 -0
  10. data/docs/Mocha.html +132 -5
  11. data/docs/Mocha/API.html +437 -206
  12. data/docs/Mocha/ClassMethods.html +13 -16
  13. data/docs/Mocha/Configuration.html +1126 -227
  14. data/docs/Mocha/Expectation.html +420 -267
  15. data/docs/Mocha/ExpectationError.html +5 -10
  16. data/docs/Mocha/ExpectationErrorFactory.html +9 -18
  17. data/docs/Mocha/Hooks.html +12 -27
  18. data/docs/Mocha/Integration.html +3 -3
  19. data/docs/Mocha/Integration/MiniTest.html +3 -3
  20. data/docs/Mocha/Integration/MiniTest/Adapter.html +5 -6
  21. data/docs/Mocha/Integration/TestUnit.html +3 -3
  22. data/docs/Mocha/Integration/TestUnit/Adapter.html +5 -6
  23. data/docs/Mocha/Mock.html +202 -162
  24. data/docs/Mocha/ObjectMethods.html +121 -68
  25. data/docs/Mocha/ParameterMatchers.html +21 -109
  26. data/docs/Mocha/ParameterMatchers/AllOf.html +4 -5
  27. data/docs/Mocha/ParameterMatchers/AnyOf.html +4 -5
  28. data/docs/Mocha/ParameterMatchers/AnyParameters.html +3 -3
  29. data/docs/Mocha/ParameterMatchers/Anything.html +3 -3
  30. data/docs/Mocha/ParameterMatchers/Base.html +8 -15
  31. data/docs/Mocha/ParameterMatchers/Equals.html +4 -5
  32. data/docs/Mocha/ParameterMatchers/EquivalentUri.html +3 -3
  33. data/docs/Mocha/ParameterMatchers/HasEntries.html +4 -5
  34. data/docs/Mocha/ParameterMatchers/HasEntry.html +4 -5
  35. data/docs/Mocha/ParameterMatchers/HasKey.html +4 -5
  36. data/docs/Mocha/ParameterMatchers/HasValue.html +4 -5
  37. data/docs/Mocha/ParameterMatchers/Includes.html +4 -5
  38. data/docs/Mocha/ParameterMatchers/InstanceOf.html +4 -5
  39. data/docs/Mocha/ParameterMatchers/IsA.html +3 -3
  40. data/docs/Mocha/ParameterMatchers/KindOf.html +4 -5
  41. data/docs/Mocha/ParameterMatchers/Not.html +4 -5
  42. data/docs/Mocha/ParameterMatchers/Optionally.html +3 -3
  43. data/docs/Mocha/ParameterMatchers/RegexpMatches.html +4 -5
  44. data/docs/Mocha/ParameterMatchers/RespondsWith.html +4 -5
  45. data/docs/Mocha/ParameterMatchers/YamlEquivalent.html +4 -5
  46. data/docs/Mocha/Sequence.html +3 -3
  47. data/docs/Mocha/StateMachine.html +13 -25
  48. data/docs/Mocha/StateMachine/State.html +4 -5
  49. data/docs/Mocha/StateMachine/StatePredicate.html +4 -5
  50. data/docs/Mocha/StubbingError.html +3 -3
  51. data/docs/_index.html +4 -22
  52. data/docs/class_list.html +1 -1
  53. data/docs/file.COPYING.html +3 -3
  54. data/docs/file.MIT-LICENSE.html +3 -3
  55. data/docs/file.README.html +36 -14
  56. data/docs/file.RELEASE.html +126 -3
  57. data/docs/frames.html +1 -1
  58. data/docs/index.html +36 -14
  59. data/docs/method_list.html +117 -37
  60. data/docs/top-level-namespace.html +3 -3
  61. data/gemfiles/Gemfile.minitest.5.11.3 +7 -0
  62. data/init.rb +1 -3
  63. data/lib/mocha.rb +8 -0
  64. data/lib/mocha/any_instance_method.rb +9 -25
  65. data/lib/mocha/api.rb +120 -56
  66. data/lib/mocha/block_matcher.rb +31 -0
  67. data/lib/mocha/cardinality.rb +26 -11
  68. data/lib/mocha/class_methods.rb +17 -15
  69. data/lib/mocha/configuration.rb +351 -67
  70. data/lib/mocha/deprecation.rb +2 -1
  71. data/lib/mocha/detection/test_unit.rb +1 -3
  72. data/lib/mocha/exception_raiser.rb +2 -1
  73. data/lib/mocha/expectation.rb +102 -63
  74. data/lib/mocha/expectation_error.rb +1 -3
  75. data/lib/mocha/expectation_list.rb +6 -6
  76. data/lib/mocha/inspect.rb +28 -26
  77. data/lib/mocha/instance_method.rb +19 -2
  78. data/lib/mocha/integration.rb +1 -3
  79. data/lib/mocha/integration/mini_test.rb +7 -0
  80. data/lib/mocha/integration/test_unit.rb +7 -0
  81. data/lib/mocha/invocation.rb +77 -0
  82. data/lib/mocha/macos_version.rb +5 -0
  83. data/lib/mocha/minitest.rb +6 -1
  84. data/lib/mocha/mock.rb +46 -31
  85. data/lib/mocha/mockery.rb +25 -61
  86. data/lib/mocha/names.rb +1 -1
  87. data/lib/mocha/object_methods.rb +13 -19
  88. data/lib/mocha/parameter_matchers.rb +1 -1
  89. data/lib/mocha/parameter_matchers/all_of.rb +1 -1
  90. data/lib/mocha/parameter_matchers/any_of.rb +1 -1
  91. data/lib/mocha/parameter_matchers/equivalent_uri.rb +0 -9
  92. data/lib/mocha/parameter_matchers/includes.rb +2 -0
  93. data/lib/mocha/parameter_matchers/instance_methods.rb +18 -0
  94. data/lib/mocha/raised_exception.rb +11 -0
  95. data/lib/mocha/return_values.rb +3 -3
  96. data/lib/mocha/setup.rb +5 -0
  97. data/lib/mocha/single_return_value.rb +2 -1
  98. data/lib/mocha/singleton_class.rb +9 -0
  99. data/lib/mocha/{class_method.rb → stubbed_method.rb} +23 -43
  100. data/lib/mocha/test_unit.rb +6 -1
  101. data/lib/mocha/thrower.rb +2 -1
  102. data/lib/mocha/thrown_object.rb +12 -0
  103. data/lib/mocha/version.rb +1 -1
  104. data/lib/mocha/yield_parameters.rb +5 -11
  105. data/mocha.gemspec +1 -3
  106. data/test/acceptance/acceptance_test_helper.rb +7 -0
  107. data/test/acceptance/bug_18914_test.rb +0 -1
  108. data/test/acceptance/bug_21465_test.rb +0 -1
  109. data/test/acceptance/bug_21563_test.rb +0 -1
  110. data/test/acceptance/display_matching_invocations_alongside_expectations_test.rb +69 -0
  111. data/test/acceptance/exception_rescue_test.rb +1 -2
  112. data/test/acceptance/expectations_on_multiple_methods_test.rb +0 -1
  113. data/test/acceptance/expected_invocation_count_test.rb +2 -3
  114. data/test/acceptance/failure_messages_test.rb +16 -1
  115. data/test/acceptance/issue_272_test.rb +1 -2
  116. data/test/acceptance/issue_65_test.rb +0 -1
  117. data/test/acceptance/issue_70_test.rb +0 -1
  118. data/test/acceptance/mocha_example_test.rb +0 -1
  119. data/test/acceptance/mocha_test_result_test.rb +0 -1
  120. data/test/acceptance/mock_built_with_first_argument_type_being_string_test.rb +98 -0
  121. data/test/acceptance/mock_test.rb +63 -6
  122. data/test/acceptance/mocked_methods_dispatch_test.rb +0 -1
  123. data/test/acceptance/multiple_expectations_failure_message_test.rb +0 -1
  124. data/test/acceptance/multiple_yielding_test.rb +59 -0
  125. data/test/acceptance/optional_parameters_test.rb +0 -1
  126. data/test/acceptance/parameter_matcher_test.rb +0 -1
  127. data/test/acceptance/partial_mocks_test.rb +0 -1
  128. data/test/acceptance/prepend_test.rb +0 -1
  129. data/test/acceptance/prevent_use_of_mocha_outside_test_test.rb +0 -1
  130. data/test/acceptance/raise_exception_test.rb +0 -1
  131. data/test/acceptance/return_value_test.rb +0 -1
  132. data/test/acceptance/sequence_test.rb +0 -1
  133. data/test/acceptance/states_test.rb +0 -1
  134. data/test/acceptance/stub_any_instance_method_defined_on_superclass_test.rb +1 -2
  135. data/test/acceptance/stub_any_instance_method_test.rb +20 -1
  136. data/test/acceptance/stub_class_method_defined_on_active_record_association_proxy_test.rb +0 -1
  137. data/test/acceptance/stub_class_method_defined_on_class_test.rb +0 -1
  138. data/test/acceptance/stub_class_method_defined_on_module_test.rb +0 -1
  139. data/test/acceptance/stub_class_method_defined_on_superclass_test.rb +1 -2
  140. data/test/acceptance/stub_everything_test.rb +0 -1
  141. data/test/acceptance/stub_instance_method_defined_on_active_record_association_proxy_test.rb +0 -1
  142. data/test/acceptance/stub_instance_method_defined_on_class_and_aliased_test.rb +0 -1
  143. data/test/acceptance/stub_instance_method_defined_on_class_test.rb +0 -1
  144. data/test/acceptance/stub_instance_method_defined_on_kernel_module_test.rb +0 -1
  145. data/test/acceptance/stub_instance_method_defined_on_module_test.rb +0 -1
  146. data/test/acceptance/stub_instance_method_defined_on_object_class_test.rb +0 -1
  147. data/test/acceptance/stub_instance_method_defined_on_singleton_class_test.rb +0 -1
  148. data/test/acceptance/stub_instance_method_defined_on_superclass_test.rb +0 -1
  149. data/test/acceptance/stub_method_defined_on_module_and_aliased_test.rb +0 -1
  150. data/test/acceptance/stub_module_method_test.rb +0 -1
  151. data/test/acceptance/stub_test.rb +0 -1
  152. data/test/acceptance/stubba_example_test.rb +0 -1
  153. data/test/acceptance/stubba_test_result_test.rb +0 -1
  154. data/test/acceptance/stubbing_error_backtrace_test.rb +4 -5
  155. data/test/acceptance/stubbing_frozen_object_test.rb +0 -1
  156. data/test/acceptance/stubbing_method_accepting_block_parameter_test.rb +0 -1
  157. data/test/acceptance/stubbing_method_unnecessarily_test.rb +5 -5
  158. data/test/acceptance/stubbing_nil_test.rb +5 -5
  159. data/test/acceptance/stubbing_non_existent_any_instance_method_test.rb +27 -11
  160. data/test/acceptance/stubbing_non_existent_class_method_test.rb +11 -11
  161. data/test/acceptance/stubbing_non_existent_instance_method_test.rb +11 -11
  162. data/test/acceptance/stubbing_non_public_any_instance_method_test.rb +8 -8
  163. data/test/acceptance/stubbing_non_public_class_method_test.rb +9 -9
  164. data/test/acceptance/stubbing_non_public_instance_method_test.rb +9 -9
  165. data/test/acceptance/stubbing_on_non_mock_object_test.rb +5 -5
  166. data/test/acceptance/stubbing_same_class_method_on_parent_and_child_classes_test.rb +0 -1
  167. data/test/acceptance/throw_test.rb +0 -1
  168. data/test/acceptance/unexpected_invocation_test.rb +0 -1
  169. data/test/acceptance/unstubbing_test.rb +0 -1
  170. data/test/acceptance/yielding_test.rb +80 -0
  171. data/test/integration/shared_tests.rb +5 -3
  172. data/test/method_definer.rb +11 -17
  173. data/test/test_runner.rb +2 -0
  174. data/test/unit/any_instance_method_test.rb +41 -40
  175. data/test/unit/cardinality_test.rb +41 -23
  176. data/test/unit/central_test.rb +0 -1
  177. data/test/unit/class_methods_test.rb +1 -1
  178. data/test/unit/configuration_test.rb +12 -12
  179. data/test/unit/exception_raiser_test.rb +10 -5
  180. data/test/unit/expectation_list_test.rb +13 -11
  181. data/test/unit/expectation_test.rb +115 -103
  182. data/test/unit/instance_method_test.rb +282 -0
  183. data/test/unit/mock_test.rb +28 -19
  184. data/test/unit/mockery_test.rb +8 -11
  185. data/test/unit/module_methods_test.rb +2 -3
  186. data/test/unit/object_inspect_test.rb +6 -4
  187. data/test/unit/object_methods_test.rb +3 -2
  188. data/test/unit/parameter_matchers/equivalent_uri_test.rb +0 -9
  189. data/test/unit/parameter_matchers/has_entries_test.rb +1 -1
  190. data/test/unit/parameter_matchers/has_entry_test.rb +1 -1
  191. data/test/unit/parameter_matchers/has_key_test.rb +1 -1
  192. data/test/unit/parameter_matchers/has_value_test.rb +1 -1
  193. data/test/unit/parameter_matchers/includes_test.rb +1 -1
  194. data/test/unit/parameter_matchers/responds_with_test.rb +1 -1
  195. data/test/unit/return_values_test.rb +25 -20
  196. data/test/unit/single_return_value_test.rb +6 -1
  197. data/test/unit/thrower_test.rb +7 -2
  198. data/test/unit/yield_parameters_test.rb +35 -53
  199. metadata +19 -34
  200. data/docs/Mocha/UnexpectedInvocation.html +0 -140
  201. data/lib/mocha/mini_test.rb +0 -5
  202. data/lib/mocha/module_method.rb +0 -6
  203. data/lib/mocha/module_methods.rb +0 -10
  204. data/lib/mocha/multiple_yields.rb +0 -15
  205. data/lib/mocha/no_yields.rb +0 -5
  206. data/lib/mocha/parameter_matchers/object.rb +0 -15
  207. data/lib/mocha/single_yield.rb +0 -13
  208. data/lib/mocha/standalone.rb +0 -4
  209. data/lib/mocha/unexpected_invocation.rb +0 -24
  210. data/lib/mocha_standalone.rb +0 -4
  211. data/test/acceptance/mock_with_initializer_block_test.rb +0 -56
  212. data/test/unit/class_method_test.rb +0 -276
  213. data/test/unit/multiple_yields_test.rb +0 -16
  214. data/test/unit/no_yields_test.rb +0 -16
  215. data/test/unit/single_yield_test.rb +0 -16
@@ -6,7 +6,7 @@
6
6
  <title>
7
7
  Top Level Namespace
8
8
 
9
- &mdash; Mocha 1.9.0
9
+ &mdash; Mocha 1.11.1
10
10
 
11
11
  </title>
12
12
 
@@ -108,9 +108,9 @@
108
108
  </div>
109
109
 
110
110
  <div id="footer">
111
- Generated on Mon Jun 17 18:38:43 2019 by
111
+ Generated on Tue Dec 17 12:49:34 2019 by
112
112
  <a href="http://yardoc.org" title="Yay! A Ruby Documentation Tool" target="_parent">yard</a>
113
- 0.9.19 (ruby-2.5.3).
113
+ 0.9.20 (ruby-2.6.5).
114
114
  </div>
115
115
 
116
116
  </div>
@@ -0,0 +1,7 @@
1
+ source 'https://rubygems.org'
2
+
3
+ gemspec :path=>"../"
4
+
5
+ group :development do
6
+ gem "minitest", "5.11.3"
7
+ end
data/init.rb CHANGED
@@ -1,3 +1 @@
1
- # Mocha should no longer be loaded at plugin load time
2
- # You should explicitly load Mocha *after* Test::Unit or MiniTest have been loaded
3
- # e.g. by adding "require 'mocha'" at the bottom of test/test_helper.rb
1
+ warn 'Mocha deprecation warning: The old-style Rails plugin will not be supported in future versions of Mocha.'
@@ -1 +1,9 @@
1
1
  require 'mocha/version'
2
+ require 'mocha/ruby_version'
3
+ require 'mocha/deprecation'
4
+
5
+ if Mocha::PRE_RUBY_V19
6
+ Mocha::Deprecation.warning(
7
+ 'Versions of Ruby earlier than v1.9 will not be supported in future versions of Mocha.'
8
+ )
9
+ end
@@ -1,36 +1,20 @@
1
1
  require 'mocha/ruby_version'
2
- require 'mocha/class_method'
2
+ require 'mocha/stubbed_method'
3
3
 
4
4
  module Mocha
5
- class AnyInstanceMethod < ClassMethod
6
- def mock
7
- stubbee.any_instance.mocha
8
- end
9
-
10
- def reset_mocha
11
- stubbee.any_instance.reset_mocha
12
- end
5
+ class AnyInstanceMethod < StubbedMethod
6
+ private
13
7
 
14
- def restore_original_method
15
- return if use_prepended_module_for_stub_method?
16
- return unless stub_method_overwrites_original_method?
17
- original_method_owner.send(:define_method, method_name, original_method)
18
- Module.instance_method(original_visibility).bind(original_method_owner).call(method_name)
8
+ def mock_owner
9
+ stubbee.any_instance
19
10
  end
20
11
 
21
- private
22
-
23
- def store_original_method
24
- @original_method = original_method_owner.instance_method(method_name)
12
+ def method_body(method)
13
+ method
25
14
  end
26
15
 
27
- def stub_method_definition
28
- method_implementation = <<-CODE
29
- def #{method_name}(*args, &block)
30
- self.class.any_instance.mocha.method_missing(:#{method_name}, *args, &block)
31
- end
32
- CODE
33
- [method_implementation, __FILE__, __LINE__ - 4]
16
+ def stubbee_method(method_name)
17
+ stubbee.instance_method(method_name)
34
18
  end
35
19
 
36
20
  def original_method_owner
@@ -3,11 +3,33 @@ require 'mocha/hooks'
3
3
  require 'mocha/mockery'
4
4
  require 'mocha/sequence'
5
5
  require 'mocha/object_methods'
6
- require 'mocha/module_methods'
7
6
  require 'mocha/class_methods'
8
7
 
9
8
  module Mocha
10
9
  # Methods added to +Test::Unit::TestCase+, +MiniTest::Unit::TestCase+ or equivalent.
10
+ # The mock creation methods are {#mock}, {#stub} and {#stub_everything}, all of which return a #{Mock}
11
+ # which can be further modified by {Mock#responds_like} and {Mock#responds_like_instance_of} methods,
12
+ # both of which return a {Mock}, too, and can therefore, be chained to the original creation methods.
13
+ #
14
+ # {Mock#responds_like} and {Mock#responds_like_instance_of} force the mock to indicate what it is
15
+ # supposed to be mocking, thus making it a safer verifying mock. They check that the underlying +responder+
16
+ # will actually respond to the methods being stubbed, throwing a +NoMethodError+ upon invocation otherwise.
17
+ #
18
+ # @example Verifying mock using {Mock#responds_like_instance_of}
19
+ # class Sheep
20
+ # def initialize
21
+ # raise "some awkward code we don't want to call"
22
+ # end
23
+ # def chew(grass); end
24
+ # end
25
+ #
26
+ # sheep = mock('sheep').responds_like_instance_of(Sheep)
27
+ # sheep.expects(:chew)
28
+ # sheep.expects(:foo)
29
+ # sheep.respond_to?(:chew) # => true
30
+ # sheep.respond_to?(:foo) # => false
31
+ # sheep.chew
32
+ # sheep.foo # => raises NoMethodError exception
11
33
  module API
12
34
  include ParameterMatchers
13
35
  include Hooks
@@ -15,22 +37,26 @@ module Mocha
15
37
  # @private
16
38
  def self.included(_mod)
17
39
  Object.send(:include, Mocha::ObjectMethods)
18
- Module.send(:include, Mocha::ModuleMethods)
19
40
  Class.send(:include, Mocha::ClassMethods)
20
41
  end
21
42
 
43
+ # @private
44
+ def self.extended(mod)
45
+ included(mod)
46
+ end
47
+
22
48
  # Builds a new mock object
23
49
  #
24
- # @param [String] name identifies mock object in error messages.
25
- # @param [Hash] expected_methods_vs_return_values expected method name symbols as keys and corresponding return values as values - these expectations are setup as if {Mock#expects} were called multiple times.
26
- # @yield optional block to be evaluated in the context of the mock object instance, giving an alternative way to setup stubbed methods.
27
- # @yield note that the block is evaulated by calling Mock#instance_eval and so things like instance variables declared in the test will not be available within the block.
28
- # @yield deprecated: use Object#tap or define stubs/expectations with an explicit receiver instead.
29
50
  # @return [Mock] a new mock object
30
51
  #
31
- # @overload def mock(name, &block)
32
- # @overload def mock(expected_methods_vs_return_values = {}, &block)
33
- # @overload def mock(name, expected_methods_vs_return_values = {}, &block)
52
+ # @overload def mock(name)
53
+ # @param [String, Symbol] name identifies mock object in error messages.
54
+ # @note Prior to v1.10.0 when +name+ was a +Symbol+, this method returned an unnamed +Mock+ that expected the method identified by +name+. This was undocumented behaviour and it will be removed in the future, but for the moment it can be reinstated using {Configuration#reinstate_undocumented_behaviour_from_v1_9=}.
55
+ # @overload def mock(expected_methods_vs_return_values = {})
56
+ # @param [Hash] expected_methods_vs_return_values expected method name symbols as keys and corresponding return values as values - these expectations are setup as if {Mock#expects} were called multiple times.
57
+ # @overload def mock(name, expected_methods_vs_return_values = {})
58
+ # @param [String, Symbol] name identifies mock object in error messages.
59
+ # @param [Hash] expected_methods_vs_return_values expected method name symbols as keys and corresponding return values as values - these expectations are setup as if {Mock#expects} were called multiple times.
34
60
  #
35
61
  # @example Using expected_methods_vs_return_values Hash to setup expectations.
36
62
  # def test_motor_starts_and_stops
@@ -39,36 +65,45 @@ module Mocha
39
65
  # assert motor.stop
40
66
  # # an error will be raised unless both Motor#start and Motor#stop have been called
41
67
  # end
42
- # @example Using the optional block to setup expectations & stubbed methods [deprecated].
43
- # def test_motor_starts_and_stops
44
- # motor = mock('motor') do
45
- # expects(:start).with(100.rpm).returns(true)
46
- # stubs(:stop).returns(true)
47
- # end
48
- # assert motor.start(100.rpm)
49
- # assert motor.stop
50
- # # an error will only be raised if Motor#start(100.rpm) has not been called
51
- # end
52
- def mock(*arguments, &block)
53
- name = arguments.shift if arguments.first.is_a?(String)
68
+ # rubocop:disable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
69
+ def mock(*arguments)
70
+ if Mocha.configuration.reinstate_undocumented_behaviour_from_v1_9?
71
+ if arguments.first.is_a?(Symbol)
72
+ method_name = arguments[0]
73
+ Deprecation.warning(
74
+ "Explicitly include `#{method_name}` in Hash of expected methods vs return values,",
75
+ " e.g. `mock(:#{method_name} => nil)`."
76
+ )
77
+ if arguments[1]
78
+ Deprecation.warning(
79
+ "In this case the 2nd argument for `mock(:##{method_name}, ...)` is ignored,",
80
+ ' but in the future a Hash of expected methods vs return values will be respected.'
81
+ )
82
+ end
83
+ elsif arguments.first.is_a?(String)
84
+ name = arguments.shift
85
+ end
86
+ elsif arguments.first.is_a?(String) || arguments.first.is_a?(Symbol)
87
+ name = arguments.shift
88
+ end
54
89
  expectations = arguments.shift || {}
55
- mock = name ? Mockery.instance.named_mock(name, &block) : Mockery.instance.unnamed_mock(&block)
90
+ mock = name ? Mockery.instance.named_mock(name) : Mockery.instance.unnamed_mock
56
91
  mock.expects(expectations)
57
92
  mock
58
93
  end
94
+ # rubocop:enable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
59
95
 
60
96
  # Builds a new mock object
61
97
  #
62
- # @param [String] name identifies mock object in error messages.
63
- # @param [Hash] stubbed_methods_vs_return_values stubbed method name symbols as keys and corresponding return values as values - these stubbed methods are setup as if {Mock#stubs} were called multiple times.
64
- # @yield optional block to be evaluated in the context of the mock object instance, giving an alternative way to setup stubbed methods.
65
- # @yield note that the block is evaulated by calling Mock#instance_eval and so things like instance variables declared in the test will not be available within the block.
66
- # @yield deprecated: use Object#tap or define stubs/expectations with an explicit receiver instead.
67
98
  # @return [Mock] a new mock object
68
99
  #
69
- # @overload def stub(name, &block)
70
- # @overload def stub(stubbed_methods_vs_return_values = {}, &block)
71
- # @overload def stub(name, stubbed_methods_vs_return_values = {}, &block)
100
+ # @overload def stub(name)
101
+ # @param [String, Symbol] name identifies mock object in error messages.
102
+ # @note Prior to v1.10.0 when +name+ was a +Symbol+, this method returned an unnamed +Mock+ that stubbed the method identified by +name+. This was undocumented behaviour and it will be removed in the future, but for the moment it can be reinstated using {Configuration#reinstate_undocumented_behaviour_from_v1_9=}.
103
+ # @overload def stub(stubbed_methods_vs_return_values = {})
104
+ # @param [Hash] stubbed_methods_vs_return_values stubbed method name symbols as keys and corresponding return values as values - these stubbed methods are setup as if {Mock#stubs} were called multiple times.
105
+ # @overload def stub(name, stubbed_methods_vs_return_values = {})
106
+ # @param [String, Symbol] name identifies mock object in error messages.
72
107
  #
73
108
  # @example Using stubbed_methods_vs_return_values Hash to setup stubbed methods.
74
109
  # def test_motor_starts_and_stops
@@ -77,37 +112,46 @@ module Mocha
77
112
  # assert motor.stop
78
113
  # # an error will not be raised even if either Motor#start or Motor#stop has not been called
79
114
  # end
80
- #
81
- # @example Using the optional block to setup expectations & stubbed methods [deprecated].
82
- # def test_motor_starts_and_stops
83
- # motor = stub('motor') do
84
- # expects(:start).with(100.rpm).returns(true)
85
- # stubs(:stop).returns(true)
86
- # end
87
- # assert motor.start(100.rpm)
88
- # assert motor.stop
89
- # # an error will only be raised if Motor#start(100.rpm) has not been called
90
- # end
91
- def stub(*arguments, &block)
92
- name = arguments.shift if arguments.first.is_a?(String)
115
+ # rubocop:disable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
116
+ def stub(*arguments)
117
+ if Mocha.configuration.reinstate_undocumented_behaviour_from_v1_9?
118
+ if arguments.first.is_a?(Symbol)
119
+ method_name = arguments[0]
120
+ Deprecation.warning(
121
+ "Explicitly include `#{method_name}` in Hash of stubbed methods vs return values,",
122
+ " e.g. `stub(:#{method_name} => nil)`."
123
+ )
124
+ if arguments[1]
125
+ Deprecation.warning(
126
+ "In this case the 2nd argument for `stub(:##{method_name}, ...)` is ignored,",
127
+ ' but in the future a Hash of stubbed methods vs return values will be respected.'
128
+ )
129
+ end
130
+ elsif arguments.first.is_a?(String)
131
+ name = arguments.shift
132
+ end
133
+ elsif arguments.first.is_a?(String) || arguments.first.is_a?(Symbol)
134
+ name = arguments.shift
135
+ end
93
136
  expectations = arguments.shift || {}
94
- stub = name ? Mockery.instance.named_mock(name, &block) : Mockery.instance.unnamed_mock(&block)
137
+ stub = name ? Mockery.instance.named_mock(name) : Mockery.instance.unnamed_mock
95
138
  stub.stubs(expectations)
96
139
  stub
97
140
  end
141
+ # rubocop:enable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
98
142
 
99
143
  # Builds a mock object that accepts calls to any method. By default it will return +nil+ for any method call.
100
144
  #
101
- # @param [String] name identifies mock object in error messages.
102
- # @param [Hash] stubbed_methods_vs_return_values stubbed method name symbols as keys and corresponding return values as values - these stubbed methods are setup as if {Mock#stubs} were called multiple times.
103
- # @yield optional block to be evaluated in the context of the mock object instance, giving an alternative way to setup stubbed methods.
104
- # @yield note that the block is evaulated by calling Mock#instance_eval and so things like instance variables declared in the test will not be available within the block.
105
- # @yield deprecated: use Object#tap or define stubs/expectations with an explicit receiver instead.
106
145
  # @return [Mock] a new mock object
107
146
  #
108
- # @overload def stub_everything(name, &block)
109
- # @overload def stub_everything(stubbed_methods_vs_return_values = {}, &block)
110
- # @overload def stub_everything(name, stubbed_methods_vs_return_values = {}, &block)
147
+ # @overload def stub_everything(name)
148
+ # @param [String, Symbol] name identifies mock object in error messages.
149
+ # @note Prior to v1.10.0 when +name+ was a +Symbol+, this method returned an unnamed +Mock+ that stubbed the method identified by +name+. This was undocumented behaviour and it will be removed in the future, but for the moment it can be reinstated using {Configuration#reinstate_undocumented_behaviour_from_v1_9=}.
150
+ # @overload def stub_everything(stubbed_methods_vs_return_values = {})
151
+ # @param [Hash] stubbed_methods_vs_return_values stubbed method name symbols as keys and corresponding return values as values - these stubbed methods are setup as if {Mock#stubs} were called multiple times.
152
+ # @overload def stub_everything(name, stubbed_methods_vs_return_values = {})
153
+ # @param [String, Symbol] name identifies mock object in error messages.
154
+ # @param [Hash] stubbed_methods_vs_return_values stubbed method name symbols as keys and corresponding return values as values - these stubbed methods are setup as if {Mock#stubs} were called multiple times.
111
155
  #
112
156
  # @example Ignore invocations of irrelevant methods.
113
157
  # def test_motor_stops
@@ -116,14 +160,34 @@ module Mocha
116
160
  # assert_nil motor.irrelevant_method_2 # => no error raised
117
161
  # assert motor.stop
118
162
  # end
119
- def stub_everything(*arguments, &block)
120
- name = arguments.shift if arguments.first.is_a?(String)
163
+ # rubocop:disable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
164
+ def stub_everything(*arguments)
165
+ if Mocha.configuration.reinstate_undocumented_behaviour_from_v1_9?
166
+ if arguments.first.is_a?(Symbol)
167
+ method_name = arguments[0]
168
+ Deprecation.warning(
169
+ "Explicitly include `#{method_name}` in Hash of stubbed methods vs return values,",
170
+ " e.g. `stub_everything(:#{method_name} => nil)`."
171
+ )
172
+ if arguments[1]
173
+ Deprecation.warning(
174
+ "In this case the 2nd argument for `stub_everything(:##{method_name}, ...)` is ignored,",
175
+ ' but in the future a Hash of stubbed methods vs return values will be respected.'
176
+ )
177
+ end
178
+ elsif arguments.first.is_a?(String)
179
+ name = arguments.shift
180
+ end
181
+ elsif arguments.first.is_a?(String) || arguments.first.is_a?(Symbol)
182
+ name = arguments.shift
183
+ end
121
184
  expectations = arguments.shift || {}
122
- stub = name ? Mockery.instance.named_mock(name, &block) : Mockery.instance.unnamed_mock(&block)
185
+ stub = name ? Mockery.instance.named_mock(name) : Mockery.instance.unnamed_mock
123
186
  stub.stub_everything
124
187
  stub.stubs(expectations)
125
188
  stub
126
189
  end
190
+ # rubocop:enable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
127
191
 
128
192
  # Builds a new sequence which can be used to constrain the order in which expectations can occur.
129
193
  #
@@ -0,0 +1,31 @@
1
+ module Mocha
2
+ module BlockMatchers
3
+ class OptionalBlock
4
+ def match?(_actual_block)
5
+ true
6
+ end
7
+
8
+ def mocha_inspect; end
9
+ end
10
+
11
+ class BlockGiven
12
+ def match?(actual_block)
13
+ !actual_block.nil?
14
+ end
15
+
16
+ def mocha_inspect
17
+ 'with block given'
18
+ end
19
+ end
20
+
21
+ class NoBlockGiven
22
+ def match?(actual_block)
23
+ actual_block.nil?
24
+ end
25
+
26
+ def mocha_inspect
27
+ 'with no block given'
28
+ end
29
+ end
30
+ end
31
+ end
@@ -26,37 +26,43 @@ module Mocha
26
26
  def initialize(required, maximum)
27
27
  @required = required
28
28
  @maximum = maximum
29
+ @invocations = []
29
30
  end
30
31
 
31
- def invocations_allowed?(invocation_count)
32
- invocation_count < maximum
32
+ def <<(invocation)
33
+ @invocations << invocation
33
34
  end
34
35
 
35
- def satisfied?(invocations_so_far)
36
- invocations_so_far >= required
36
+ def invocations_allowed?
37
+ @invocations.size < maximum
38
+ end
39
+
40
+ def satisfied?
41
+ @invocations.size >= required
37
42
  end
38
43
 
39
44
  def needs_verifying?
40
45
  !allowed_any_number_of_times?
41
46
  end
42
47
 
43
- def verified?(invocation_count)
44
- (invocation_count >= required) && (invocation_count <= maximum)
48
+ def verified?
49
+ (@invocations.size >= required) && (@invocations.size <= maximum)
45
50
  end
46
51
 
47
52
  def allowed_any_number_of_times?
48
53
  required.zero? && infinite?(maximum)
49
54
  end
50
55
 
51
- def used?(invocation_count)
52
- (invocation_count > 0) || maximum.zero?
56
+ def used?
57
+ @invocations.any? || maximum.zero?
53
58
  end
54
59
 
55
- def mocha_inspect
60
+ # rubocop:disable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
61
+ def anticipated_times
56
62
  if allowed_any_number_of_times?
57
63
  'allowed any number of times'
58
64
  elsif required.zero? && maximum.zero?
59
- 'expected never'
65
+ "expected #{times(maximum)}"
60
66
  elsif required == maximum
61
67
  "expected exactly #{times(required)}"
62
68
  elsif infinite?(maximum)
@@ -67,6 +73,15 @@ module Mocha
67
73
  "expected between #{required} and #{times(maximum)}"
68
74
  end
69
75
  end
76
+ # rubocop:enable Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity
77
+
78
+ def invoked_times
79
+ "invoked #{times(@invocations.size)}"
80
+ end
81
+
82
+ def actual_invocations
83
+ @invocations.map(&:full_description).join
84
+ end
70
85
 
71
86
  protected
72
87
 
@@ -74,7 +89,7 @@ module Mocha
74
89
 
75
90
  def times(number)
76
91
  case number
77
- when 0 then 'no times'
92
+ when 0 then 'never'
78
93
  when 1 then 'once'
79
94
  when 2 then 'twice'
80
95
  else "#{number} times"
@@ -1,15 +1,9 @@
1
1
  require 'mocha/mockery'
2
- require 'mocha/class_method'
3
2
  require 'mocha/any_instance_method'
4
3
 
5
4
  module Mocha
6
5
  # Methods added to all classes to allow mocking and stubbing on real (i.e. non-mock) objects.
7
6
  module ClassMethods
8
- # @private
9
- def stubba_method
10
- Mocha::ClassMethod
11
- end
12
-
13
7
  # @private
14
8
  class AnyInstance
15
9
  def initialize(klass)
@@ -28,17 +22,15 @@ module Mocha
28
22
  Mocha::AnyInstanceMethod
29
23
  end
30
24
 
31
- attr_reader :stubba_object
25
+ def stubba_class
26
+ @stubba_object
27
+ end
32
28
 
33
- def method_exists?(method, include_public_methods = true)
34
- if include_public_methods
35
- return true if @stubba_object.public_instance_methods(true).include?(method)
36
- return true if @stubba_object.allocate.respond_to?(method.to_sym)
37
- end
38
- return true if @stubba_object.protected_instance_methods(true).include?(method)
39
- return true if @stubba_object.private_instance_methods(true).include?(method)
40
- false
29
+ def respond_to?(method)
30
+ @stubba_object.allocate.respond_to?(method.to_sym)
41
31
  end
32
+
33
+ attr_reader :stubba_object
42
34
  end
43
35
 
44
36
  # @return [Mock] a mock object which will detect calls to any instance of this class.
@@ -56,5 +48,15 @@ module Mocha
56
48
  end
57
49
  @any_instance ||= AnyInstance.new(self)
58
50
  end
51
+
52
+ # @private
53
+ # rubocop:disable Metrics/CyclomaticComplexity
54
+ def __method_visibility__(method, include_public_methods = true)
55
+ (include_public_methods && public_method_defined?(method) && :public) ||
56
+ (protected_method_defined?(method) && :protected) ||
57
+ (private_method_defined?(method) && :private)
58
+ end
59
+ # rubocop:enable Metrics/CyclomaticComplexity
60
+ alias_method :__method_exists__?, :__method_visibility__
59
61
  end
60
62
  end