flexmock 1.3.0 → 1.3.1
Sign up to get free protection for your applications and to get access to all the features.
- data/Gemfile +5 -2
- data/Gemfile.lock +15 -16
- data/README.md +22 -7
- data/TAGS +456 -2457
- data/doc/index.rdoc +1 -1
- data/doc/releases/flexmock-1.3.1.rdoc +171 -0
- data/lib/flexmock/core.rb +5 -0
- data/lib/flexmock/core_class_methods.rb +1 -1
- data/lib/flexmock/default_framework_adapter.rb +2 -2
- data/lib/flexmock/explicit_needed.rb +5 -1
- data/lib/flexmock/mock_container.rb +5 -1
- data/lib/flexmock/partial_mock.rb +5 -0
- data/lib/flexmock/rspec.rb +2 -2
- data/lib/flexmock/test_unit_integration.rb +3 -3
- data/lib/flexmock/validators.rb +26 -28
- data/lib/flexmock/version.rb +1 -1
- data/test/aliasing_test.rb +49 -42
- data/test/based_partials_test.rb +51 -0
- data/test/default_framework_adapter_test.rb +4 -4
- data/test/partial_mock_test.rb +15 -0
- data/test/spys_test.rb +7 -7
- data/test/test_setup.rb +8 -2
- metadata +13 -13
data/doc/index.rdoc
CHANGED
@@ -4,7 +4,7 @@ FlexMock is a simple, but flexible, mock object library for Ruby unit
|
|
4
4
|
testing.
|
5
5
|
|
6
6
|
This is the API documentation site for flexmock. You can find the user
|
7
|
-
documentation at http://github.com/jimweirich/flexmock
|
7
|
+
documentation at http://github.com/jimweirich/flexmock
|
8
8
|
|
9
9
|
== Links
|
10
10
|
|
@@ -0,0 +1,171 @@
|
|
1
|
+
= FlexMock 1.3.1 Released
|
2
|
+
|
3
|
+
FlexMock is a flexible mocking library for use in unit testing and
|
4
|
+
behavior specification in Ruby. This release is a minor release with a
|
5
|
+
few bug fixes and some simple features.
|
6
|
+
|
7
|
+
== Changes in 1.3.1
|
8
|
+
|
9
|
+
* Removed use of assert_block (which is deprecated in MiniTest).
|
10
|
+
|
11
|
+
* Documentation fixes.
|
12
|
+
|
13
|
+
== Changes in 1.3.0
|
14
|
+
|
15
|
+
=== Features
|
16
|
+
|
17
|
+
* Add 'and' and 'on' modifiers for the RSpec spy matcher.
|
18
|
+
|
19
|
+
* Add 'and' and 'on' options to the assert_spy_called test method.
|
20
|
+
|
21
|
+
* General documentation improvement.
|
22
|
+
|
23
|
+
=== Bug Fixes
|
24
|
+
|
25
|
+
* Fix bug in should_fail test helper that was not detecting failed
|
26
|
+
failures.
|
27
|
+
|
28
|
+
== Changes in 1.2.0
|
29
|
+
|
30
|
+
=== Features
|
31
|
+
|
32
|
+
* Add auto configure by requiring 'flexmock/rspec/configure'.
|
33
|
+
|
34
|
+
== Changes in 1.1.0
|
35
|
+
|
36
|
+
=== Features
|
37
|
+
|
38
|
+
* Add block support to pass_thru.
|
39
|
+
|
40
|
+
== Changes in 1.0.0
|
41
|
+
|
42
|
+
=== Features
|
43
|
+
|
44
|
+
* Mocks may now have a base class that limits what methods may be
|
45
|
+
mocked. This allows early detection of outdated mock setups when the
|
46
|
+
methods in the class are refactored.
|
47
|
+
|
48
|
+
* Spy assertions are now allowed. The verification of the calling of
|
49
|
+
mocked methods may now be done in the "then" portion of the test,
|
50
|
+
after the code under test has been run. This allows for much more
|
51
|
+
natural Given/When/Then style testing.
|
52
|
+
|
53
|
+
* A custom assert method (assert_spy_called) has been added to make
|
54
|
+
spy assertions easy when using Test::Unit or MiniTest.
|
55
|
+
|
56
|
+
* An RSpec matcher (have_received) has been added to make spy
|
57
|
+
assertions easy when using RSpec.
|
58
|
+
|
59
|
+
=== Bug Fixes
|
60
|
+
|
61
|
+
* Now correctly handling the mocking of meta-programmed methods.
|
62
|
+
|
63
|
+
* Using the documented +singleton_methods+ method.
|
64
|
+
|
65
|
+
* Accidently trying to partial mock a regular mock is now a no-op.
|
66
|
+
|
67
|
+
== What is FlexMock?
|
68
|
+
|
69
|
+
FlexMock is a flexible framework for creating mock object for testing.
|
70
|
+
When running unit tests, it is often desirable to use isolate the
|
71
|
+
objects being tested from the "real world" by having them interact
|
72
|
+
with simplified test objects. Sometimes these test objects simply
|
73
|
+
return values when called, other times they verify that certain
|
74
|
+
methods were called with particular arguments in a particular order.
|
75
|
+
|
76
|
+
FlexMock makes creating these test objects easy.
|
77
|
+
|
78
|
+
=== Features
|
79
|
+
|
80
|
+
* Easy integration with both Test::Unit and RSpec. Mocks created with the
|
81
|
+
flexmock method are automatically verified at the end of the test or
|
82
|
+
example.
|
83
|
+
|
84
|
+
* A fluent interface that allows mock behavior to be specified very
|
85
|
+
easily.
|
86
|
+
|
87
|
+
* A "record mode" where an existing implementation can record its
|
88
|
+
interaction with a mock for later validation against a new
|
89
|
+
implementation.
|
90
|
+
|
91
|
+
* Easy mocking of individual methods in existing, non-mock objects.
|
92
|
+
|
93
|
+
* Easy mocking of chains of method calls.
|
94
|
+
|
95
|
+
* The ability to cause classes to instantiate test instances (instead of real
|
96
|
+
instances) for the duration of a test.
|
97
|
+
|
98
|
+
=== Example
|
99
|
+
|
100
|
+
Suppose you had a Dog object that wagged a tail when it was happy.
|
101
|
+
Something like this:
|
102
|
+
|
103
|
+
class Dog
|
104
|
+
def initialize(a_tail)
|
105
|
+
@tail = a_tail
|
106
|
+
end
|
107
|
+
def happy
|
108
|
+
@tail.wag
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
To test the +Dog+ class without a real +Tail+ object (perhaps because
|
113
|
+
real +Tail+ objects activate servos in some robotic equipment), you
|
114
|
+
can do something like this:
|
115
|
+
|
116
|
+
RSpec.configure do |config|
|
117
|
+
config.mock_with :flexmock
|
118
|
+
end
|
119
|
+
|
120
|
+
describe Dog do
|
121
|
+
it "wags its tail when happy" do
|
122
|
+
tail = flexmock("tail")
|
123
|
+
tail.should_receive(:wag).once
|
124
|
+
dog = Dog.new(tail)
|
125
|
+
dog.happy
|
126
|
+
end
|
127
|
+
end
|
128
|
+
|
129
|
+
FlexMock will automatically verify that the mocked tail object received the
|
130
|
+
message +wag+ exactly one time. If it doesn't, the test will not pass.
|
131
|
+
|
132
|
+
Here's the same thing using the new spy support:
|
133
|
+
|
134
|
+
describe Dog do
|
135
|
+
it "wags its tail when happy" do
|
136
|
+
tail = flexmock("tail")
|
137
|
+
dog = Dog.new(tail)
|
138
|
+
dog.happy
|
139
|
+
tail.should have_received(:wag)
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
This style works particularly well with the rspec-given library.
|
144
|
+
|
145
|
+
require 'rspec/given'
|
146
|
+
|
147
|
+
describe Dog do
|
148
|
+
context "when the dog is happy" do
|
149
|
+
Given(:tail) { flexmock(:on, Tail) }
|
150
|
+
Given(:dog) { Dog.new(tail) }
|
151
|
+
|
152
|
+
When { dog.happy }
|
153
|
+
|
154
|
+
Then { tail.should have_received(:wag) }
|
155
|
+
end
|
156
|
+
end
|
157
|
+
|
158
|
+
See the FlexMock documentation at http://flexmock.rubyforge.org for details on
|
159
|
+
specifying arguments and return values on mocked methods, as well as a simple
|
160
|
+
technique for mocking tail objects when the Dog class creates the tail objects
|
161
|
+
directly.
|
162
|
+
|
163
|
+
== Availability
|
164
|
+
|
165
|
+
You can make sure you have the latest version with a quick RubyGems command:
|
166
|
+
|
167
|
+
gem install flexmock (you may need root/admin privileges)
|
168
|
+
|
169
|
+
You will find documentation at: http://flexmock.rubyforge.org.
|
170
|
+
|
171
|
+
-- Jim Weirich
|
data/lib/flexmock/core.rb
CHANGED
@@ -52,6 +52,11 @@ class FlexMock
|
|
52
52
|
attr_reader :flexmock_name
|
53
53
|
attr_accessor :flexmock_container
|
54
54
|
|
55
|
+
class << self
|
56
|
+
attr_accessor :partials_are_based
|
57
|
+
end
|
58
|
+
self.partials_are_based = false
|
59
|
+
|
55
60
|
# Create a FlexMock object with the given name. The name is used in
|
56
61
|
# error messages. If no container is given, create a new, one-off
|
57
62
|
# container for this mock.
|
@@ -67,7 +67,7 @@ class FlexMock
|
|
67
67
|
# Check will assert the block returns true. If it doesn't, an
|
68
68
|
# assertion failure is triggered with the given message.
|
69
69
|
def check(msg, &block) # :nodoc:
|
70
|
-
FlexMock.framework_adapter.
|
70
|
+
FlexMock.framework_adapter.make_assertion(msg, &block)
|
71
71
|
end
|
72
72
|
|
73
73
|
end
|
@@ -13,7 +13,7 @@ require 'flexmock/noop'
|
|
13
13
|
|
14
14
|
class FlexMock
|
15
15
|
class DefaultFrameworkAdapter
|
16
|
-
def
|
16
|
+
def make_assertion(msg, &block)
|
17
17
|
unless yield
|
18
18
|
msg = msg.call if msg.is_a?(Proc)
|
19
19
|
fail assertion_failed_error, msg
|
@@ -21,7 +21,7 @@ class FlexMock
|
|
21
21
|
end
|
22
22
|
|
23
23
|
def assert_equal(a, b, msg=nil)
|
24
|
-
|
24
|
+
make_assertion(msg || "Expected equal") { a == b }
|
25
25
|
end
|
26
26
|
|
27
27
|
class AssertionFailedError < StandardError; end
|
@@ -5,7 +5,7 @@ class FlexMock
|
|
5
5
|
# methods supported by the base class. Attempting to add an stub to
|
6
6
|
# a method not defined on the base class will cause the expectation
|
7
7
|
# to be wrapped in an ExplicitNeeded wrapper. The wrapper will throw
|
8
|
-
# an exception unless the
|
8
|
+
# an exception unless the explicitly method is immediately called on
|
9
9
|
# the expectation.
|
10
10
|
#
|
11
11
|
class ExplicitNeeded
|
@@ -25,6 +25,10 @@ class FlexMock
|
|
25
25
|
@explicit
|
26
26
|
end
|
27
27
|
|
28
|
+
def mock=(m)
|
29
|
+
@expectation.mock = m
|
30
|
+
end
|
31
|
+
|
28
32
|
def method_missing(sym, *args, &block)
|
29
33
|
if explicit?
|
30
34
|
@expectation.send(sym, *args, &block)
|
@@ -160,7 +160,11 @@ class FlexMock
|
|
160
160
|
mock ||= FlexMock.new(name || "unknown", self)
|
161
161
|
result = mock
|
162
162
|
end
|
163
|
-
|
163
|
+
if base_class
|
164
|
+
mock.flexmock_based_on(base_class)
|
165
|
+
elsif domain_obj && FlexMock.partials_are_based
|
166
|
+
mock.flexmock_based_on(domain_obj.class)
|
167
|
+
end
|
164
168
|
mock.flexmock_define_expectation(location, quick_defs)
|
165
169
|
yield(mock) if block_given?
|
166
170
|
flexmock_remember(mock)
|
@@ -205,6 +205,11 @@ class FlexMock
|
|
205
205
|
@mock.flexmock_expectations_for(method_name)
|
206
206
|
end
|
207
207
|
|
208
|
+
# Forward the based on request.
|
209
|
+
def flexmock_based_on(*args)
|
210
|
+
@mock.flexmock_based_on(*args)
|
211
|
+
end
|
212
|
+
|
208
213
|
private
|
209
214
|
|
210
215
|
def check_allocate_method(allocate_method)
|
data/lib/flexmock/rspec.rb
CHANGED
@@ -21,14 +21,14 @@ class FlexMock
|
|
21
21
|
end
|
22
22
|
|
23
23
|
class RSpecFrameworkAdapter
|
24
|
-
def
|
24
|
+
def make_assertion(msg, &block)
|
25
25
|
msg = msg.call if msg.is_a?(Proc)
|
26
26
|
SpecModule::Expectations.fail_with(msg) unless yield
|
27
27
|
end
|
28
28
|
|
29
29
|
def assert_equal(a, b, msg=nil)
|
30
30
|
message = msg || "Expected equal"
|
31
|
-
|
31
|
+
make_assertion(message + "\n<#{a}> expected, but was\n<#{b}>") { a == b }
|
32
32
|
end
|
33
33
|
|
34
34
|
class AssertionFailedError < StandardError; end
|
@@ -9,7 +9,7 @@
|
|
9
9
|
# above copyright notice is included.
|
10
10
|
#+++
|
11
11
|
|
12
|
-
require 'test/unit'
|
12
|
+
require 'test/unit/assertions'
|
13
13
|
require 'flexmock/base'
|
14
14
|
require 'flexmock/test_unit_assert_spy_called'
|
15
15
|
|
@@ -47,10 +47,10 @@ class FlexMock
|
|
47
47
|
class TestUnitFrameworkAdapter
|
48
48
|
include Test::Unit::Assertions
|
49
49
|
|
50
|
-
def
|
50
|
+
def make_assertion(msg, &block)
|
51
51
|
unless yield
|
52
52
|
msg = msg.call if msg.is_a?(Proc)
|
53
|
-
|
53
|
+
assert(false, msg)
|
54
54
|
end
|
55
55
|
rescue assertion_failed_error => ex
|
56
56
|
ex.message.sub!(/Expected block to return true value./,'')
|
data/lib/flexmock/validators.rb
CHANGED
@@ -50,6 +50,22 @@ class FlexMock
|
|
50
50
|
".times(#{@limit})"
|
51
51
|
end
|
52
52
|
end
|
53
|
+
|
54
|
+
def describe_limit
|
55
|
+
@limit.to_s
|
56
|
+
end
|
57
|
+
|
58
|
+
def validate_count(n, &block)
|
59
|
+
@exp.flexmock_location_filter do
|
60
|
+
FlexMock.framework_adapter.make_assertion(
|
61
|
+
lambda {
|
62
|
+
"Method '#{@exp}' called incorrect number of times\n" +
|
63
|
+
"#{describe_limit} matching #{calls(@limit)} expected\n" +
|
64
|
+
"#{n} matching #{calls(n)} found\n" +
|
65
|
+
describe_calls(@exp.mock)
|
66
|
+
}, &block)
|
67
|
+
end
|
68
|
+
end
|
53
69
|
end
|
54
70
|
|
55
71
|
####################################################################
|
@@ -59,16 +75,7 @@ class FlexMock
|
|
59
75
|
# Validate that the method expectation was called exactly +n+
|
60
76
|
# times.
|
61
77
|
def validate(n)
|
62
|
-
@
|
63
|
-
FlexMock.framework_adapter.assert_block(
|
64
|
-
lambda {
|
65
|
-
"Method '#{@exp}' called incorrect number of times\n" +
|
66
|
-
"#{@limit} matching #{calls(@limit)} expected\n" +
|
67
|
-
"#{n} matching #{calls(n)} found\n" +
|
68
|
-
describe_calls(@exp.mock)
|
69
|
-
}
|
70
|
-
) { @limit == n }
|
71
|
-
end
|
78
|
+
validate_count(n) { @limit == n }
|
72
79
|
end
|
73
80
|
end
|
74
81
|
|
@@ -79,15 +86,7 @@ class FlexMock
|
|
79
86
|
# Validate the method expectation was called no more than +n+
|
80
87
|
# times.
|
81
88
|
def validate(n)
|
82
|
-
@
|
83
|
-
FlexMock.framework_adapter.assert_block(
|
84
|
-
lambda {
|
85
|
-
"Method '#{@exp}' called incorrect number of times\n" +
|
86
|
-
"At least #{@limit} matching #{calls(@limit)} expected\n" +
|
87
|
-
"#{n} matching #{calls(n)} found\n" +
|
88
|
-
describe_calls(@exp.mock)
|
89
|
-
}) { n >= @limit }
|
90
|
-
end
|
89
|
+
validate_count(n) { n >= @limit }
|
91
90
|
end
|
92
91
|
|
93
92
|
# Human readable description of the validator.
|
@@ -106,6 +105,10 @@ class FlexMock
|
|
106
105
|
def eligible?(n)
|
107
106
|
true
|
108
107
|
end
|
108
|
+
|
109
|
+
def describe_limit
|
110
|
+
"At least #{@limit}"
|
111
|
+
end
|
109
112
|
end
|
110
113
|
|
111
114
|
####################################################################
|
@@ -114,15 +117,7 @@ class FlexMock
|
|
114
117
|
class AtMostCountValidator < CountValidator
|
115
118
|
# Validate the method expectation was called at least +n+ times.
|
116
119
|
def validate(n)
|
117
|
-
@
|
118
|
-
FlexMock.framework_adapter.assert_block(
|
119
|
-
lambda {
|
120
|
-
"Method '#{@exp}' called incorrect number of times\n" +
|
121
|
-
"At most #{@limit} matching #{calls(@limit)} expected\n" +
|
122
|
-
"#{n} matching #{calls(n)} found\n" +
|
123
|
-
describe_calls(@exp.mock)
|
124
|
-
}) { n <= @limit }
|
125
|
-
end
|
120
|
+
validate_count(n) { n <= @limit }
|
126
121
|
end
|
127
122
|
|
128
123
|
# Human readable description of the validator
|
@@ -130,5 +125,8 @@ class FlexMock
|
|
130
125
|
".at_most#{super}"
|
131
126
|
end
|
132
127
|
|
128
|
+
def describe_limit
|
129
|
+
"At most #{@limit}"
|
130
|
+
end
|
133
131
|
end
|
134
132
|
end
|
data/lib/flexmock/version.rb
CHANGED
data/test/aliasing_test.rb
CHANGED
@@ -2,58 +2,65 @@
|
|
2
2
|
|
3
3
|
require 'test/test_setup'
|
4
4
|
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
result.at_least.once unless result.call_count_constrained?
|
10
|
-
result
|
11
|
-
end
|
12
|
-
def stubs(*args)
|
13
|
-
should_receive(*args)
|
14
|
-
end
|
15
|
-
end
|
5
|
+
# This is an experimental test to see if mock and stub methods can
|
6
|
+
# easily be added to a flexmock core. These test are disabled by
|
7
|
+
# default, and should be reviewed before something more production
|
8
|
+
# oriented is provided.
|
16
9
|
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
10
|
+
if ENV['TEST_ALIASES']
|
11
|
+
|
12
|
+
class FlexMock
|
13
|
+
module StubsAndExpects
|
14
|
+
def expects(*args)
|
15
|
+
result = should_receive(*args)
|
16
|
+
result.at_least.once unless result.call_count_constrained?
|
17
|
+
result
|
18
|
+
end
|
19
|
+
def stubs(*args)
|
20
|
+
should_receive(*args)
|
21
|
+
end
|
22
|
+
end
|
21
23
|
|
22
|
-
|
24
|
+
module MockContainer
|
25
|
+
alias :mock :flexmock
|
26
|
+
alias :stub :flexmock
|
27
|
+
end
|
23
28
|
|
24
|
-
class PartialMockProxy
|
25
29
|
include StubsAndExpects
|
26
|
-
|
30
|
+
|
31
|
+
class PartialMockProxy
|
32
|
+
include StubsAndExpects
|
33
|
+
MOCK_METHODS << :stubs << :expects
|
34
|
+
end
|
27
35
|
end
|
28
|
-
end
|
29
36
|
|
30
|
-
class AliasingTest < Test::Unit::TestCase
|
31
|
-
|
37
|
+
class AliasingTest < Test::Unit::TestCase
|
38
|
+
include FlexMock::TestCase
|
32
39
|
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
40
|
+
def test_mocking
|
41
|
+
m = mock("a cute dog").expects(:pat).twice.and_return(:woof!).mock
|
42
|
+
assert_equal :woof!, m.pat
|
43
|
+
assert_equal :woof!, m.pat
|
44
|
+
end
|
38
45
|
|
39
|
-
|
40
|
-
|
41
|
-
|
46
|
+
def test_once_mocking
|
47
|
+
mock("a cute dog").expects(:pat).and_return(:woof!).mock
|
48
|
+
end
|
42
49
|
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
50
|
+
def test_twice_mocking
|
51
|
+
m = mock("a cute dog").expects(:pat).and_return(:woof!).twice.mock
|
52
|
+
assert_raises(assertion_failed_error) { m.flexmock_verify }
|
53
|
+
end
|
47
54
|
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
55
|
+
def test_stubbing
|
56
|
+
m = stub("a cute dog").expects(:pat).and_return(:woof!).mock
|
57
|
+
assert_equal :woof!, m.pat
|
58
|
+
end
|
52
59
|
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
60
|
+
def test_partial
|
61
|
+
obj = Object.new
|
62
|
+
stub(obj).stubs(:wag).and_return(:tail)
|
63
|
+
assert_equal :tail, obj.wag
|
64
|
+
end
|
57
65
|
end
|
58
66
|
end
|
59
|
-
|