opal-rspec-cj 0.4.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +3 -0
- data/.gitmodules +15 -0
- data/.travis.yml +13 -0
- data/.yardopts +5 -0
- data/CHANGELOG.md +25 -0
- data/Gemfile +8 -0
- data/README.md +147 -0
- data/Rakefile +26 -0
- data/config.ru +10 -0
- data/example/Gemfile +4 -0
- data/example/README.md +13 -0
- data/example/Rakefile +8 -0
- data/example/opal/user.rb +11 -0
- data/example/spec/user_spec.rb +15 -0
- data/lib/opal-rspec.rb +2 -0
- data/lib/opal/rspec.rb +20 -0
- data/lib/opal/rspec/rake_task.rb +63 -0
- data/lib/opal/rspec/version.rb +5 -0
- data/opal-rspec.gemspec +21 -0
- data/opal/opal-rspec.rb +1 -0
- data/opal/opal/rspec.rb +25 -0
- data/opal/opal/rspec/async.rb +289 -0
- data/opal/opal/rspec/browser_formatter.rb +188 -0
- data/opal/opal/rspec/fixes.rb +116 -0
- data/opal/opal/rspec/requires.rb +45 -0
- data/opal/opal/rspec/runner.rb +69 -0
- data/opal/opal/rspec/sprockets_runner.rb.erb +11 -0
- data/opal/opal/rspec/text_formatter.rb +74 -0
- data/spec/async_spec.rb +38 -0
- data/spec/example_spec.rb +163 -0
- data/spec/matchers_spec.rb +201 -0
- data/spec/mock_spec.rb +63 -0
- data/spec/named_subject_spec.rb +11 -0
- data/spec/should_syntax_spec.rb +17 -0
- data/vendor/spec_runner.js +50 -0
- data/vendor_lib/rspec-expectations.rb +1 -0
- data/vendor_lib/rspec.rb +3 -0
- data/vendor_lib/rspec/autorun.rb +2 -0
- data/vendor_lib/rspec/core.rb +203 -0
- data/vendor_lib/rspec/core/backport_random.rb +302 -0
- data/vendor_lib/rspec/core/backtrace_formatter.rb +65 -0
- data/vendor_lib/rspec/core/command_line.rb +36 -0
- data/vendor_lib/rspec/core/configuration.rb +1129 -0
- data/vendor_lib/rspec/core/configuration_options.rb +143 -0
- data/vendor_lib/rspec/core/drb_command_line.rb +26 -0
- data/vendor_lib/rspec/core/drb_options.rb +87 -0
- data/vendor_lib/rspec/core/dsl.rb +26 -0
- data/vendor_lib/rspec/core/example.rb +312 -0
- data/vendor_lib/rspec/core/example_group.rb +540 -0
- data/vendor_lib/rspec/core/filter_manager.rb +224 -0
- data/vendor_lib/rspec/core/flat_map.rb +17 -0
- data/vendor_lib/rspec/core/formatters.rb +54 -0
- data/vendor_lib/rspec/core/formatters/base_formatter.rb +291 -0
- data/vendor_lib/rspec/core/formatters/base_text_formatter.rb +307 -0
- data/vendor_lib/rspec/core/formatters/deprecation_formatter.rb +193 -0
- data/vendor_lib/rspec/core/formatters/documentation_formatter.rb +67 -0
- data/vendor_lib/rspec/core/formatters/helpers.rb +82 -0
- data/vendor_lib/rspec/core/formatters/html_formatter.rb +155 -0
- data/vendor_lib/rspec/core/formatters/html_printer.rb +408 -0
- data/vendor_lib/rspec/core/formatters/json_formatter.rb +99 -0
- data/vendor_lib/rspec/core/formatters/progress_formatter.rb +32 -0
- data/vendor_lib/rspec/core/formatters/snippet_extractor.rb +101 -0
- data/vendor_lib/rspec/core/hooks.rb +535 -0
- data/vendor_lib/rspec/core/memoized_helpers.rb +431 -0
- data/vendor_lib/rspec/core/metadata.rb +313 -0
- data/vendor_lib/rspec/core/mocking/with_absolutely_nothing.rb +11 -0
- data/vendor_lib/rspec/core/mocking/with_flexmock.rb +27 -0
- data/vendor_lib/rspec/core/mocking/with_mocha.rb +52 -0
- data/vendor_lib/rspec/core/mocking/with_rr.rb +27 -0
- data/vendor_lib/rspec/core/mocking/with_rspec.rb +27 -0
- data/vendor_lib/rspec/core/option_parser.rb +234 -0
- data/vendor_lib/rspec/core/ordering.rb +154 -0
- data/vendor_lib/rspec/core/pending.rb +110 -0
- data/vendor_lib/rspec/core/project_initializer.rb +88 -0
- data/vendor_lib/rspec/core/rake_task.rb +128 -0
- data/vendor_lib/rspec/core/reporter.rb +132 -0
- data/vendor_lib/rspec/core/ruby_project.rb +44 -0
- data/vendor_lib/rspec/core/runner.rb +97 -0
- data/vendor_lib/rspec/core/shared_context.rb +53 -0
- data/vendor_lib/rspec/core/shared_example_group.rb +146 -0
- data/vendor_lib/rspec/core/shared_example_group/collection.rb +27 -0
- data/vendor_lib/rspec/core/version.rb +7 -0
- data/vendor_lib/rspec/core/warnings.rb +22 -0
- data/vendor_lib/rspec/core/world.rb +131 -0
- data/vendor_lib/rspec/expectations.rb +75 -0
- data/vendor_lib/rspec/expectations/differ.rb +154 -0
- data/vendor_lib/rspec/expectations/errors.rb +9 -0
- data/vendor_lib/rspec/expectations/expectation_target.rb +87 -0
- data/vendor_lib/rspec/expectations/extensions.rb +1 -0
- data/vendor_lib/rspec/expectations/extensions/object.rb +29 -0
- data/vendor_lib/rspec/expectations/fail_with.rb +79 -0
- data/vendor_lib/rspec/expectations/handler.rb +68 -0
- data/vendor_lib/rspec/expectations/syntax.rb +182 -0
- data/vendor_lib/rspec/expectations/version.rb +8 -0
- data/vendor_lib/rspec/matchers.rb +633 -0
- data/vendor_lib/rspec/matchers/built_in.rb +39 -0
- data/vendor_lib/rspec/matchers/built_in/base_matcher.rb +68 -0
- data/vendor_lib/rspec/matchers/built_in/be.rb +213 -0
- data/vendor_lib/rspec/matchers/built_in/be_instance_of.rb +15 -0
- data/vendor_lib/rspec/matchers/built_in/be_kind_of.rb +11 -0
- data/vendor_lib/rspec/matchers/built_in/be_within.rb +55 -0
- data/vendor_lib/rspec/matchers/built_in/change.rb +141 -0
- data/vendor_lib/rspec/matchers/built_in/cover.rb +21 -0
- data/vendor_lib/rspec/matchers/built_in/eq.rb +22 -0
- data/vendor_lib/rspec/matchers/built_in/eql.rb +23 -0
- data/vendor_lib/rspec/matchers/built_in/equal.rb +48 -0
- data/vendor_lib/rspec/matchers/built_in/exist.rb +26 -0
- data/vendor_lib/rspec/matchers/built_in/has.rb +48 -0
- data/vendor_lib/rspec/matchers/built_in/include.rb +61 -0
- data/vendor_lib/rspec/matchers/built_in/match.rb +17 -0
- data/vendor_lib/rspec/matchers/built_in/match_array.rb +51 -0
- data/vendor_lib/rspec/matchers/built_in/raise_error.rb +154 -0
- data/vendor_lib/rspec/matchers/built_in/respond_to.rb +74 -0
- data/vendor_lib/rspec/matchers/built_in/satisfy.rb +30 -0
- data/vendor_lib/rspec/matchers/built_in/start_and_end_with.rb +48 -0
- data/vendor_lib/rspec/matchers/built_in/throw_symbol.rb +94 -0
- data/vendor_lib/rspec/matchers/built_in/yield.rb +297 -0
- data/vendor_lib/rspec/matchers/compatibility.rb +14 -0
- data/vendor_lib/rspec/matchers/configuration.rb +113 -0
- data/vendor_lib/rspec/matchers/dsl.rb +23 -0
- data/vendor_lib/rspec/matchers/generated_descriptions.rb +35 -0
- data/vendor_lib/rspec/matchers/matcher.rb +301 -0
- data/vendor_lib/rspec/matchers/method_missing.rb +12 -0
- data/vendor_lib/rspec/matchers/operator_matcher.rb +99 -0
- data/vendor_lib/rspec/matchers/pretty.rb +70 -0
- data/vendor_lib/rspec/matchers/test_unit_integration.rb +11 -0
- data/vendor_lib/rspec/mocks.rb +100 -0
- data/vendor_lib/rspec/mocks/any_instance/chain.rb +92 -0
- data/vendor_lib/rspec/mocks/any_instance/expectation_chain.rb +47 -0
- data/vendor_lib/rspec/mocks/any_instance/message_chains.rb +75 -0
- data/vendor_lib/rspec/mocks/any_instance/recorder.rb +200 -0
- data/vendor_lib/rspec/mocks/any_instance/stub_chain.rb +45 -0
- data/vendor_lib/rspec/mocks/any_instance/stub_chain_chain.rb +23 -0
- data/vendor_lib/rspec/mocks/argument_list_matcher.rb +104 -0
- data/vendor_lib/rspec/mocks/argument_matchers.rb +264 -0
- data/vendor_lib/rspec/mocks/arity_calculator.rb +66 -0
- data/vendor_lib/rspec/mocks/configuration.rb +111 -0
- data/vendor_lib/rspec/mocks/error_generator.rb +203 -0
- data/vendor_lib/rspec/mocks/errors.rb +12 -0
- data/vendor_lib/rspec/mocks/example_methods.rb +201 -0
- data/vendor_lib/rspec/mocks/extensions/marshal.rb +17 -0
- data/vendor_lib/rspec/mocks/framework.rb +36 -0
- data/vendor_lib/rspec/mocks/instance_method_stasher.rb +112 -0
- data/vendor_lib/rspec/mocks/matchers/have_received.rb +99 -0
- data/vendor_lib/rspec/mocks/matchers/receive.rb +112 -0
- data/vendor_lib/rspec/mocks/matchers/receive_messages.rb +72 -0
- data/vendor_lib/rspec/mocks/message_expectation.rb +643 -0
- data/vendor_lib/rspec/mocks/method_double.rb +209 -0
- data/vendor_lib/rspec/mocks/method_reference.rb +95 -0
- data/vendor_lib/rspec/mocks/mock.rb +7 -0
- data/vendor_lib/rspec/mocks/mutate_const.rb +406 -0
- data/vendor_lib/rspec/mocks/object_reference.rb +90 -0
- data/vendor_lib/rspec/mocks/order_group.rb +82 -0
- data/vendor_lib/rspec/mocks/proxy.rb +269 -0
- data/vendor_lib/rspec/mocks/proxy_for_nil.rb +37 -0
- data/vendor_lib/rspec/mocks/space.rb +95 -0
- data/vendor_lib/rspec/mocks/standalone.rb +3 -0
- data/vendor_lib/rspec/mocks/stub_chain.rb +51 -0
- data/vendor_lib/rspec/mocks/syntax.rb +374 -0
- data/vendor_lib/rspec/mocks/targets.rb +90 -0
- data/vendor_lib/rspec/mocks/test_double.rb +109 -0
- data/vendor_lib/rspec/mocks/verifying_double.rb +77 -0
- data/vendor_lib/rspec/mocks/verifying_message_expecation.rb +60 -0
- data/vendor_lib/rspec/mocks/verifying_proxy.rb +151 -0
- data/vendor_lib/rspec/mocks/version.rb +7 -0
- data/vendor_lib/rspec/support.rb +6 -0
- data/vendor_lib/rspec/support/caller_filter.rb +56 -0
- data/vendor_lib/rspec/support/spec.rb +14 -0
- data/vendor_lib/rspec/support/spec/deprecation_helpers.rb +29 -0
- data/vendor_lib/rspec/support/spec/in_sub_process.rb +40 -0
- data/vendor_lib/rspec/support/spec/stderr_splitter.rb +50 -0
- data/vendor_lib/rspec/support/version.rb +7 -0
- data/vendor_lib/rspec/support/warnings.rb +41 -0
- data/vendor_lib/rspec/version.rb +5 -0
- metadata +268 -0
|
@@ -0,0 +1,431 @@
|
|
|
1
|
+
module RSpec
|
|
2
|
+
module Core
|
|
3
|
+
module MemoizedHelpers
|
|
4
|
+
# @note `subject` was contributed by Joe Ferris to support the one-liner
|
|
5
|
+
# syntax embraced by shoulda matchers:
|
|
6
|
+
#
|
|
7
|
+
# describe Widget do
|
|
8
|
+
# it { should validate_presence_of(:name) }
|
|
9
|
+
# end
|
|
10
|
+
#
|
|
11
|
+
# While the examples below demonstrate how to use `subject`
|
|
12
|
+
# explicitly in examples, we recommend that you define a method with
|
|
13
|
+
# an intention revealing name instead.
|
|
14
|
+
#
|
|
15
|
+
# @example
|
|
16
|
+
#
|
|
17
|
+
# # explicit declaration of subject
|
|
18
|
+
# describe Person do
|
|
19
|
+
# subject { Person.new(:birthdate => 19.years.ago) }
|
|
20
|
+
# it "should be eligible to vote" do
|
|
21
|
+
# subject.should be_eligible_to_vote
|
|
22
|
+
# # ^ ^ explicit reference to subject not recommended
|
|
23
|
+
# end
|
|
24
|
+
# end
|
|
25
|
+
#
|
|
26
|
+
# # implicit subject => { Person.new }
|
|
27
|
+
# describe Person do
|
|
28
|
+
# it "should be eligible to vote" do
|
|
29
|
+
# subject.should be_eligible_to_vote
|
|
30
|
+
# # ^ ^ explicit reference to subject not recommended
|
|
31
|
+
# end
|
|
32
|
+
# end
|
|
33
|
+
#
|
|
34
|
+
# # one-liner syntax - should is invoked on subject
|
|
35
|
+
# describe Person do
|
|
36
|
+
# it { should be_eligible_to_vote }
|
|
37
|
+
# end
|
|
38
|
+
#
|
|
39
|
+
# @note Because `subject` is designed to create state that is reset between
|
|
40
|
+
# each example, and `before(:all)` is designed to setup state that is
|
|
41
|
+
# shared across _all_ examples in an example group, `subject` is _not_
|
|
42
|
+
# intended to be used in a `before(:all)` hook. RSpec 2.13.1 prints
|
|
43
|
+
# a warning when you reference a `subject` from `before(:all)` and we plan
|
|
44
|
+
# to have it raise an error in RSpec 3.
|
|
45
|
+
#
|
|
46
|
+
# @see #should
|
|
47
|
+
def subject
|
|
48
|
+
__memoized.fetch(:subject) do
|
|
49
|
+
__memoized[:subject] = begin
|
|
50
|
+
described = described_class || self.class.description
|
|
51
|
+
Class === described ? described.new : described
|
|
52
|
+
end
|
|
53
|
+
end
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
# When `should` is called with no explicit receiver, the call is
|
|
57
|
+
# delegated to the object returned by `subject`. Combined with an
|
|
58
|
+
# implicit subject this supports very concise expressions.
|
|
59
|
+
#
|
|
60
|
+
# @example
|
|
61
|
+
#
|
|
62
|
+
# describe Person do
|
|
63
|
+
# it { should be_eligible_to_vote }
|
|
64
|
+
# end
|
|
65
|
+
#
|
|
66
|
+
# @see #subject
|
|
67
|
+
def should(matcher=nil, message=nil)
|
|
68
|
+
RSpec::Expectations::PositiveExpectationHandler.handle_matcher(subject, matcher, message)
|
|
69
|
+
end
|
|
70
|
+
|
|
71
|
+
# Just like `should`, `should_not` delegates to the subject (implicit or
|
|
72
|
+
# explicit) of the example group.
|
|
73
|
+
#
|
|
74
|
+
# @example
|
|
75
|
+
#
|
|
76
|
+
# describe Person do
|
|
77
|
+
# it { should_not be_eligible_to_vote }
|
|
78
|
+
# end
|
|
79
|
+
#
|
|
80
|
+
# @see #subject
|
|
81
|
+
def should_not(matcher=nil, message=nil)
|
|
82
|
+
RSpec::Expectations::NegativeExpectationHandler.handle_matcher(subject, matcher, message)
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
private
|
|
86
|
+
|
|
87
|
+
# @private
|
|
88
|
+
def __memoized
|
|
89
|
+
@__memoized ||= {}
|
|
90
|
+
end
|
|
91
|
+
|
|
92
|
+
# Used internally to customize the behavior of the
|
|
93
|
+
# memoized hash when used in a `before(:all)` hook.
|
|
94
|
+
#
|
|
95
|
+
# @private
|
|
96
|
+
class AllHookMemoizedHash
|
|
97
|
+
def self.isolate_for_all_hook(example_group_instance)
|
|
98
|
+
hash = self
|
|
99
|
+
|
|
100
|
+
example_group_instance.instance_eval do
|
|
101
|
+
@__memoized = hash
|
|
102
|
+
|
|
103
|
+
begin
|
|
104
|
+
yield
|
|
105
|
+
ensure
|
|
106
|
+
@__memoized = nil
|
|
107
|
+
end
|
|
108
|
+
end
|
|
109
|
+
end
|
|
110
|
+
|
|
111
|
+
def self.fetch(key, &block)
|
|
112
|
+
description = if key == :subject
|
|
113
|
+
"subject"
|
|
114
|
+
else
|
|
115
|
+
"let declaration `#{key}`"
|
|
116
|
+
end
|
|
117
|
+
|
|
118
|
+
raise <<-EOS
|
|
119
|
+
#{description} accessed in #{article} #{hook_expression} hook at:
|
|
120
|
+
#{CallerFilter.first_non_rspec_line}
|
|
121
|
+
|
|
122
|
+
`let` and `subject` declarations are not intended to be called
|
|
123
|
+
in #{article} #{hook_expression} hook, as they exist to define state that
|
|
124
|
+
is reset between each example, while #{hook_expression} exists to
|
|
125
|
+
#{hook_intention}.
|
|
126
|
+
EOS
|
|
127
|
+
end
|
|
128
|
+
|
|
129
|
+
class Before < self
|
|
130
|
+
def self.hook_expression
|
|
131
|
+
"`before(:all)`"
|
|
132
|
+
end
|
|
133
|
+
|
|
134
|
+
def self.article
|
|
135
|
+
"a"
|
|
136
|
+
end
|
|
137
|
+
|
|
138
|
+
def self.hook_intention
|
|
139
|
+
"define state that is shared across examples in an example group"
|
|
140
|
+
end
|
|
141
|
+
end
|
|
142
|
+
|
|
143
|
+
class After < self
|
|
144
|
+
def self.hook_expression
|
|
145
|
+
"`after(:all)`"
|
|
146
|
+
end
|
|
147
|
+
|
|
148
|
+
def self.article
|
|
149
|
+
"an"
|
|
150
|
+
end
|
|
151
|
+
|
|
152
|
+
def self.hook_intention
|
|
153
|
+
"cleanup state that is shared across examples in an example group"
|
|
154
|
+
end
|
|
155
|
+
end
|
|
156
|
+
end
|
|
157
|
+
|
|
158
|
+
def self.included(mod)
|
|
159
|
+
mod.extend(ClassMethods)
|
|
160
|
+
end
|
|
161
|
+
|
|
162
|
+
module ClassMethods
|
|
163
|
+
# Generates a method whose return value is memoized after the first
|
|
164
|
+
# call. Useful for reducing duplication between examples that assign
|
|
165
|
+
# values to the same local variable.
|
|
166
|
+
#
|
|
167
|
+
# @note `let` _can_ enhance readability when used sparingly (1,2, or
|
|
168
|
+
# maybe 3 declarations) in any given example group, but that can
|
|
169
|
+
# quickly degrade with overuse. YMMV.
|
|
170
|
+
#
|
|
171
|
+
# @note `let` uses an `||=` conditional that has the potential to
|
|
172
|
+
# behave in surprising ways in examples that spawn separate threads,
|
|
173
|
+
# though we have yet to see this in practice. You've been warned.
|
|
174
|
+
#
|
|
175
|
+
# @note Because `let` is designed to create state that is reset between
|
|
176
|
+
# each example, and `before(:all)` is designed to setup state that is
|
|
177
|
+
# shared across _all_ examples in an example group, `let` is _not_
|
|
178
|
+
# intended to be used in a `before(:all)` hook. RSpec 2.13.1 prints
|
|
179
|
+
# a warning when you reference a `let` from `before(:all)` and we plan
|
|
180
|
+
# to have it raise an error in RSpec 3.
|
|
181
|
+
#
|
|
182
|
+
# @example
|
|
183
|
+
#
|
|
184
|
+
# describe Thing do
|
|
185
|
+
# let(:thing) { Thing.new }
|
|
186
|
+
#
|
|
187
|
+
# it "does something" do
|
|
188
|
+
# # first invocation, executes block, memoizes and returns result
|
|
189
|
+
# thing.do_something
|
|
190
|
+
#
|
|
191
|
+
# # second invocation, returns the memoized value
|
|
192
|
+
# thing.should be_something
|
|
193
|
+
# end
|
|
194
|
+
# end
|
|
195
|
+
def let(name, &block)
|
|
196
|
+
# We have to pass the block directly to `define_method` to
|
|
197
|
+
# allow it to use method constructs like `super` and `return`.
|
|
198
|
+
raise "#let or #subject called without a block" if block.nil?
|
|
199
|
+
MemoizedHelpers.module_for(self).send(:define_method, name, &block)
|
|
200
|
+
|
|
201
|
+
# Apply the memoization. The method has been defined in an ancestor
|
|
202
|
+
# module so we can use `super` here to get the value.
|
|
203
|
+
if block.arity == 1
|
|
204
|
+
define_method(name) { __memoized.fetch(name) { |k| __memoized[k] = super(RSpec.current_example, &nil) } }
|
|
205
|
+
else
|
|
206
|
+
define_method(name) { __memoized.fetch(name) { |k| __memoized[k] = super(&nil) } }
|
|
207
|
+
end
|
|
208
|
+
end
|
|
209
|
+
|
|
210
|
+
# Just like `let`, except the block is invoked by an implicit `before`
|
|
211
|
+
# hook. This serves a dual purpose of setting up state and providing a
|
|
212
|
+
# memoized reference to that state.
|
|
213
|
+
#
|
|
214
|
+
# @example
|
|
215
|
+
#
|
|
216
|
+
# class Thing
|
|
217
|
+
# def self.count
|
|
218
|
+
# @count ||= 0
|
|
219
|
+
# end
|
|
220
|
+
#
|
|
221
|
+
# def self.count=(val)
|
|
222
|
+
# @count += val
|
|
223
|
+
# end
|
|
224
|
+
#
|
|
225
|
+
# def self.reset_count
|
|
226
|
+
# @count = 0
|
|
227
|
+
# end
|
|
228
|
+
#
|
|
229
|
+
# def initialize
|
|
230
|
+
# self.class.count += 1
|
|
231
|
+
# end
|
|
232
|
+
# end
|
|
233
|
+
#
|
|
234
|
+
# describe Thing do
|
|
235
|
+
# after(:each) { Thing.reset_count }
|
|
236
|
+
#
|
|
237
|
+
# context "using let" do
|
|
238
|
+
# let(:thing) { Thing.new }
|
|
239
|
+
#
|
|
240
|
+
# it "is not invoked implicitly" do
|
|
241
|
+
# Thing.count.should eq(0)
|
|
242
|
+
# end
|
|
243
|
+
#
|
|
244
|
+
# it "can be invoked explicitly" do
|
|
245
|
+
# thing
|
|
246
|
+
# Thing.count.should eq(1)
|
|
247
|
+
# end
|
|
248
|
+
# end
|
|
249
|
+
#
|
|
250
|
+
# context "using let!" do
|
|
251
|
+
# let!(:thing) { Thing.new }
|
|
252
|
+
#
|
|
253
|
+
# it "is invoked implicitly" do
|
|
254
|
+
# Thing.count.should eq(1)
|
|
255
|
+
# end
|
|
256
|
+
#
|
|
257
|
+
# it "returns memoized version on first invocation" do
|
|
258
|
+
# thing
|
|
259
|
+
# Thing.count.should eq(1)
|
|
260
|
+
# end
|
|
261
|
+
# end
|
|
262
|
+
# end
|
|
263
|
+
def let!(name, &block)
|
|
264
|
+
let(name, &block)
|
|
265
|
+
before { __send__(name) }
|
|
266
|
+
end
|
|
267
|
+
|
|
268
|
+
# Declares a `subject` for an example group which can then be the
|
|
269
|
+
# implicit receiver (through delegation) of calls to `should`.
|
|
270
|
+
#
|
|
271
|
+
# Given a `name`, defines a method with that name which returns the
|
|
272
|
+
# `subject`. This lets you declare the subject once and access it
|
|
273
|
+
# implicitly in one-liners and explicitly using an intention revealing
|
|
274
|
+
# name.
|
|
275
|
+
#
|
|
276
|
+
# @param [String,Symbol] name used to define an accessor with an
|
|
277
|
+
# intention revealing name
|
|
278
|
+
# @param block defines the value to be returned by `subject` in examples
|
|
279
|
+
#
|
|
280
|
+
# @example
|
|
281
|
+
#
|
|
282
|
+
# describe CheckingAccount, "with $50" do
|
|
283
|
+
# subject { CheckingAccount.new(Money.new(50, :USD)) }
|
|
284
|
+
# it { should have_a_balance_of(Money.new(50, :USD)) }
|
|
285
|
+
# it { should_not be_overdrawn }
|
|
286
|
+
# end
|
|
287
|
+
#
|
|
288
|
+
# describe CheckingAccount, "with a non-zero starting balance" do
|
|
289
|
+
# subject(:account) { CheckingAccount.new(Money.new(50, :USD)) }
|
|
290
|
+
# it { should_not be_overdrawn }
|
|
291
|
+
# it "has a balance equal to the starting balance" do
|
|
292
|
+
# account.balance.should eq(Money.new(50, :USD))
|
|
293
|
+
# end
|
|
294
|
+
# end
|
|
295
|
+
#
|
|
296
|
+
# @see MemoizedHelpers#should
|
|
297
|
+
def subject(name=nil, &block)
|
|
298
|
+
if name
|
|
299
|
+
let(name, &block)
|
|
300
|
+
alias_method :subject, name
|
|
301
|
+
|
|
302
|
+
self::NamedSubjectPreventSuper.send(:define_method, name) do
|
|
303
|
+
raise NotImplementedError, "`super` in named subjects is not supported"
|
|
304
|
+
end
|
|
305
|
+
else
|
|
306
|
+
let(:subject, &block)
|
|
307
|
+
end
|
|
308
|
+
end
|
|
309
|
+
|
|
310
|
+
# Just like `subject`, except the block is invoked by an implicit `before`
|
|
311
|
+
# hook. This serves a dual purpose of setting up state and providing a
|
|
312
|
+
# memoized reference to that state.
|
|
313
|
+
#
|
|
314
|
+
# @example
|
|
315
|
+
#
|
|
316
|
+
# class Thing
|
|
317
|
+
# def self.count
|
|
318
|
+
# @count ||= 0
|
|
319
|
+
# end
|
|
320
|
+
#
|
|
321
|
+
# def self.count=(val)
|
|
322
|
+
# @count += val
|
|
323
|
+
# end
|
|
324
|
+
#
|
|
325
|
+
# def self.reset_count
|
|
326
|
+
# @count = 0
|
|
327
|
+
# end
|
|
328
|
+
#
|
|
329
|
+
# def initialize
|
|
330
|
+
# self.class.count += 1
|
|
331
|
+
# end
|
|
332
|
+
# end
|
|
333
|
+
#
|
|
334
|
+
# describe Thing do
|
|
335
|
+
# after(:each) { Thing.reset_count }
|
|
336
|
+
#
|
|
337
|
+
# context "using subject" do
|
|
338
|
+
# subject { Thing.new }
|
|
339
|
+
#
|
|
340
|
+
# it "is not invoked implicitly" do
|
|
341
|
+
# Thing.count.should eq(0)
|
|
342
|
+
# end
|
|
343
|
+
#
|
|
344
|
+
# it "can be invoked explicitly" do
|
|
345
|
+
# subject
|
|
346
|
+
# Thing.count.should eq(1)
|
|
347
|
+
# end
|
|
348
|
+
# end
|
|
349
|
+
#
|
|
350
|
+
# context "using subject!" do
|
|
351
|
+
# subject!(:thing) { Thing.new }
|
|
352
|
+
#
|
|
353
|
+
# it "is invoked implicitly" do
|
|
354
|
+
# Thing.count.should eq(1)
|
|
355
|
+
# end
|
|
356
|
+
#
|
|
357
|
+
# it "returns memoized version on first invocation" do
|
|
358
|
+
# subject
|
|
359
|
+
# Thing.count.should eq(1)
|
|
360
|
+
# end
|
|
361
|
+
# end
|
|
362
|
+
# end
|
|
363
|
+
def subject!(name=nil, &block)
|
|
364
|
+
subject(name, &block)
|
|
365
|
+
before { subject }
|
|
366
|
+
end
|
|
367
|
+
end
|
|
368
|
+
|
|
369
|
+
# @api private
|
|
370
|
+
#
|
|
371
|
+
# Gets the LetDefinitions module. The module is mixed into
|
|
372
|
+
# the example group and is used to hold all let definitions.
|
|
373
|
+
# This is done so that the block passed to `let` can be
|
|
374
|
+
# forwarded directly on to `define_method`, so that all method
|
|
375
|
+
# constructs (including `super` and `return`) can be used in
|
|
376
|
+
# a `let` block.
|
|
377
|
+
#
|
|
378
|
+
# The memoization is provided by a method definition on the
|
|
379
|
+
# example group that supers to the LetDefinitions definition
|
|
380
|
+
# in order to get the value to memoize.
|
|
381
|
+
def self.module_for(example_group)
|
|
382
|
+
get_constant_or_yield(example_group, :LetDefinitions) do
|
|
383
|
+
mod = Module.new do
|
|
384
|
+
include Module.new {
|
|
385
|
+
example_group.const_set(:NamedSubjectPreventSuper, self)
|
|
386
|
+
}
|
|
387
|
+
end
|
|
388
|
+
|
|
389
|
+
example_group.const_set(:LetDefinitions, mod)
|
|
390
|
+
mod
|
|
391
|
+
end
|
|
392
|
+
end
|
|
393
|
+
|
|
394
|
+
# @api private
|
|
395
|
+
def self.define_helpers_on(example_group)
|
|
396
|
+
example_group.send(:include, module_for(example_group))
|
|
397
|
+
end
|
|
398
|
+
|
|
399
|
+
if Module.method(:const_defined?).arity == 1 # for 1.8
|
|
400
|
+
# @api private
|
|
401
|
+
#
|
|
402
|
+
# Gets the named constant or yields.
|
|
403
|
+
# On 1.8, const_defined? / const_get do not take into
|
|
404
|
+
# account the inheritance hierarchy.
|
|
405
|
+
def self.get_constant_or_yield(example_group, name)
|
|
406
|
+
if example_group.const_defined?(name)
|
|
407
|
+
example_group.const_get(name)
|
|
408
|
+
else
|
|
409
|
+
yield
|
|
410
|
+
end
|
|
411
|
+
end
|
|
412
|
+
else
|
|
413
|
+
# @api private
|
|
414
|
+
#
|
|
415
|
+
# Gets the named constant or yields.
|
|
416
|
+
# On 1.9, const_defined? / const_get take into account the
|
|
417
|
+
# the inheritance by default, and accept an argument to
|
|
418
|
+
# disable this behavior. It's important that we don't
|
|
419
|
+
# consider inheritance here; each example group level that
|
|
420
|
+
# uses a `let` should get its own `LetDefinitions` module.
|
|
421
|
+
def self.get_constant_or_yield(example_group, name)
|
|
422
|
+
if example_group.const_defined?(name, (check_ancestors = false))
|
|
423
|
+
example_group.const_get(name, check_ancestors)
|
|
424
|
+
else
|
|
425
|
+
yield
|
|
426
|
+
end
|
|
427
|
+
end
|
|
428
|
+
end
|
|
429
|
+
end
|
|
430
|
+
end
|
|
431
|
+
end
|
|
@@ -0,0 +1,313 @@
|
|
|
1
|
+
module RSpec
|
|
2
|
+
module Core
|
|
3
|
+
# Each ExampleGroup class and Example instance owns an instance of
|
|
4
|
+
# Metadata, which is Hash extended to support lazy evaluation of values
|
|
5
|
+
# associated with keys that may or may not be used by any example or group.
|
|
6
|
+
#
|
|
7
|
+
# In addition to metadata that is used internally, this also stores
|
|
8
|
+
# user-supplied metadata, e.g.
|
|
9
|
+
#
|
|
10
|
+
# describe Something, :type => :ui do
|
|
11
|
+
# it "does something", :slow => true do
|
|
12
|
+
# # ...
|
|
13
|
+
# end
|
|
14
|
+
# end
|
|
15
|
+
#
|
|
16
|
+
# `:type => :ui` is stored in the Metadata owned by the example group, and
|
|
17
|
+
# `:slow => true` is stored in the Metadata owned by the example. These can
|
|
18
|
+
# then be used to select which examples are run using the `--tag` option on
|
|
19
|
+
# the command line, or several methods on `Configuration` used to filter a
|
|
20
|
+
# run (e.g. `filter_run_including`, `filter_run_excluding`, etc).
|
|
21
|
+
#
|
|
22
|
+
# @see Example#metadata
|
|
23
|
+
# @see ExampleGroup.metadata
|
|
24
|
+
# @see FilterManager
|
|
25
|
+
# @see Configuration#filter_run_including
|
|
26
|
+
# @see Configuration#filter_run_excluding
|
|
27
|
+
class Metadata < Hash
|
|
28
|
+
|
|
29
|
+
def self.relative_path(line)
|
|
30
|
+
line = line.sub(File.expand_path("."), ".")
|
|
31
|
+
line = line.sub(/\A([^:]+:\d+)$/, '\\1')
|
|
32
|
+
return nil if line == '-e:1'
|
|
33
|
+
line
|
|
34
|
+
rescue SecurityError
|
|
35
|
+
nil
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
# @private
|
|
39
|
+
# Used internally to build a hash from an args array.
|
|
40
|
+
# Symbols are converted into hash keys with a value of `true`.
|
|
41
|
+
# This is done to support simple tagging using a symbol, rather
|
|
42
|
+
# than needing to do `:symbol => true`.
|
|
43
|
+
def self.build_hash_from(args)
|
|
44
|
+
hash = args.last.is_a?(Hash) ? args.pop : {}
|
|
45
|
+
|
|
46
|
+
while args.last.is_a?(Symbol)
|
|
47
|
+
hash[args.pop] = true
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
hash
|
|
51
|
+
end
|
|
52
|
+
|
|
53
|
+
module MetadataHash
|
|
54
|
+
|
|
55
|
+
# @private
|
|
56
|
+
# Supports lazy evaluation of some values. Extended by
|
|
57
|
+
# ExampleMetadataHash and GroupMetadataHash, which get mixed in to
|
|
58
|
+
# Metadata for ExampleGroups and Examples (respectively).
|
|
59
|
+
def [](key)
|
|
60
|
+
store_computed(key) unless has_key?(key)
|
|
61
|
+
super
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
def fetch(key, *args)
|
|
65
|
+
store_computed(key) unless has_key?(key)
|
|
66
|
+
super
|
|
67
|
+
end
|
|
68
|
+
|
|
69
|
+
private
|
|
70
|
+
|
|
71
|
+
def store_computed(key)
|
|
72
|
+
case key
|
|
73
|
+
when :location
|
|
74
|
+
store(:location, location)
|
|
75
|
+
when :file_path, :line_number
|
|
76
|
+
file_path, line_number = file_and_line_number
|
|
77
|
+
store(:file_path, file_path)
|
|
78
|
+
store(:line_number, line_number)
|
|
79
|
+
when :execution_result
|
|
80
|
+
store(:execution_result, {})
|
|
81
|
+
when :describes, :described_class
|
|
82
|
+
klass = described_class
|
|
83
|
+
store(:described_class, klass)
|
|
84
|
+
# TODO (2011-11-07 DC) deprecate :describes as a key
|
|
85
|
+
store(:describes, klass)
|
|
86
|
+
when :full_description
|
|
87
|
+
store(:full_description, full_description)
|
|
88
|
+
when :description
|
|
89
|
+
store(:description, build_description_from(*self[:description_args]))
|
|
90
|
+
when :description_args
|
|
91
|
+
store(:description_args, [])
|
|
92
|
+
end
|
|
93
|
+
end
|
|
94
|
+
|
|
95
|
+
def location
|
|
96
|
+
"#{self[:file_path]}:#{self[:line_number]}"
|
|
97
|
+
end
|
|
98
|
+
|
|
99
|
+
def file_and_line_number
|
|
100
|
+
first_caller_from_outside_rspec =~ /(.+?):(\d+)(|:\d+)/
|
|
101
|
+
return [Metadata::relative_path($1), $2.to_i]
|
|
102
|
+
end
|
|
103
|
+
|
|
104
|
+
def first_caller_from_outside_rspec
|
|
105
|
+
self[:caller].detect {|l| l !~ /\/lib\/rspec\/core/}
|
|
106
|
+
end
|
|
107
|
+
|
|
108
|
+
def method_description_after_module?(parent_part, child_part)
|
|
109
|
+
return false unless parent_part.is_a?(Module)
|
|
110
|
+
child_part =~ /^(#|::|\.)/
|
|
111
|
+
end
|
|
112
|
+
|
|
113
|
+
def build_description_from(first_part = '', *parts)
|
|
114
|
+
description, _ = parts.inject([first_part.to_s, first_part]) do |(desc, last_part), this_part|
|
|
115
|
+
this_part = this_part.to_s
|
|
116
|
+
this_part = (' ' + this_part) unless method_description_after_module?(last_part, this_part)
|
|
117
|
+
[(desc + this_part), this_part]
|
|
118
|
+
end
|
|
119
|
+
|
|
120
|
+
description
|
|
121
|
+
end
|
|
122
|
+
end
|
|
123
|
+
|
|
124
|
+
# Mixed in to Metadata for an Example (extends MetadataHash) to support
|
|
125
|
+
# lazy evaluation of some values.
|
|
126
|
+
module ExampleMetadataHash
|
|
127
|
+
include MetadataHash
|
|
128
|
+
|
|
129
|
+
def described_class
|
|
130
|
+
self[:example_group].described_class
|
|
131
|
+
end
|
|
132
|
+
|
|
133
|
+
def full_description
|
|
134
|
+
build_description_from(self[:example_group][:full_description], *self[:description_args])
|
|
135
|
+
end
|
|
136
|
+
end
|
|
137
|
+
|
|
138
|
+
# Mixed in to Metadata for an ExampleGroup (extends MetadataHash) to
|
|
139
|
+
# support lazy evaluation of some values.
|
|
140
|
+
module GroupMetadataHash
|
|
141
|
+
include MetadataHash
|
|
142
|
+
|
|
143
|
+
def described_class
|
|
144
|
+
container_stack.each do |g|
|
|
145
|
+
[:described_class, :describes].each do |key|
|
|
146
|
+
if g.has_key?(key)
|
|
147
|
+
value = g[key]
|
|
148
|
+
return value unless value.nil?
|
|
149
|
+
end
|
|
150
|
+
end
|
|
151
|
+
end
|
|
152
|
+
|
|
153
|
+
container_stack.reverse.each do |g|
|
|
154
|
+
candidate = g[:description_args].first
|
|
155
|
+
return candidate unless String === candidate || Symbol === candidate
|
|
156
|
+
end
|
|
157
|
+
|
|
158
|
+
nil
|
|
159
|
+
end
|
|
160
|
+
|
|
161
|
+
def full_description
|
|
162
|
+
build_description_from(*FlatMap.flat_map(container_stack.reverse) {|a| a[:description_args]})
|
|
163
|
+
end
|
|
164
|
+
|
|
165
|
+
def container_stack
|
|
166
|
+
@container_stack ||= begin
|
|
167
|
+
groups = [group = self]
|
|
168
|
+
while group.has_key?(:example_group)
|
|
169
|
+
groups << group[:example_group]
|
|
170
|
+
group = group[:example_group]
|
|
171
|
+
end
|
|
172
|
+
groups
|
|
173
|
+
end
|
|
174
|
+
end
|
|
175
|
+
end
|
|
176
|
+
|
|
177
|
+
def initialize(parent_group_metadata=nil)
|
|
178
|
+
if parent_group_metadata
|
|
179
|
+
update(parent_group_metadata)
|
|
180
|
+
store(:example_group, {:example_group => parent_group_metadata[:example_group].extend(GroupMetadataHash)}.extend(GroupMetadataHash))
|
|
181
|
+
else
|
|
182
|
+
store(:example_group, {}.extend(GroupMetadataHash))
|
|
183
|
+
end
|
|
184
|
+
|
|
185
|
+
yield self if block_given?
|
|
186
|
+
end
|
|
187
|
+
|
|
188
|
+
# @private
|
|
189
|
+
def process(*args)
|
|
190
|
+
user_metadata = args.last.is_a?(Hash) ? args.pop : {}
|
|
191
|
+
ensure_valid_keys(user_metadata)
|
|
192
|
+
|
|
193
|
+
self[:example_group].store(:description_args, args)
|
|
194
|
+
self[:example_group].store(:caller, user_metadata.delete(:caller) || caller)
|
|
195
|
+
|
|
196
|
+
update(user_metadata)
|
|
197
|
+
end
|
|
198
|
+
|
|
199
|
+
# @private
|
|
200
|
+
def for_example(description, user_metadata)
|
|
201
|
+
dup.extend(ExampleMetadataHash).configure_for_example(description, user_metadata)
|
|
202
|
+
end
|
|
203
|
+
|
|
204
|
+
# @private
|
|
205
|
+
def any_apply?(filters)
|
|
206
|
+
filters.any? {|k,v| filter_applies?(k,v)}
|
|
207
|
+
end
|
|
208
|
+
|
|
209
|
+
# @private
|
|
210
|
+
def all_apply?(filters)
|
|
211
|
+
filters.all? {|k,v| filter_applies?(k,v)}
|
|
212
|
+
end
|
|
213
|
+
|
|
214
|
+
# @private
|
|
215
|
+
def filter_applies?(key, value, metadata=self)
|
|
216
|
+
return metadata.filter_applies_to_any_value?(key, value) if Array === metadata[key] && !(Proc === value)
|
|
217
|
+
return metadata.line_number_filter_applies?(value) if key == :line_numbers
|
|
218
|
+
return metadata.location_filter_applies?(value) if key == :locations
|
|
219
|
+
return metadata.filters_apply?(key, value) if Hash === value
|
|
220
|
+
|
|
221
|
+
return false unless metadata.has_key?(key)
|
|
222
|
+
|
|
223
|
+
case value
|
|
224
|
+
when Regexp
|
|
225
|
+
metadata[key] =~ value
|
|
226
|
+
when Proc
|
|
227
|
+
case value.arity
|
|
228
|
+
when 0 then value.call
|
|
229
|
+
when 2 then value.call(metadata[key], metadata)
|
|
230
|
+
else value.call(metadata[key])
|
|
231
|
+
end
|
|
232
|
+
else
|
|
233
|
+
metadata[key].to_s == value.to_s
|
|
234
|
+
end
|
|
235
|
+
end
|
|
236
|
+
|
|
237
|
+
# @private
|
|
238
|
+
def filters_apply?(key, value)
|
|
239
|
+
value.all? {|k, v| filter_applies?(k, v, self[key])}
|
|
240
|
+
end
|
|
241
|
+
|
|
242
|
+
# @private
|
|
243
|
+
def filter_applies_to_any_value?(key, value)
|
|
244
|
+
self[key].any? {|v| filter_applies?(key, v, {key => value})}
|
|
245
|
+
end
|
|
246
|
+
|
|
247
|
+
# @private
|
|
248
|
+
def location_filter_applies?(locations)
|
|
249
|
+
# it ignores location filters for other files
|
|
250
|
+
line_number = example_group_declaration_line(locations)
|
|
251
|
+
line_number ? line_number_filter_applies?(line_number) : true
|
|
252
|
+
end
|
|
253
|
+
|
|
254
|
+
# @private
|
|
255
|
+
def line_number_filter_applies?(line_numbers)
|
|
256
|
+
preceding_declaration_lines = line_numbers.map {|n| RSpec.world.preceding_declaration_line(n)}
|
|
257
|
+
!(relevant_line_numbers & preceding_declaration_lines).empty?
|
|
258
|
+
end
|
|
259
|
+
|
|
260
|
+
protected
|
|
261
|
+
|
|
262
|
+
def configure_for_example(description, user_metadata)
|
|
263
|
+
store(:description_args, [description]) if description
|
|
264
|
+
store(:caller, user_metadata.delete(:caller) || caller)
|
|
265
|
+
update(user_metadata)
|
|
266
|
+
end
|
|
267
|
+
|
|
268
|
+
private
|
|
269
|
+
|
|
270
|
+
RESERVED_KEYS = [
|
|
271
|
+
:description,
|
|
272
|
+
:example_group,
|
|
273
|
+
:execution_result,
|
|
274
|
+
:file_path,
|
|
275
|
+
:full_description,
|
|
276
|
+
:line_number,
|
|
277
|
+
:location
|
|
278
|
+
]
|
|
279
|
+
|
|
280
|
+
def ensure_valid_keys(user_metadata)
|
|
281
|
+
RESERVED_KEYS.each do |key|
|
|
282
|
+
if user_metadata.has_key?(key)
|
|
283
|
+
raise <<-EOM
|
|
284
|
+
#{"*"*50}
|
|
285
|
+
:#{key} is not allowed
|
|
286
|
+
|
|
287
|
+
RSpec reserves some hash keys for its own internal use,
|
|
288
|
+
including :#{key}, which is used on:
|
|
289
|
+
|
|
290
|
+
#{CallerFilter.first_non_rspec_line}.
|
|
291
|
+
|
|
292
|
+
Here are all of RSpec's reserved hash keys:
|
|
293
|
+
|
|
294
|
+
#{RESERVED_KEYS.join("\n ")}
|
|
295
|
+
#{"*"*50}
|
|
296
|
+
EOM
|
|
297
|
+
end
|
|
298
|
+
end
|
|
299
|
+
end
|
|
300
|
+
|
|
301
|
+
def example_group_declaration_line(locations)
|
|
302
|
+
locations[File.expand_path(self[:example_group][:file_path])] if self[:example_group]
|
|
303
|
+
end
|
|
304
|
+
|
|
305
|
+
# TODO - make this a method on metadata - the problem is
|
|
306
|
+
# metadata[:example_group] is not always a kind of GroupMetadataHash.
|
|
307
|
+
def relevant_line_numbers(metadata=self)
|
|
308
|
+
[metadata[:line_number]] + (metadata[:example_group] ? relevant_line_numbers(metadata[:example_group]) : [])
|
|
309
|
+
end
|
|
310
|
+
|
|
311
|
+
end
|
|
312
|
+
end
|
|
313
|
+
end
|