webspicy 0.20.22 → 0.21.0
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/README.md +1 -1
- data/bin/webspicy +4 -0
- data/lib/webspicy/configuration/scope.rb +6 -13
- data/lib/webspicy/configuration.rb +8 -0
- data/lib/webspicy/support/deep_merge.rb +36 -0
- data/lib/webspicy/support.rb +1 -0
- data/lib/webspicy/tester/file_checker.rb +1 -1
- data/lib/webspicy/tester/reporter/documentation.rb +7 -11
- data/lib/webspicy/tester/reporter/exceptions.rb +6 -2
- data/lib/webspicy/tester/reporter/file_summary.rb +6 -2
- data/lib/webspicy/tester/reporter/summary.rb +12 -8
- data/lib/webspicy/tester/reporter.rb +2 -1
- data/lib/webspicy/tester.rb +28 -10
- data/lib/webspicy/version.rb +2 -2
- data/spec/unit/support/test_deep_merge.rb +45 -0
- data/spec/unit/test_configuration.rb +29 -0
- metadata +19 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 297aab67c7235a6a0100db0b857b15d7897beeaa58aa8a8572b4b13df5ccba28
|
4
|
+
data.tar.gz: a5694c1b122ed3c135790b04e837d88be9baad28c2b5b8cb594311339dc89c34
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 6c68dfdb65de4636f3cbc85703f6224e7ff7bcf9edd77cdf92aa67de47fd7086ceb2b1c4b91317a944f0acd618c03c8caa9c3f2c5e76f004b4e0fe779adc7085
|
7
|
+
data.tar.gz: 994aa8228e557a3baef8612653dc5d6d30377b2c405eb6f35508acd0c953865d23bd2da0f73be5492ea70c538fdb9ba2ef3fcc4f0f1253e7b6ee9cce2ea8e35a
|
data/README.md
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
[](https://github.com/enspirit/webspicy/actions/workflows/integration.yml/badge.svg)
|
2
2
|
# Webspicy
|
3
3
|
|
4
4
|
A specification and test framework for web services seen as black-box software
|
data/bin/webspicy
CHANGED
@@ -21,6 +21,7 @@
|
|
21
21
|
#/ - TAG=... execute only tests matching the given tag
|
22
22
|
#/ - FAILFAST=yes|no stop executing tests on first failure
|
23
23
|
#/ - INSECURE=yes|no allow insecure server connections when using SSL
|
24
|
+
#/ - WATCH=FOLDER,FOLDER,... run in watch mode (run tests when files change)
|
24
25
|
#/
|
25
26
|
require 'webspicy'
|
26
27
|
require 'webspicy/tester'
|
@@ -42,6 +43,9 @@ ARGV.options do |opts|
|
|
42
43
|
opts.on('-k', '--insecure') {
|
43
44
|
ENV['INSECURE'] = 'yes'
|
44
45
|
}
|
46
|
+
opts.on('-w', '--watch') {
|
47
|
+
ENV['WATCH'] = '.'
|
48
|
+
}
|
45
49
|
opts.on("--debug") {
|
46
50
|
ENV['LOG_LEVEL'] = 'DEBUG'
|
47
51
|
Webspicy::LOGGER.level = Logger.const_get(ENV['LOG_LEVEL'])
|
@@ -136,20 +136,13 @@ module Webspicy
|
|
136
136
|
|
137
137
|
def expand_example(service, example)
|
138
138
|
return example unless service.default_example
|
139
|
-
h1 = service.default_example.to_info
|
140
|
-
h2 = example.to_info
|
141
|
-
ex = config.factory.test_case(merge_maps(h1, h2), self)
|
142
|
-
ex.bind(service, example.counterexample?)
|
143
|
-
end
|
144
139
|
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
end
|
152
|
-
end
|
140
|
+
merged = Support::DeepMerge.deep_merge(
|
141
|
+
service.default_example.to_info,
|
142
|
+
example.to_info
|
143
|
+
)
|
144
|
+
ex = config.factory.test_case(merged, self)
|
145
|
+
ex.bind(service, example.counterexample?)
|
153
146
|
end
|
154
147
|
|
155
148
|
# Returns a proc that implements file_filter strategy according to the
|
@@ -43,6 +43,7 @@ module Webspicy
|
|
43
43
|
@scope_factory = ->(config){ Scope.new(config) }
|
44
44
|
@client = Web::HttpClient
|
45
45
|
@reporter = default_reporter
|
46
|
+
@watch_list = default_watchlist
|
46
47
|
Path.require_tree(@folder/'support') if (@folder/'support').exists?
|
47
48
|
@world = Support::World.new(folder/'world', self)
|
48
49
|
yield(self) if block_given?
|
@@ -451,6 +452,13 @@ module Webspicy
|
|
451
452
|
end
|
452
453
|
attr_accessor :reporter
|
453
454
|
|
455
|
+
def default_watchlist
|
456
|
+
return nil unless list = ENV['WATCH']
|
457
|
+
|
458
|
+
list.split(',').map{|x| Path.pwd/x }
|
459
|
+
end
|
460
|
+
attr_accessor :watch_list
|
461
|
+
|
454
462
|
# Returns the Data system to use for parsing schemas
|
455
463
|
#
|
456
464
|
# The data system associated with a configuration is build when the
|
@@ -0,0 +1,36 @@
|
|
1
|
+
module Webspicy
|
2
|
+
module Support
|
3
|
+
class DeepMerge
|
4
|
+
class << self
|
5
|
+
|
6
|
+
def deep_merge(h1, h2)
|
7
|
+
merge_maps(deep_dup(h1), deep_dup(h2))
|
8
|
+
end
|
9
|
+
|
10
|
+
def deep_dup(x)
|
11
|
+
case x
|
12
|
+
when Array
|
13
|
+
x.map{|y| deep_dup(y) }
|
14
|
+
when Hash
|
15
|
+
x.each_with_object({}){|(k,v),memo| memo[k] = deep_dup(v) }
|
16
|
+
else
|
17
|
+
x
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
private
|
22
|
+
|
23
|
+
def merge_maps(h1, h2)
|
24
|
+
h1.merge(h2) do |k,v1,v2|
|
25
|
+
case v1
|
26
|
+
when Hash then merge_maps(v1, v2)
|
27
|
+
when Array then v1 + v2
|
28
|
+
else v2
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
data/lib/webspicy/support.rb
CHANGED
@@ -28,6 +28,7 @@ module Webspicy
|
|
28
28
|
end # module Support
|
29
29
|
end # module Webspicy
|
30
30
|
require_relative 'support/data_object'
|
31
|
+
require_relative 'support/deep_merge'
|
31
32
|
require_relative 'support/status_range'
|
32
33
|
require_relative 'support/colorize'
|
33
34
|
require_relative 'support/world'
|
@@ -51,16 +51,14 @@ module Webspicy
|
|
51
51
|
end
|
52
52
|
|
53
53
|
def before_service
|
54
|
-
@
|
54
|
+
@test_case_seen = false
|
55
55
|
end
|
56
56
|
|
57
57
|
def before_test_case
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
@spec_file_line_printed = true
|
63
|
-
end
|
58
|
+
@test_case_seen = true
|
59
|
+
io.puts spec_file_line(spec_file)
|
60
|
+
io.puts
|
61
|
+
io.flush
|
64
62
|
io.puts service_line(service, test_case)
|
65
63
|
io.flush
|
66
64
|
end
|
@@ -86,10 +84,8 @@ module Webspicy
|
|
86
84
|
end
|
87
85
|
|
88
86
|
def service_done
|
89
|
-
|
90
|
-
|
91
|
-
io.flush
|
92
|
-
end
|
87
|
+
io.puts if @test_case_seen
|
88
|
+
io.flush
|
93
89
|
end
|
94
90
|
|
95
91
|
end # class Documentation
|
@@ -6,8 +6,7 @@ module Webspicy
|
|
6
6
|
|
7
7
|
def initialize(*args, &bl)
|
8
8
|
super
|
9
|
-
|
10
|
-
@failed_results = []
|
9
|
+
clear
|
11
10
|
end
|
12
11
|
attr_reader :failed_results, :spec_file_errors
|
13
12
|
|
@@ -24,6 +23,11 @@ module Webspicy
|
|
24
23
|
report_failed_results
|
25
24
|
end
|
26
25
|
|
26
|
+
def clear
|
27
|
+
@spec_file_errors = []
|
28
|
+
@failed_results = []
|
29
|
+
end
|
30
|
+
|
27
31
|
private
|
28
32
|
|
29
33
|
def report_spec_file_errors
|
@@ -5,8 +5,7 @@ module Webspicy
|
|
5
5
|
|
6
6
|
def initialize(*args, &bl)
|
7
7
|
super
|
8
|
-
|
9
|
-
@errors_count = 0
|
8
|
+
clear
|
10
9
|
end
|
11
10
|
attr_reader :spec_files_count, :errors_count
|
12
11
|
|
@@ -31,6 +30,11 @@ module Webspicy
|
|
31
30
|
io.flush
|
32
31
|
end
|
33
32
|
|
33
|
+
def clear
|
34
|
+
@spec_files_count = 0
|
35
|
+
@errors_count = 0
|
36
|
+
end
|
37
|
+
|
34
38
|
private
|
35
39
|
|
36
40
|
def success?
|
@@ -5,14 +5,7 @@ module Webspicy
|
|
5
5
|
|
6
6
|
def initialize(*args, &bl)
|
7
7
|
super
|
8
|
-
|
9
|
-
@examples_count = 0
|
10
|
-
@counterexamples_count = 0
|
11
|
-
@assertions_count = 0
|
12
|
-
#
|
13
|
-
@spec_file_errors_count = 0
|
14
|
-
@errors_count = 0
|
15
|
-
@failures_count = 0
|
8
|
+
clear
|
16
9
|
end
|
17
10
|
attr_reader :spec_files_count, :examples_count, :counterexamples_count
|
18
11
|
attr_reader :assertions_count
|
@@ -54,6 +47,17 @@ module Webspicy
|
|
54
47
|
io.flush
|
55
48
|
end
|
56
49
|
|
50
|
+
def clear
|
51
|
+
@spec_files_count = 0
|
52
|
+
@examples_count = 0
|
53
|
+
@counterexamples_count = 0
|
54
|
+
@assertions_count = 0
|
55
|
+
#
|
56
|
+
@spec_file_errors_count = 0
|
57
|
+
@errors_count = 0
|
58
|
+
@failures_count = 0
|
59
|
+
end
|
60
|
+
|
57
61
|
def total_error_count
|
58
62
|
@spec_file_errors_count + @errors_count + @failures_count
|
59
63
|
end
|
data/lib/webspicy/tester.rb
CHANGED
@@ -32,9 +32,23 @@ module Webspicy
|
|
32
32
|
end
|
33
33
|
|
34
34
|
def call
|
35
|
+
res = call_once(true)
|
36
|
+
if folders = config.watch_list
|
37
|
+
require 'listen'
|
38
|
+
listener = Listen.to(*folders) do
|
39
|
+
reporter.clear
|
40
|
+
res = call_once(false)
|
41
|
+
end
|
42
|
+
listener.start
|
43
|
+
sleep
|
44
|
+
end
|
45
|
+
res
|
46
|
+
end
|
47
|
+
|
48
|
+
def call_once(all = true)
|
35
49
|
reporter.init(self)
|
36
50
|
begin
|
37
|
-
run_config
|
51
|
+
run_config(all)
|
38
52
|
rescue FailFast
|
39
53
|
end
|
40
54
|
reporter.report
|
@@ -64,19 +78,21 @@ module Webspicy
|
|
64
78
|
|
65
79
|
protected
|
66
80
|
|
67
|
-
def run_config
|
81
|
+
def run_config(all = true)
|
68
82
|
config.each_scope do |scope|
|
69
83
|
@scope = scope
|
70
84
|
@hooks = Support::Hooks.for(scope.config)
|
71
85
|
@client = scope.get_client
|
72
|
-
run_scope
|
86
|
+
run_scope(all)
|
73
87
|
end
|
74
88
|
end
|
75
89
|
|
76
|
-
def run_scope
|
77
|
-
|
78
|
-
|
79
|
-
|
90
|
+
def run_scope(all = true)
|
91
|
+
if all
|
92
|
+
reporter.before_all
|
93
|
+
hooks.fire_before_all(self)
|
94
|
+
reporter.before_all_done
|
95
|
+
end
|
80
96
|
reporter.before_scope
|
81
97
|
scope.each_specification_file do |spec_file|
|
82
98
|
@specification = load_specification(spec_file)
|
@@ -90,9 +106,11 @@ module Webspicy
|
|
90
106
|
end
|
91
107
|
end
|
92
108
|
reporter.scope_done
|
93
|
-
|
94
|
-
|
95
|
-
|
109
|
+
if all
|
110
|
+
reporter.after_all
|
111
|
+
hooks.fire_after_all(self)
|
112
|
+
reporter.after_all_done
|
113
|
+
end
|
96
114
|
end
|
97
115
|
|
98
116
|
def load_specification(spec_file)
|
data/lib/webspicy/version.rb
CHANGED
@@ -0,0 +1,45 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
module Webspicy
|
3
|
+
module Support
|
4
|
+
describe DeepMerge, 'deep_dup' do
|
5
|
+
it 'works on scalars' do
|
6
|
+
expect(DeepMerge.deep_dup(12)).to eql(12)
|
7
|
+
end
|
8
|
+
|
9
|
+
it 'works on arrays' do
|
10
|
+
xs = [1, 2, 3]
|
11
|
+
expect(DeepMerge.deep_dup(xs)).to eql(xs)
|
12
|
+
expect(DeepMerge.deep_dup(xs)).not_to be(xs)
|
13
|
+
end
|
14
|
+
|
15
|
+
it 'works on hashes' do
|
16
|
+
xs = { foo: "bar" }
|
17
|
+
expect(DeepMerge.deep_dup(xs)).to eql(xs)
|
18
|
+
expect(DeepMerge.deep_dup(xs)).not_to be(xs)
|
19
|
+
end
|
20
|
+
|
21
|
+
it 'works recursively' do
|
22
|
+
xs = { foo: { bar: [1, 2, 3] } }
|
23
|
+
got = DeepMerge.deep_dup(xs)
|
24
|
+
expect(got[:foo]).not_to be(xs[:foo])
|
25
|
+
expect(got[:foo][:bar]).not_to be(xs[:foo][:bar])
|
26
|
+
end
|
27
|
+
end
|
28
|
+
describe DeepMerge, 'deep_merge' do
|
29
|
+
it 'works as expected' do
|
30
|
+
h1 = { foo: { bar: "baz" }, times: [1, 2], only: { me: true } }
|
31
|
+
h2 = { foo: { bor: "boz" }, times: [3, 4], not: { you: false } }
|
32
|
+
expected = {
|
33
|
+
foo: { bar: "baz", bor: "boz" },
|
34
|
+
times: [1, 2, 3, 4],
|
35
|
+
only: { me: true },
|
36
|
+
not: { you: false }
|
37
|
+
}
|
38
|
+
got = DeepMerge.deep_merge(h1, h2)
|
39
|
+
expect(got).to eql(expected)
|
40
|
+
expect(got[:only]).not_to be(h1[:only])
|
41
|
+
expect(got[:not]).not_to be(h2[:not])
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
@@ -184,6 +184,35 @@ module Webspicy
|
|
184
184
|
end
|
185
185
|
end
|
186
186
|
|
187
|
+
describe 'watch_list' do
|
188
|
+
|
189
|
+
it 'is nil by default' do
|
190
|
+
config = Configuration.new
|
191
|
+
expect(config.watch_list).to be_nil
|
192
|
+
end
|
193
|
+
|
194
|
+
it 'supports a single relative folder on WATCH env variable' do
|
195
|
+
ENV['WATCH'] = 'a_folder'
|
196
|
+
config = Configuration.new
|
197
|
+
expect(config.watch_list).to eql([Path.pwd/"a_folder"])
|
198
|
+
ENV.delete('WATCH')
|
199
|
+
end
|
200
|
+
|
201
|
+
it 'supports a commalist on WATCH env variable' do
|
202
|
+
ENV['WATCH'] = 'a_folder,another'
|
203
|
+
config = Configuration.new
|
204
|
+
expect(config.watch_list).to eql([Path.pwd/"a_folder", Path.pwd/'another'])
|
205
|
+
ENV.delete('WATCH')
|
206
|
+
end
|
207
|
+
|
208
|
+
it 'supports absolute folders too' do
|
209
|
+
ENV['WATCH'] = 'a_folder,/tmp'
|
210
|
+
config = Configuration.new
|
211
|
+
expect(config.watch_list).to eql([Path.pwd/"a_folder", Path('/tmp')])
|
212
|
+
ENV.delete('WATCH')
|
213
|
+
end
|
214
|
+
end
|
215
|
+
|
187
216
|
describe 'insecure' do
|
188
217
|
|
189
218
|
it 'is false by default' do
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: webspicy
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.21.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Bernard Lambeau
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2022-06-21 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rake
|
@@ -246,6 +246,20 @@ dependencies:
|
|
246
246
|
- - "~>"
|
247
247
|
- !ruby/object:Gem::Version
|
248
248
|
version: 0.7.0
|
249
|
+
- !ruby/object:Gem::Dependency
|
250
|
+
name: listen
|
251
|
+
requirement: !ruby/object:Gem::Requirement
|
252
|
+
requirements:
|
253
|
+
- - "~>"
|
254
|
+
- !ruby/object:Gem::Version
|
255
|
+
version: '3.7'
|
256
|
+
type: :runtime
|
257
|
+
prerelease: false
|
258
|
+
version_requirements: !ruby/object:Gem::Requirement
|
259
|
+
requirements:
|
260
|
+
- - "~>"
|
261
|
+
- !ruby/object:Gem::Version
|
262
|
+
version: '3.7'
|
249
263
|
description: Webspicy helps testing web services as software operation black boxes
|
250
264
|
email: blambeau@gmail.com
|
251
265
|
executables:
|
@@ -287,6 +301,7 @@ files:
|
|
287
301
|
- lib/webspicy/support.rb
|
288
302
|
- lib/webspicy/support/colorize.rb
|
289
303
|
- lib/webspicy/support/data_object.rb
|
304
|
+
- lib/webspicy/support/deep_merge.rb
|
290
305
|
- lib/webspicy/support/hooks.rb
|
291
306
|
- lib/webspicy/support/status_range.rb
|
292
307
|
- lib/webspicy/support/world.rb
|
@@ -354,6 +369,7 @@ files:
|
|
354
369
|
- spec/unit/support/hooks/test_fire_after_each.rb
|
355
370
|
- spec/unit/support/hooks/test_fire_around.rb
|
356
371
|
- spec/unit/support/hooks/test_fire_before_each.rb
|
372
|
+
- spec/unit/support/test_deep_merge.rb
|
357
373
|
- spec/unit/support/test_status_range.rb
|
358
374
|
- spec/unit/support/world/fixtures/array.json
|
359
375
|
- spec/unit/support/world/fixtures/queue.rb
|
@@ -392,7 +408,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
392
408
|
- !ruby/object:Gem::Version
|
393
409
|
version: '0'
|
394
410
|
requirements: []
|
395
|
-
rubygems_version: 3.
|
411
|
+
rubygems_version: 3.3.7
|
396
412
|
signing_key:
|
397
413
|
specification_version: 4
|
398
414
|
summary: Webspicy helps testing web services as software operation black boxes!
|