spectre-core 1.15.2 → 2.0.1

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.
@@ -1,310 +1,178 @@
1
- require_relative '../spectre'
2
- require_relative 'helpers'
3
- require_relative 'logging'
4
-
5
1
  require 'ostruct'
6
2
 
7
-
8
3
  module Spectre
9
4
  module Assertion
10
- class ::Object
11
- def should_be(value)
12
- evaluate(value, "#{self} should be #{value}") do |x|
13
- self.to_s == x.to_s
14
- end
15
- end
16
-
17
- def should_be_empty
18
- raise AssertionFailure.new("#{self.to_s.trim} should be empty", nil, self) unless self.nil?
19
- end
20
-
21
- def should_not_be(val)
22
- raise AssertionFailure.new("#{self.to_s.trim} should not be #{val.to_s.trim}", val, self) unless self.to_s != val.to_s
23
- end
24
-
25
- def should_not_exist
26
- raise AssertionFailure.new("#{self.to_s.trim} should not exist, but it does", val, self) unless self.to_s != nil
27
- end
28
-
29
- def should_not_be_empty
30
- raise AssertionFailure.new('empty value', 'nothing', self) unless self != nil
31
- end
32
-
33
- def evaluate val, message, &block
34
- val = Evaluation.new(val) unless val.is_a? Evaluation
35
- raise AssertionFailure.new(message, val, self) unless val.run &block
5
+ class ValueWrapper
6
+ def self.wrap val
7
+ val.is_a?(ValueWrapper) ? val : ValueWrapper.new(val)
36
8
  end
37
9
 
38
- def or other
39
- OrEvaluation.new(self, other)
40
- end
41
-
42
- def and other
43
- AndEvaluation.new(self, other)
44
- end
45
- end
46
-
47
- class ::NilClass
48
- def should_be(val)
49
- raise AssertionFailure.new("Value is empty, but it should be '#{val.to_s.trim}'", val, nil) unless val == nil
50
- end
51
-
52
- def should_be_empty
10
+ def initialize val
11
+ @val = val
53
12
  end
54
13
 
55
- def should_not_be(val)
56
- raise AssertionFailure.new(val, 'nil') unless val != nil
14
+ def evaluate predicate, actual, negate
15
+ !(!negate ^ predicate.call(@val, actual))
57
16
  end
58
17
 
59
- def should_not_exist
60
- end
18
+ # :nodoc:
19
+ def to_s
20
+ return "\"#{@val}\"" if @val.is_a?(String)
21
+ return @val.inspect if @val.is_a?(Regexp)
61
22
 
62
- def should_not_be_empty
63
- raise AssertionFailure.new('Value is empty', 'nil')
23
+ @val.to_s
64
24
  end
65
25
  end
66
26
 
67
- class ::Hash
68
- def should_contain(other)
69
- raise AssertionFailure.new(other, self) unless self.merge(other) == self
27
+ class OrWrapper < ValueWrapper
28
+ def initialize first, second
29
+ super(first)
30
+ @first = ValueWrapper.wrap(first)
31
+ @second = ValueWrapper.wrap(second)
70
32
  end
71
33
 
72
- def should_not_contain(other)
73
- raise AssertionFailure.new(other, self) unless self.merge(other) != self
34
+ def evaluate predicate, actual, negate
35
+ @first.evaluate(predicate, actual, negate) or @second.evaluate(predicate, actual, negate)
74
36
  end
75
37
 
76
- def should_be_empty
77
- raise AssertionFailure.new('Object should be empty', nil, self) unless self.empty?
78
- end
79
-
80
- def should_not_be_empty
81
- raise AssertionFailure.new('Object should not be empty', nil, self) if self.empty?
38
+ # :nodoc:
39
+ def to_s
40
+ "#{@first} or #{@second}"
82
41
  end
83
42
  end
84
43
 
85
- class ::OpenStruct
86
- def should_be_empty
87
- raise AssertionFailure.new('Object should be empty', nil, self) unless self.to_h.empty?
44
+ class AndWrapper < ValueWrapper
45
+ def initialize first, second
46
+ super(first)
47
+ @first = ValueWrapper.wrap(first)
48
+ @second = ValueWrapper.wrap(second)
88
49
  end
89
50
 
90
- def should_not_be_empty
91
- raise AssertionFailure.new('Object should not be empty', nil, self) if self.to_h.empty?
51
+ def evaluate predicate, actual, negate
52
+ @first.evaluate(predicate, actual, negate) and @second.evaluate(predicate, actual, negate)
92
53
  end
93
- end
94
-
95
- class ::Array
96
- def should_contain(val)
97
- list = self
98
-
99
- if val.is_a? Hash and self.all? { |x| x.is_a? OpenStruct or x.is_a? Hash }
100
- list = self.map { |x| OpenStruct.new(x) }
101
- val = OpenStruct.new(val)
102
- end
103
-
104
- evaluate(val, "#{list} should contain #{val.to_s}") do |x|
105
- list.include? x
106
- end
107
- end
108
-
109
- def should_not_contain(val)
110
- list = self
111
-
112
- if val.is_a? Hash and self.all? { |x| x.is_a? OpenStruct or x.is_a? Hash }
113
- list = self.map { |x| OpenStruct.new(x) }
114
- val = OpenStruct.new(val)
115
- end
116
54
 
117
- raise AssertionFailure.new("[#{list.join(', ').trim}] should not contain '#{val.to_s.trim}'", val, list) if list.include? val
118
- end
119
-
120
- def should_be_empty
121
- raise AssertionFailure.new('List is not empty', self) unless self.empty?
122
- end
123
-
124
- def should_not_be_empty
125
- raise AssertionFailure.new('List is empty', self) if self.empty?
55
+ # :nodoc:
56
+ def to_s
57
+ "#{@first} and #{@second}"
126
58
  end
127
59
  end
128
60
 
129
- class ::String
130
- def should_be(val)
131
- evaluate(val, "'#{self}' should be '#{val}'") do |x|
132
- self.to_s == x.to_s
133
- end
134
- end
135
-
136
- def should_be_empty
137
- raise AssertionFailure.new("'#{self.trim}' should be empty", nil, self) unless self.empty?
138
- end
61
+ class Evaluation
62
+ @@location_cache = {}
139
63
 
140
- def should_not_be(val)
141
- evaluate(val, "'#{self}' should not be '#{val}'") do |x|
142
- self.to_s != x.to_s
143
- end
144
- end
64
+ attr_reader :actual, :expected, :method, :negate, :desc, :failure, :call_location
145
65
 
146
- def should_not_be_empty
147
- raise AssertionFailure.new('Text should not be empty', 'nothing', self) unless not self.empty?
148
- end
66
+ def initialize call_location, actual, expected, method, predicate, negate: false
67
+ @call_location = call_location
68
+ @actual = actual
69
+ @expected = ValueWrapper.wrap(expected)
70
+ @predicate = predicate
71
+ @negate = negate
72
+ @failure = nil
149
73
 
150
- def should_contain(val)
151
- evaluate(val, "'#{self.trim}' should contain '#{val.to_s}'") do |x|
152
- self.include? x.to_s
153
- end
154
- end
74
+ # Maybe not the most elegant way, but it works for now
75
+ # as long as the `.to` call is on the same line as the variable
76
+ location = call_location
77
+ .find { |x| x.label.include? 'Spectre::Engine#load_files' or x.base_label == '<top (required)>' }
155
78
 
156
- def should_not_contain(val)
157
- evaluate(val, "'#{self}' should not contain '#{val}'") do |x|
158
- not self.include? x.to_s
159
- end
160
- end
79
+ path = location.path
161
80
 
162
- def should_match(regex)
163
- evaluate(regex, "'#{self.trim}' should match /#{regex}/") do |x|
164
- self.match(x)
81
+ if @@location_cache.key?(path)
82
+ file_content = @@location_cache[path]
83
+ else
84
+ file_content = File.read(path)
85
+ @@location_cache[path] = file_content
165
86
  end
166
- end
167
87
 
168
- def should_not_match(regex)
169
- evaluate(regex, "'#{self.trim}' should not match '#{regex}'") do |x|
170
- not self.match(x)
171
- end
172
- end
88
+ @var_name = file_content
89
+ .lines[location.lineno - 1]
90
+ .strip
91
+ .match(/[\s\(]([^\s]+|\[.*\]|{.*})\.(to|not_to)[\s\(]/)
92
+ .captures
93
+ .first
94
+ .strip
173
95
 
174
- alias :| :or
175
- alias :& :and
176
- end
96
+ @repr = @var_name
97
+ @repr += ' not' if @negate
98
+ @repr += " to #{method}"
99
+ @repr += expected.nil? ? ' empty' : " #{@expected}"
177
100
 
178
- class Evaluation
179
- def initialize val
180
- @val = val
181
- end
101
+ success = @expected.evaluate(@predicate, @actual, @negate)
182
102
 
183
- def run &block
184
- evaluate(@val, block)
185
- end
103
+ return if success
186
104
 
187
- def evaluate(val, predicate)
188
- if val.is_a? Evaluation
189
- val.run &predicate
190
- else
191
- predicate.call(val)
192
- end
105
+ @failure = if @negate
106
+ 'it does not'
107
+ else
108
+ "got #{@actual.inspect}"
109
+ end
193
110
  end
194
111
 
112
+ # :nodoc:
195
113
  def to_s
196
- @val.to_s
114
+ @repr
197
115
  end
198
116
  end
199
117
 
200
- class OrEvaluation < Evaluation
201
- def initialize val, other
202
- @val = val
203
- @other = other
204
- end
205
-
206
- def run &block
207
- res1 = evaluate(@val, block)
208
- res2 = evaluate(@other, block)
209
- res1 or res2
118
+ [
119
+ ::Array, ::Hash, ::String, ::Integer, ::Float,
120
+ ::NilClass, ::TrueClass, ::FalseClass, ::OpenStruct
121
+ ].each do |cls|
122
+ cls.define_method(:not_to) do |params|
123
+ Evaluation.new(caller_locations, self, *params, negate: true)
210
124
  end
211
125
 
212
- def to_s
213
- "(#{@val} or #{@other})"
126
+ cls.define_method(:to) do |params|
127
+ Evaluation.new(caller_locations, self, *params)
214
128
  end
215
129
  end
216
130
 
217
- class AndEvaluation < Evaluation
218
- def initialize val, other
219
- @val = val
220
- @other = other
131
+ [::Array, ::Hash, ::String, ::Integer, ::Float, ::Regexp].each do |cls|
132
+ cls.define_method(:or) do |other|
133
+ OrWrapper.new(self, other)
221
134
  end
222
135
 
223
- def run &block
224
- res1 = evaluate(@val, block)
225
- res2 = evaluate(@other, block)
226
- res1 and res2
227
- end
228
-
229
- def to_s
230
- "(#{@val} and #{@other})"
231
- end
232
- end
233
-
234
- class AssertionFailure < ExpectationFailure
235
- attr_reader :expected, :actual
236
-
237
- def initialize message, expected=nil, actual=nil, expectation=nil
238
- super message, expectation
239
- @expected = expected
240
- @actual = actual
136
+ cls.define_method(:and) do |other|
137
+ AndWrapper.new(self, other)
241
138
  end
242
139
  end
243
140
 
244
-
245
141
  class << self
246
- @@success = nil
247
-
248
- def expect desc
249
- status = 'unknown'
250
-
251
- begin
252
- Spectre::Logging.log_process("expect #{desc}")
253
- yield
254
- Spectre::Logging.log_status(desc, Logging::Status::OK)
255
- status = :ok
256
- rescue Interrupt => e
257
- status = :skipped
258
- raise e
259
- rescue AssertionFailure => e
260
- Spectre::Logging.log_status(desc, Logging::Status::FAILED)
261
- status = :failed
262
- raise AssertionFailure.new(e.message, e.expected, e.actual, desc), cause: nil
263
- rescue Exception => e
264
- Spectre::Logging.log_status(desc, Logging::Status::ERROR)
265
- status = :error
266
- raise AssertionFailure.new("An unexpected error occurred during expectation: #{e.message}", nil, nil, desc), cause: e
267
- ensure
268
- Spectre::Runner.current.expectations.append([desc, status])
269
- end
270
- end
271
-
272
- alias assert expect
273
-
274
- def observe desc = nil
275
- prefix = 'observing'
276
- prefix += " '#{desc}'" if desc
277
-
278
- begin
279
- Spectre::Logging.log_info(prefix) if desc
280
- yield
281
- @@success = true
282
- @@logger.info("#{prefix} finished with success")
283
- rescue Interrupt => e
284
- raise e
285
- rescue Exception => e
286
- error_message = "#{prefix} finished with failure: #{e.message}"
287
- error_message += "\n" + e.backtrace.join("\n") if @@debug
288
-
289
- @@logger.info(error_message)
290
- @@success = false
291
- end
292
- end
293
-
294
- def success?
295
- @@success
296
- end
297
-
298
- def fail_with message
299
- raise AssertionFailure.new(message)
142
+ def be expected_val
143
+ [
144
+ expected_val,
145
+ __method__,
146
+ proc { |expected, actual| expected.inspect == actual.inspect },
147
+ ]
148
+ end
149
+
150
+ def be_empty
151
+ [
152
+ nil,
153
+ 'be',
154
+ proc { |_, actual| actual.nil? or (actual.respond_to?(:empty?) and actual.empty?) }
155
+ ]
156
+ end
157
+
158
+ def contain expected_val
159
+ [
160
+ expected_val,
161
+ __method__,
162
+ proc do |expected, actual|
163
+ expected = expected.to_s if actual.is_a? String
164
+ actual.respond_to? :include? and actual.include?(expected)
165
+ end
166
+ ]
167
+ end
168
+
169
+ def match regex
170
+ [
171
+ regex,
172
+ __method__,
173
+ proc { |expected, actual| actual.match?(expected) }
174
+ ]
300
175
  end
301
176
  end
302
-
303
- Spectre.register do |config|
304
- @@logger = Spectre::Logging::ModuleLogger.new(config, 'spectre/assertion')
305
- @@debug = config['debug']
306
- end
307
-
308
- Spectre.delegate :expect, :assert, :observe, :success?, :fail_with, to: self
309
177
  end
310
178
  end
@@ -0,0 +1,89 @@
1
+ require_relative 'assertion'
2
+
3
+ module Spectre
4
+ ##
5
+ # This module provides old (Spectre v1) style assertions
6
+ # for compatibility reasons. This will be removed in the future
7
+ #
8
+ module Expectation
9
+ class ::Object
10
+ def should_be value
11
+ predicate = proc { |expected, actual| expected.to_s == actual.to_s }
12
+ value = Spectre::Assertion::ValueWrapper.wrap(value)
13
+ success = value.evaluate(predicate, self, false)
14
+
15
+ return if success
16
+
17
+ raise Failure, "#{self} should be #{value}"
18
+ end
19
+
20
+ def should_be_empty
21
+ predicate = proc { |_, actual| actual.nil? or (actual.respond_to?(:empty?) and actual.empty?) }
22
+ value = Spectre::Assertion::ValueWrapper.wrap(nil)
23
+ success = value.evaluate(predicate, self, false)
24
+
25
+ return if success
26
+
27
+ raise Failure, "#{self} should be empty"
28
+ end
29
+
30
+ def should_not_be(value)
31
+ predicate = proc { |expected, actual| expected.to_s == actual.to_s }
32
+ value = Spectre::Assertion::ValueWrapper.wrap(value)
33
+ success = value.evaluate(predicate, self, true)
34
+
35
+ return if success
36
+
37
+ raise Failure, "#{self} should not be #{value}"
38
+ end
39
+
40
+ def should_not_exist
41
+ predicate = proc { |expected, _| expected.respond_to? :nil? and expected.nil? }
42
+ value = Spectre::Assertion::ValueWrapper.wrap(value)
43
+ success = value.evaluate(predicate, self, true)
44
+
45
+ return if success
46
+
47
+ raise Failure, "#{self} should not exist"
48
+ end
49
+
50
+ def should_not_be_empty
51
+ predicate = proc { |_, actual| actual.nil? or (actual.respond_to?(:empty?) and actual.empty?) }
52
+ value = Spectre::Assertion::ValueWrapper.wrap(nil)
53
+ success = value.evaluate(predicate, self, true)
54
+
55
+ return if success
56
+
57
+ raise Failure, "#{self} should not be empty"
58
+ end
59
+
60
+ def should_contain value
61
+ predicate = proc do |expected, actual|
62
+ expected = expected.to_s if actual.is_a? String
63
+ actual.respond_to? :include? and actual.include?(expected)
64
+ end
65
+
66
+ value = Spectre::Assertion::ValueWrapper.wrap(value)
67
+ success = value.evaluate(predicate, self, false)
68
+
69
+ return if success
70
+
71
+ raise Failure, "#{self} should contain #{value}"
72
+ end
73
+
74
+ def should_not_contain value
75
+ predicate = proc do |expected, actual|
76
+ expected = expected.to_s if actual.is_a? String
77
+ actual.respond_to? :include? and actual.include?(expected)
78
+ end
79
+
80
+ value = Spectre::Assertion::ValueWrapper.wrap(value)
81
+ success = value.evaluate(predicate, self, true)
82
+
83
+ return if success
84
+
85
+ raise Failure, "#{self} should not contain #{value}"
86
+ end
87
+ end
88
+ end
89
+ end