jferris-mocha 0.9.5.0.1240002286

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.
Files changed (180) hide show
  1. data/COPYING +3 -0
  2. data/MIT-LICENSE +7 -0
  3. data/README +37 -0
  4. data/RELEASE +269 -0
  5. data/Rakefile +217 -0
  6. data/examples/misc.rb +43 -0
  7. data/examples/mocha.rb +25 -0
  8. data/examples/stubba.rb +64 -0
  9. data/lib/mocha/any_instance_method.rb +55 -0
  10. data/lib/mocha/api.rb +232 -0
  11. data/lib/mocha/argument_iterator.rb +21 -0
  12. data/lib/mocha/backtrace_filter.rb +17 -0
  13. data/lib/mocha/cardinality.rb +95 -0
  14. data/lib/mocha/central.rb +27 -0
  15. data/lib/mocha/change_state_side_effect.rb +19 -0
  16. data/lib/mocha/class_method.rb +87 -0
  17. data/lib/mocha/configuration.rb +60 -0
  18. data/lib/mocha/deprecation.rb +22 -0
  19. data/lib/mocha/exception_raiser.rb +17 -0
  20. data/lib/mocha/expectation.rb +484 -0
  21. data/lib/mocha/expectation_error.rb +15 -0
  22. data/lib/mocha/expectation_list.rb +50 -0
  23. data/lib/mocha/in_state_ordering_constraint.rb +19 -0
  24. data/lib/mocha/inspect.rb +67 -0
  25. data/lib/mocha/instance_method.rb +16 -0
  26. data/lib/mocha/integration/bacon/assertion_counter.rb +23 -0
  27. data/lib/mocha/integration/bacon/version_11_and_above.rb +34 -0
  28. data/lib/mocha/integration/bacon.rb +1 -0
  29. data/lib/mocha/integration/mini_test/assertion_counter.rb +23 -0
  30. data/lib/mocha/integration/mini_test/version_131_and_above.rb +49 -0
  31. data/lib/mocha/integration/mini_test.rb +1 -0
  32. data/lib/mocha/integration/test_unit/assertion_counter.rb +23 -0
  33. data/lib/mocha/integration/test_unit/gem_version_200.rb +49 -0
  34. data/lib/mocha/integration/test_unit/gem_version_201_and_above.rb +49 -0
  35. data/lib/mocha/integration/test_unit/ruby_version_185_and_below.rb +48 -0
  36. data/lib/mocha/integration/test_unit/ruby_version_186_and_above.rb +50 -0
  37. data/lib/mocha/integration/test_unit.rb +4 -0
  38. data/lib/mocha/invocation.rb +10 -0
  39. data/lib/mocha/is_a.rb +9 -0
  40. data/lib/mocha/logger.rb +15 -0
  41. data/lib/mocha/metaclass.rb +13 -0
  42. data/lib/mocha/method_matcher.rb +21 -0
  43. data/lib/mocha/mock.rb +200 -0
  44. data/lib/mocha/mockery.rb +189 -0
  45. data/lib/mocha/module_method.rb +16 -0
  46. data/lib/mocha/multiple_yields.rb +20 -0
  47. data/lib/mocha/names.rb +53 -0
  48. data/lib/mocha/no_yields.rb +11 -0
  49. data/lib/mocha/object.rb +187 -0
  50. data/lib/mocha/parameter_matchers/all_of.rb +42 -0
  51. data/lib/mocha/parameter_matchers/any_of.rb +47 -0
  52. data/lib/mocha/parameter_matchers/any_parameters.rb +40 -0
  53. data/lib/mocha/parameter_matchers/anything.rb +33 -0
  54. data/lib/mocha/parameter_matchers/base.rb +15 -0
  55. data/lib/mocha/parameter_matchers/equals.rb +42 -0
  56. data/lib/mocha/parameter_matchers/has_entries.rb +45 -0
  57. data/lib/mocha/parameter_matchers/has_entry.rb +57 -0
  58. data/lib/mocha/parameter_matchers/has_key.rb +42 -0
  59. data/lib/mocha/parameter_matchers/has_value.rb +42 -0
  60. data/lib/mocha/parameter_matchers/includes.rb +40 -0
  61. data/lib/mocha/parameter_matchers/instance_of.rb +42 -0
  62. data/lib/mocha/parameter_matchers/is_a.rb +42 -0
  63. data/lib/mocha/parameter_matchers/kind_of.rb +42 -0
  64. data/lib/mocha/parameter_matchers/not.rb +42 -0
  65. data/lib/mocha/parameter_matchers/object.rb +15 -0
  66. data/lib/mocha/parameter_matchers/optionally.rb +55 -0
  67. data/lib/mocha/parameter_matchers/regexp_matches.rb +43 -0
  68. data/lib/mocha/parameter_matchers/responds_with.rb +43 -0
  69. data/lib/mocha/parameter_matchers/yaml_equivalent.rb +43 -0
  70. data/lib/mocha/parameter_matchers.rb +27 -0
  71. data/lib/mocha/parameters_matcher.rb +37 -0
  72. data/lib/mocha/pretty_parameters.rb +28 -0
  73. data/lib/mocha/return_values.rb +31 -0
  74. data/lib/mocha/sequence.rb +42 -0
  75. data/lib/mocha/single_return_value.rb +17 -0
  76. data/lib/mocha/single_yield.rb +18 -0
  77. data/lib/mocha/state_machine.rb +91 -0
  78. data/lib/mocha/stubbing_error.rb +16 -0
  79. data/lib/mocha/unexpected_invocation.rb +18 -0
  80. data/lib/mocha/yield_parameters.rb +31 -0
  81. data/lib/mocha.rb +69 -0
  82. data/lib/mocha_standalone.rb +2 -0
  83. data/lib/stubba.rb +4 -0
  84. data/test/acceptance/acceptance_test_helper.rb +38 -0
  85. data/test/acceptance/api_test.rb +139 -0
  86. data/test/acceptance/bacon_spec.rb +67 -0
  87. data/test/acceptance/bacon_test.rb +110 -0
  88. data/test/acceptance/bug_18914_test.rb +43 -0
  89. data/test/acceptance/bug_21465_test.rb +34 -0
  90. data/test/acceptance/bug_21563_test.rb +25 -0
  91. data/test/acceptance/expected_invocation_count_test.rb +196 -0
  92. data/test/acceptance/failure_messages_test.rb +64 -0
  93. data/test/acceptance/minitest_test.rb +146 -0
  94. data/test/acceptance/mocha_example_test.rb +98 -0
  95. data/test/acceptance/mocha_test_result_test.rb +84 -0
  96. data/test/acceptance/mock_test.rb +100 -0
  97. data/test/acceptance/mock_with_initializer_block_test.rb +51 -0
  98. data/test/acceptance/mocked_methods_dispatch_test.rb +78 -0
  99. data/test/acceptance/optional_parameters_test.rb +70 -0
  100. data/test/acceptance/parameter_matcher_test.rb +209 -0
  101. data/test/acceptance/partial_mocks_test.rb +47 -0
  102. data/test/acceptance/return_value_test.rb +52 -0
  103. data/test/acceptance/sequence_test.rb +186 -0
  104. data/test/acceptance/spy_test.rb +109 -0
  105. data/test/acceptance/states_test.rb +70 -0
  106. data/test/acceptance/stub_any_instance_method_test.rb +195 -0
  107. data/test/acceptance/stub_class_method_test.rb +203 -0
  108. data/test/acceptance/stub_everything_test.rb +56 -0
  109. data/test/acceptance/stub_instance_method_test.rb +203 -0
  110. data/test/acceptance/stub_module_method_test.rb +163 -0
  111. data/test/acceptance/stub_test.rb +52 -0
  112. data/test/acceptance/stubba_example_test.rb +102 -0
  113. data/test/acceptance/stubba_test.rb +15 -0
  114. data/test/acceptance/stubba_test_result_test.rb +66 -0
  115. data/test/acceptance/stubbing_error_backtrace_test.rb +64 -0
  116. data/test/acceptance/stubbing_method_unnecessarily_test.rb +65 -0
  117. data/test/acceptance/stubbing_non_existent_any_instance_method_test.rb +130 -0
  118. data/test/acceptance/stubbing_non_existent_class_method_test.rb +157 -0
  119. data/test/acceptance/stubbing_non_existent_instance_method_test.rb +147 -0
  120. data/test/acceptance/stubbing_non_public_any_instance_method_test.rb +130 -0
  121. data/test/acceptance/stubbing_non_public_class_method_test.rb +163 -0
  122. data/test/acceptance/stubbing_non_public_instance_method_test.rb +143 -0
  123. data/test/acceptance/stubbing_on_non_mock_object_test.rb +64 -0
  124. data/test/deprecation_disabler.rb +15 -0
  125. data/test/execution_point.rb +36 -0
  126. data/test/matcher_helpers.rb +5 -0
  127. data/test/method_definer.rb +24 -0
  128. data/test/simple_counter.rb +13 -0
  129. data/test/test_helper.rb +16 -0
  130. data/test/test_runner.rb +33 -0
  131. data/test/unit/any_instance_method_test.rb +126 -0
  132. data/test/unit/array_inspect_test.rb +16 -0
  133. data/test/unit/assert_received_test.rb +136 -0
  134. data/test/unit/backtrace_filter_test.rb +19 -0
  135. data/test/unit/cardinality_test.rb +56 -0
  136. data/test/unit/central_test.rb +65 -0
  137. data/test/unit/change_state_side_effect_test.rb +41 -0
  138. data/test/unit/class_method_test.rb +237 -0
  139. data/test/unit/date_time_inspect_test.rb +21 -0
  140. data/test/unit/exception_raiser_test.rb +42 -0
  141. data/test/unit/expectation_list_test.rb +57 -0
  142. data/test/unit/expectation_test.rb +526 -0
  143. data/test/unit/hash_inspect_test.rb +16 -0
  144. data/test/unit/have_received_test.rb +137 -0
  145. data/test/unit/in_state_ordering_constraint_test.rb +43 -0
  146. data/test/unit/invocation_test.rb +17 -0
  147. data/test/unit/metaclass_test.rb +22 -0
  148. data/test/unit/method_matcher_test.rb +23 -0
  149. data/test/unit/mock_test.rb +329 -0
  150. data/test/unit/mockery_test.rb +163 -0
  151. data/test/unit/multiple_yields_test.rb +18 -0
  152. data/test/unit/no_yields_test.rb +18 -0
  153. data/test/unit/object_inspect_test.rb +37 -0
  154. data/test/unit/object_test.rb +82 -0
  155. data/test/unit/parameter_matchers/all_of_test.rb +26 -0
  156. data/test/unit/parameter_matchers/any_of_test.rb +26 -0
  157. data/test/unit/parameter_matchers/anything_test.rb +21 -0
  158. data/test/unit/parameter_matchers/equals_test.rb +25 -0
  159. data/test/unit/parameter_matchers/has_entries_test.rb +51 -0
  160. data/test/unit/parameter_matchers/has_entry_test.rb +82 -0
  161. data/test/unit/parameter_matchers/has_key_test.rb +36 -0
  162. data/test/unit/parameter_matchers/has_value_test.rb +37 -0
  163. data/test/unit/parameter_matchers/includes_test.rb +25 -0
  164. data/test/unit/parameter_matchers/instance_of_test.rb +25 -0
  165. data/test/unit/parameter_matchers/is_a_test.rb +25 -0
  166. data/test/unit/parameter_matchers/kind_of_test.rb +25 -0
  167. data/test/unit/parameter_matchers/not_test.rb +26 -0
  168. data/test/unit/parameter_matchers/regexp_matches_test.rb +25 -0
  169. data/test/unit/parameter_matchers/responds_with_test.rb +25 -0
  170. data/test/unit/parameter_matchers/stub_matcher.rb +27 -0
  171. data/test/unit/parameter_matchers/yaml_equivalent_test.rb +25 -0
  172. data/test/unit/parameters_matcher_test.rb +121 -0
  173. data/test/unit/return_values_test.rb +63 -0
  174. data/test/unit/sequence_test.rb +104 -0
  175. data/test/unit/single_return_value_test.rb +14 -0
  176. data/test/unit/single_yield_test.rb +18 -0
  177. data/test/unit/state_machine_test.rb +98 -0
  178. data/test/unit/string_inspect_test.rb +11 -0
  179. data/test/unit/yield_parameters_test.rb +93 -0
  180. metadata +245 -0
@@ -0,0 +1,484 @@
1
+ require 'mocha/method_matcher'
2
+ require 'mocha/parameters_matcher'
3
+ require 'mocha/expectation_error'
4
+ require 'mocha/return_values'
5
+ require 'mocha/exception_raiser'
6
+ require 'mocha/yield_parameters'
7
+ require 'mocha/is_a'
8
+ require 'mocha/in_state_ordering_constraint'
9
+ require 'mocha/change_state_side_effect'
10
+ require 'mocha/cardinality'
11
+
12
+ module Mocha # :nodoc:
13
+
14
+ # Methods on expectations returned from Mock#expects, Mock#stubs, Object#expects and Object#stubs.
15
+ class Expectation
16
+
17
+ # :call-seq: times(range) -> expectation
18
+ #
19
+ # Modifies expectation so that the number of calls to the expected method must be within a specific +range+.
20
+ #
21
+ # +range+ can be specified as an exact integer or as a range of integers
22
+ # object = mock()
23
+ # object.expects(:expected_method).times(3)
24
+ # 3.times { object.expected_method }
25
+ # # => verify succeeds
26
+ #
27
+ # object = mock()
28
+ # object.expects(:expected_method).times(3)
29
+ # 2.times { object.expected_method }
30
+ # # => verify fails
31
+ #
32
+ # object = mock()
33
+ # object.expects(:expected_method).times(2..4)
34
+ # 3.times { object.expected_method }
35
+ # # => verify succeeds
36
+ #
37
+ # object = mock()
38
+ # object.expects(:expected_method).times(2..4)
39
+ # object.expected_method
40
+ # # => verify fails
41
+ def times(range)
42
+ @cardinality = Cardinality.times(range)
43
+ self
44
+ end
45
+
46
+ # :call-seq: twice() -> expectation
47
+ #
48
+ # Modifies expectation so that the expected method must be called exactly twice.
49
+ # object = mock()
50
+ # object.expects(:expected_method).twice
51
+ # object.expected_method
52
+ # object.expected_method
53
+ # # => verify succeeds
54
+ #
55
+ # object = mock()
56
+ # object.expects(:expected_method).twice
57
+ # object.expected_method
58
+ # object.expected_method
59
+ # object.expected_method
60
+ # # => verify fails
61
+ #
62
+ # object = mock()
63
+ # object.expects(:expected_method).twice
64
+ # object.expected_method
65
+ # # => verify fails
66
+ def twice
67
+ @cardinality = Cardinality.exactly(2)
68
+ self
69
+ end
70
+
71
+ # :call-seq: once() -> expectation
72
+ #
73
+ # Modifies expectation so that the expected method must be called exactly once.
74
+ # Note that this is the default behaviour for an expectation, but you may wish to use it for clarity/emphasis.
75
+ # object = mock()
76
+ # object.expects(:expected_method).once
77
+ # object.expected_method
78
+ # # => verify succeeds
79
+ #
80
+ # object = mock()
81
+ # object.expects(:expected_method).once
82
+ # object.expected_method
83
+ # object.expected_method
84
+ # # => verify fails
85
+ #
86
+ # object = mock()
87
+ # object.expects(:expected_method).once
88
+ # # => verify fails
89
+ def once
90
+ @cardinality = Cardinality.exactly(1)
91
+ self
92
+ end
93
+
94
+ # :call-seq: never() -> expectation
95
+ #
96
+ # Modifies expectation so that the expected method must never be called.
97
+ # object = mock()
98
+ # object.expects(:expected_method).never
99
+ # object.expected_method
100
+ # # => verify fails
101
+ #
102
+ # object = mock()
103
+ # object.expects(:expected_method).never
104
+ # object.expected_method
105
+ # # => verify succeeds
106
+ def never
107
+ @cardinality = Cardinality.exactly(0)
108
+ self
109
+ end
110
+
111
+ # :call-seq: at_least(minimum_number_of_times) -> expectation
112
+ #
113
+ # Modifies expectation so that the expected method must be called at least a +minimum_number_of_times+.
114
+ # object = mock()
115
+ # object.expects(:expected_method).at_least(2)
116
+ # 3.times { object.expected_method }
117
+ # # => verify succeeds
118
+ #
119
+ # object = mock()
120
+ # object.expects(:expected_method).at_least(2)
121
+ # object.expected_method
122
+ # # => verify fails
123
+ def at_least(minimum_number_of_times)
124
+ @cardinality = Cardinality.at_least(minimum_number_of_times)
125
+ self
126
+ end
127
+
128
+ # :call-seq: at_least_once() -> expectation
129
+ #
130
+ # Modifies expectation so that the expected method must be called at least once.
131
+ # object = mock()
132
+ # object.expects(:expected_method).at_least_once
133
+ # object.expected_method
134
+ # # => verify succeeds
135
+ #
136
+ # object = mock()
137
+ # object.expects(:expected_method).at_least_once
138
+ # # => verify fails
139
+ def at_least_once
140
+ at_least(1)
141
+ self
142
+ end
143
+
144
+ # :call-seq: at_most(maximum_number_of_times) -> expectation
145
+ #
146
+ # Modifies expectation so that the expected method must be called at most a +maximum_number_of_times+.
147
+ # object = mock()
148
+ # object.expects(:expected_method).at_most(2)
149
+ # 2.times { object.expected_method }
150
+ # # => verify succeeds
151
+ #
152
+ # object = mock()
153
+ # object.expects(:expected_method).at_most(2)
154
+ # 3.times { object.expected_method }
155
+ # # => verify fails
156
+ def at_most(maximum_number_of_times)
157
+ @cardinality = Cardinality.at_most(maximum_number_of_times)
158
+ self
159
+ end
160
+
161
+ # :call-seq: at_most_once() -> expectation
162
+ #
163
+ # Modifies expectation so that the expected method must be called at most once.
164
+ # object = mock()
165
+ # object.expects(:expected_method).at_most_once
166
+ # object.expected_method
167
+ # # => verify succeeds
168
+ #
169
+ # object = mock()
170
+ # object.expects(:expected_method).at_most_once
171
+ # 2.times { object.expected_method }
172
+ # # => verify fails
173
+ def at_most_once()
174
+ at_most(1)
175
+ self
176
+ end
177
+
178
+ # :call-seq: with(*expected_parameters, &matching_block) -> expectation
179
+ #
180
+ # Modifies expectation so that the expected method must be called with +expected_parameters+.
181
+ # object = mock()
182
+ # object.expects(:expected_method).with(:param1, :param2)
183
+ # object.expected_method(:param1, :param2)
184
+ # # => verify succeeds
185
+ #
186
+ # object = mock()
187
+ # object.expects(:expected_method).with(:param1, :param2)
188
+ # object.expected_method(:param3)
189
+ # # => verify fails
190
+ # May be used with parameter matchers in Mocha::ParameterMatchers.
191
+ #
192
+ # If a +matching_block+ is given, the block is called with the parameters passed to the expected method.
193
+ # The expectation is matched if the block evaluates to +true+.
194
+ # object = mock()
195
+ # object.expects(:expected_method).with() { |value| value % 4 == 0 }
196
+ # object.expected_method(16)
197
+ # # => verify succeeds
198
+ #
199
+ # object = mock()
200
+ # object.expects(:expected_method).with() { |value| value % 4 == 0 }
201
+ # object.expected_method(17)
202
+ # # => verify fails
203
+ def with(*expected_parameters, &matching_block)
204
+ @parameters_matcher = ParametersMatcher.new(expected_parameters, &matching_block)
205
+ self
206
+ end
207
+
208
+ # :call-seq: yields(*parameters) -> expectation
209
+ #
210
+ # Modifies expectation so that when the expected method is called, it yields with the specified +parameters+.
211
+ # object = mock()
212
+ # object.expects(:expected_method).yields('result')
213
+ # yielded_value = nil
214
+ # object.expected_method { |value| yielded_value = value }
215
+ # yielded_value # => 'result'
216
+ # May be called multiple times on the same expectation for consecutive invocations. Also see Expectation#then.
217
+ # object = mock()
218
+ # object.stubs(:expected_method).yields(1).then.yields(2)
219
+ # yielded_values_from_first_invocation = []
220
+ # yielded_values_from_second_invocation = []
221
+ # object.expected_method { |value| yielded_values_from_first_invocation << value } # first invocation
222
+ # object.expected_method { |value| yielded_values_from_second_invocation << value } # second invocation
223
+ # yielded_values_from_first_invocation # => [1]
224
+ # yielded_values_from_second_invocation # => [2]
225
+ def yields(*parameters)
226
+ @yield_parameters.add(*parameters)
227
+ self
228
+ end
229
+
230
+ # :call-seq: multiple_yields(*parameter_groups) -> expectation
231
+ #
232
+ # Modifies expectation so that when the expected method is called, it yields multiple times per invocation with the specified +parameter_groups+.
233
+ # object = mock()
234
+ # object.expects(:expected_method).multiple_yields(['result_1', 'result_2'], ['result_3'])
235
+ # yielded_values = []
236
+ # object.expected_method { |*values| yielded_values << values }
237
+ # yielded_values # => [['result_1', 'result_2'], ['result_3]]
238
+ # May be called multiple times on the same expectation for consecutive invocations. Also see Expectation#then.
239
+ # object = mock()
240
+ # object.stubs(:expected_method).multiple_yields([1, 2], [3]).then.multiple_yields([4], [5, 6])
241
+ # yielded_values_from_first_invocation = []
242
+ # yielded_values_from_second_invocation = []
243
+ # object.expected_method { |*values| yielded_values_from_first_invocation << values } # first invocation
244
+ # object.expected_method { |*values| yielded_values_from_second_invocation << values } # second invocation
245
+ # yielded_values_from_first_invocation # => [[1, 2], [3]]
246
+ # yielded_values_from_second_invocation # => [[4], [5, 6]]
247
+ def multiple_yields(*parameter_groups)
248
+ @yield_parameters.multiple_add(*parameter_groups)
249
+ self
250
+ end
251
+
252
+ # :call-seq: returns(value) -> expectation
253
+ # returns(*values) -> expectation
254
+ #
255
+ # Modifies expectation so that when the expected method is called, it returns the specified +value+.
256
+ # object = mock()
257
+ # object.stubs(:stubbed_method).returns('result')
258
+ # object.stubbed_method # => 'result'
259
+ # object.stubbed_method # => 'result'
260
+ # If multiple +values+ are given, these are returned in turn on consecutive calls to the method.
261
+ # object = mock()
262
+ # object.stubs(:stubbed_method).returns(1, 2)
263
+ # object.stubbed_method # => 1
264
+ # object.stubbed_method # => 2
265
+ # May be called multiple times on the same expectation. Also see Expectation#then.
266
+ # object = mock()
267
+ # object.stubs(:expected_method).returns(1, 2).then.returns(3)
268
+ # object.expected_method # => 1
269
+ # object.expected_method # => 2
270
+ # object.expected_method # => 3
271
+ # May be called in conjunction with Expectation#raises on the same expectation.
272
+ # object = mock()
273
+ # object.stubs(:expected_method).returns(1, 2).then.raises(Exception)
274
+ # object.expected_method # => 1
275
+ # object.expected_method # => 2
276
+ # object.expected_method # => raises exception of class Exception1
277
+ # Note that in Ruby a method returning multiple values is exactly equivalent to a method returning an Array of those values.
278
+ # object = mock()
279
+ # object.stubs(:expected_method).returns([1, 2])
280
+ # x, y = object.expected_method
281
+ # x # => 1
282
+ # y # => 2
283
+ def returns(*values)
284
+ @return_values += ReturnValues.build(*values)
285
+ self
286
+ end
287
+
288
+ # :call-seq: raises(exception = RuntimeError, message = nil) -> expectation
289
+ #
290
+ # Modifies expectation so that when the expected method is called, it raises the specified +exception+ with the specified +message+.
291
+ # object = mock()
292
+ # object.expects(:expected_method).raises(Exception, 'message')
293
+ # object.expected_method # => raises exception of class Exception and with message 'message'
294
+ # May be called multiple times on the same expectation. Also see Expectation#then.
295
+ # object = mock()
296
+ # object.stubs(:expected_method).raises(Exception1).then.raises(Exception2)
297
+ # object.expected_method # => raises exception of class Exception1
298
+ # object.expected_method # => raises exception of class Exception2
299
+ # May be called in conjunction with Expectation#returns on the same expectation.
300
+ # object = mock()
301
+ # object.stubs(:expected_method).raises(Exception).then.returns(2, 3)
302
+ # object.expected_method # => raises exception of class Exception1
303
+ # object.expected_method # => 2
304
+ # object.expected_method # => 3
305
+ def raises(exception = RuntimeError, message = nil)
306
+ @return_values += ReturnValues.new(ExceptionRaiser.new(exception, message))
307
+ self
308
+ end
309
+
310
+ # :call-seq: then() -> expectation
311
+ # then(state_machine.is(state)) -> expectation
312
+ #
313
+ # <tt>then()</tt> is used as syntactic sugar to improve readability. It has no effect on state of the expectation.
314
+ # object = mock()
315
+ # object.stubs(:expected_method).returns(1, 2).then.raises(Exception).then.returns(4)
316
+ # object.expected_method # => 1
317
+ # object.expected_method # => 2
318
+ # object.expected_method # => raises exception of class Exception
319
+ # object.expected_method # => 4
320
+ #
321
+ # <tt>then(state_machine.is(state))</tt> is used to change the +state_machine+ to the specified +state+ when the invocation occurs.
322
+ #
323
+ # See also API#states, StateMachine and Expectation#when.
324
+ # power = states('power').starts_as('off')
325
+ #
326
+ # radio = mock('radio')
327
+ # radio.expects(:switch_on).then(power.is('on'))
328
+ # radio.expects(:select_channel).with('BBC Radio 4').when(power.is('on'))
329
+ # radio.expects(:adjust_volume).with(+5).when(power.is('on'))
330
+ # radio.expects(:select_channel).with('BBC World Service').when(power.is('on'))
331
+ # radio.expects(:adjust_volume).with(-5).when(power.is('on'))
332
+ # radio.expects(:switch_off).then(power.is('off'))
333
+ def then(*parameters)
334
+ if parameters.length == 1
335
+ state = parameters.first
336
+ add_side_effect(ChangeStateSideEffect.new(state))
337
+ end
338
+ self
339
+ end
340
+
341
+ # :call-seq: when(state_machine.is(state)) -> exception
342
+ #
343
+ # Constrains the expectation to occur only when the +state_machine+ is in the named +state+.
344
+ #
345
+ # See also API#states, StateMachine#starts_as and Expectation#then.
346
+ # power = states('power').starts_as('off')
347
+ #
348
+ # radio = mock('radio')
349
+ # radio.expects(:switch_on).then(power.is('on'))
350
+ # radio.expects(:select_channel).with('BBC Radio 4').when(power.is('on'))
351
+ # radio.expects(:adjust_volume).with(+5).when(power.is('on'))
352
+ # radio.expects(:select_channel).with('BBC World Service').when(power.is('on'))
353
+ # radio.expects(:adjust_volume).with(-5).when(power.is('on'))
354
+ # radio.expects(:switch_off).then(power.is('off'))
355
+ def when(state_predicate)
356
+ add_ordering_constraint(InStateOrderingConstraint.new(state_predicate))
357
+ self
358
+ end
359
+
360
+ # :call-seq: in_sequence(*sequences) -> expectation
361
+ #
362
+ # Constrains this expectation so that it must be invoked at the current point in the sequence.
363
+ #
364
+ # To expect a sequence of invocations, write the expectations in order and add the in_sequence(sequence) clause to each one.
365
+ #
366
+ # Expectations in a sequence can have any invocation count.
367
+ #
368
+ # If an expectation in a sequence is stubbed, rather than expected, it can be skipped in the sequence.
369
+ #
370
+ # See also API#sequence.
371
+ # breakfast = sequence('breakfast')
372
+ #
373
+ # egg = mock('egg')
374
+ # egg.expects(:crack).in_sequence(breakfast)
375
+ # egg.expects(:fry).in_sequence(breakfast)
376
+ # egg.expects(:eat).in_sequence(breakfast)
377
+ def in_sequence(*sequences)
378
+ sequences.each { |sequence| add_in_sequence_ordering_constraint(sequence) }
379
+ self
380
+ end
381
+
382
+ # :stopdoc:
383
+
384
+ attr_reader :backtrace
385
+ attr_accessor :invocation_count
386
+
387
+ def initialize(mock, expected_method_name, backtrace = nil)
388
+ @mock = mock
389
+ @method_matcher = MethodMatcher.new(expected_method_name.to_sym)
390
+ @parameters_matcher = ParametersMatcher.new
391
+ @ordering_constraints = []
392
+ @side_effects = []
393
+ @cardinality, @invocation_count = Cardinality.exactly(1), 0
394
+ @return_values = ReturnValues.new
395
+ @yield_parameters = YieldParameters.new
396
+ @backtrace = backtrace || caller
397
+ end
398
+
399
+ def add_ordering_constraint(ordering_constraint)
400
+ @ordering_constraints << ordering_constraint
401
+ end
402
+
403
+ def add_in_sequence_ordering_constraint(sequence)
404
+ sequence.constrain_as_next_in_sequence(self)
405
+ end
406
+
407
+ def add_side_effect(side_effect)
408
+ @side_effects << side_effect
409
+ end
410
+
411
+ def perform_side_effects
412
+ @side_effects.each { |side_effect| side_effect.perform }
413
+ end
414
+
415
+ def in_correct_order?
416
+ @ordering_constraints.all? { |ordering_constraint| ordering_constraint.allows_invocation_now? }
417
+ end
418
+
419
+ def matches_method?(method_name)
420
+ @method_matcher.match?(method_name)
421
+ end
422
+
423
+ def match?(actual_method_name, *actual_parameters)
424
+ @method_matcher.match?(actual_method_name) && @parameters_matcher.match?(actual_parameters) && in_correct_order?
425
+ end
426
+
427
+ def invocations_allowed?
428
+ @cardinality.invocations_allowed?(@invocation_count)
429
+ end
430
+
431
+ def satisfied?
432
+ @cardinality.satisfied?(@invocation_count)
433
+ end
434
+
435
+ def invoke(args)
436
+ @invocation_count += 1
437
+ Mockery.instance.invocation(@mock, method_name, args)
438
+ perform_side_effects()
439
+ if block_given? then
440
+ @yield_parameters.next_invocation.each do |yield_parameters|
441
+ yield(*yield_parameters)
442
+ end
443
+ end
444
+ @return_values.next
445
+ end
446
+
447
+ def verified?(assertion_counter = nil)
448
+ assertion_counter.increment if assertion_counter && @cardinality.needs_verifying?
449
+ @cardinality.verified?(@invocation_count)
450
+ end
451
+
452
+ def used?
453
+ @cardinality.used?(@invocation_count)
454
+ end
455
+
456
+ def mocha_inspect
457
+ message = "#{@cardinality.mocha_inspect}, "
458
+ message << case @invocation_count
459
+ when 0 then "not yet invoked"
460
+ when 1 then "already invoked once"
461
+ when 2 then "already invoked twice"
462
+ else "already invoked #{@invocation_count} times"
463
+ end
464
+ message << ": "
465
+ message << method_signature
466
+ message << "; #{@ordering_constraints.map { |oc| oc.mocha_inspect }.join("; ")}" unless @ordering_constraints.empty?
467
+ message
468
+ end
469
+
470
+ def method_signature
471
+ "#{@mock.mocha_inspect}.#{@method_matcher.mocha_inspect}#{@parameters_matcher.mocha_inspect}"
472
+ end
473
+
474
+ private
475
+
476
+ def method_name
477
+ @method_matcher.expected_method_name
478
+ end
479
+
480
+ # :startdoc:
481
+
482
+ end
483
+
484
+ end
@@ -0,0 +1,15 @@
1
+ require 'mocha/backtrace_filter'
2
+
3
+ module Mocha
4
+
5
+ class ExpectationError < StandardError
6
+
7
+ def initialize(message = nil, backtrace = [])
8
+ super(message)
9
+ filter = BacktraceFilter.new
10
+ set_backtrace(filter.filtered(backtrace))
11
+ end
12
+
13
+ end
14
+
15
+ end
@@ -0,0 +1,50 @@
1
+ module Mocha # :nodoc:
2
+
3
+ class ExpectationList
4
+
5
+ def initialize
6
+ @expectations = []
7
+ end
8
+
9
+ def add(expectation)
10
+ @expectations.unshift(expectation)
11
+ expectation
12
+ end
13
+
14
+ def matches_method?(method_name)
15
+ @expectations.any? { |expectation| expectation.matches_method?(method_name) }
16
+ end
17
+
18
+ def match(method_name, *arguments)
19
+ matching_expectations(method_name, *arguments).first
20
+ end
21
+
22
+ def match_allowing_invocation(method_name, *arguments)
23
+ matching_expectations(method_name, *arguments).detect { |e| e.invocations_allowed? }
24
+ end
25
+
26
+ def verified?(assertion_counter = nil)
27
+ @expectations.all? { |expectation| expectation.verified?(assertion_counter) }
28
+ end
29
+
30
+ def to_a
31
+ @expectations
32
+ end
33
+
34
+ def to_set
35
+ @expectations.to_set
36
+ end
37
+
38
+ def length
39
+ @expectations.length
40
+ end
41
+
42
+ private
43
+
44
+ def matching_expectations(method_name, *arguments)
45
+ @expectations.select { |e| e.match?(method_name, *arguments) }
46
+ end
47
+
48
+ end
49
+
50
+ end
@@ -0,0 +1,19 @@
1
+ module Mocha
2
+
3
+ class InStateOrderingConstraint
4
+
5
+ def initialize(state_predicate)
6
+ @state_predicate = state_predicate
7
+ end
8
+
9
+ def allows_invocation_now?
10
+ @state_predicate.active?
11
+ end
12
+
13
+ def mocha_inspect
14
+ "when #{@state_predicate.mocha_inspect}"
15
+ end
16
+
17
+ end
18
+
19
+ end
@@ -0,0 +1,67 @@
1
+ require 'date'
2
+
3
+ module Mocha
4
+
5
+ module ObjectMethods
6
+ def mocha_inspect
7
+ address = self.__id__ * 2
8
+ address += 0x100000000 if address < 0
9
+ inspect =~ /#</ ? "#<#{self.class}:0x#{'%x' % address}>" : inspect
10
+ end
11
+ end
12
+
13
+ module StringMethods
14
+ def mocha_inspect
15
+ inspect.gsub(/\"/, "'")
16
+ end
17
+ end
18
+
19
+ module ArrayMethods
20
+ def mocha_inspect
21
+ "[#{collect { |member| member.mocha_inspect }.join(', ')}]"
22
+ end
23
+ end
24
+
25
+ module HashMethods
26
+ def mocha_inspect
27
+ "{#{collect { |key, value| "#{key.mocha_inspect} => #{value.mocha_inspect}" }.join(', ')}}"
28
+ end
29
+ end
30
+
31
+ module TimeMethods
32
+ def mocha_inspect
33
+ "#{inspect} (#{to_f} secs)"
34
+ end
35
+ end
36
+
37
+ module DateMethods
38
+ def mocha_inspect
39
+ to_s
40
+ end
41
+ end
42
+
43
+ end
44
+
45
+ class Object
46
+ include Mocha::ObjectMethods
47
+ end
48
+
49
+ class String
50
+ include Mocha::StringMethods
51
+ end
52
+
53
+ class Array
54
+ include Mocha::ArrayMethods
55
+ end
56
+
57
+ class Hash
58
+ include Mocha::HashMethods
59
+ end
60
+
61
+ class Time
62
+ include Mocha::TimeMethods
63
+ end
64
+
65
+ class Date
66
+ include Mocha::DateMethods
67
+ end
@@ -0,0 +1,16 @@
1
+ require 'mocha/class_method'
2
+
3
+ module Mocha
4
+
5
+ class InstanceMethod < ClassMethod
6
+
7
+ def method_exists?(method)
8
+ return true if stubbee.public_methods(false).include?(method)
9
+ return true if stubbee.protected_methods(false).include?(method)
10
+ return true if stubbee.private_methods(false).include?(method)
11
+ return false
12
+ end
13
+
14
+ end
15
+
16
+ end
@@ -0,0 +1,23 @@
1
+ module Mocha
2
+
3
+ module Integration
4
+
5
+ module Bacon
6
+
7
+ class AssertionCounter
8
+
9
+ def initialize(counter)
10
+ @counter = counter
11
+ end
12
+
13
+ def increment
14
+ @counter[:requirements] += 1
15
+ end
16
+
17
+ end
18
+
19
+ end
20
+
21
+ end
22
+
23
+ end