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.
- checksums.yaml +4 -4
- data/exe/spectre +271 -463
- data/lib/spectre/assertion.rb +118 -250
- data/lib/spectre/expectation.rb +89 -0
- data/lib/spectre/helpers.rb +56 -59
- data/lib/spectre/version.rb +3 -0
- data/lib/spectre.rb +1510 -334
- metadata +57 -32
- data/lib/spectre/async.rb +0 -31
- data/lib/spectre/bag.rb +0 -17
- data/lib/spectre/curl.rb +0 -398
- data/lib/spectre/diagnostic.rb +0 -39
- data/lib/spectre/environment.rb +0 -30
- data/lib/spectre/http/basic_auth.rb +0 -25
- data/lib/spectre/http/keystone.rb +0 -99
- data/lib/spectre/http.rb +0 -394
- data/lib/spectre/logging/console.rb +0 -156
- data/lib/spectre/logging/file.rb +0 -106
- data/lib/spectre/logging.rb +0 -183
- data/lib/spectre/mixin.rb +0 -61
- data/lib/spectre/reporter/console.rb +0 -104
- data/lib/spectre/reporter.rb +0 -17
- data/lib/spectre/resources.rb +0 -53
data/lib/spectre/assertion.rb
CHANGED
@@ -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
|
11
|
-
def
|
12
|
-
|
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
|
39
|
-
|
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
|
56
|
-
|
14
|
+
def evaluate predicate, actual, negate
|
15
|
+
!(!negate ^ predicate.call(@val, actual))
|
57
16
|
end
|
58
17
|
|
59
|
-
|
60
|
-
|
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
|
-
|
63
|
-
raise AssertionFailure.new('Value is empty', 'nil')
|
23
|
+
@val.to_s
|
64
24
|
end
|
65
25
|
end
|
66
26
|
|
67
|
-
class
|
68
|
-
def
|
69
|
-
|
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
|
73
|
-
|
34
|
+
def evaluate predicate, actual, negate
|
35
|
+
@first.evaluate(predicate, actual, negate) or @second.evaluate(predicate, actual, negate)
|
74
36
|
end
|
75
37
|
|
76
|
-
|
77
|
-
|
78
|
-
|
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
|
86
|
-
def
|
87
|
-
|
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
|
91
|
-
|
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
|
-
|
118
|
-
|
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
|
130
|
-
|
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
|
-
|
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
|
147
|
-
|
148
|
-
|
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
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
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
|
-
|
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
|
-
|
163
|
-
|
164
|
-
|
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
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
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
|
-
|
175
|
-
|
176
|
-
|
96
|
+
@repr = @var_name
|
97
|
+
@repr += ' not' if @negate
|
98
|
+
@repr += " to #{method}"
|
99
|
+
@repr += expected.nil? ? ' empty' : " #{@expected}"
|
177
100
|
|
178
|
-
|
179
|
-
def initialize val
|
180
|
-
@val = val
|
181
|
-
end
|
101
|
+
success = @expected.evaluate(@predicate, @actual, @negate)
|
182
102
|
|
183
|
-
|
184
|
-
evaluate(@val, block)
|
185
|
-
end
|
103
|
+
return if success
|
186
104
|
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
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
|
-
@
|
114
|
+
@repr
|
197
115
|
end
|
198
116
|
end
|
199
117
|
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
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
|
-
|
213
|
-
|
126
|
+
cls.define_method(:to) do |params|
|
127
|
+
Evaluation.new(caller_locations, self, *params)
|
214
128
|
end
|
215
129
|
end
|
216
130
|
|
217
|
-
|
218
|
-
|
219
|
-
|
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
|
-
|
224
|
-
|
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
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
|
254
|
-
|
255
|
-
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
267
|
-
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
272
|
-
|
273
|
-
|
274
|
-
|
275
|
-
|
276
|
-
|
277
|
-
|
278
|
-
|
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
|