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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: bb926aeeae60b8f3568595dfef40404aa9242b3bc1b44194d73b6716b662a315
4
- data.tar.gz: ae5969698e0dfee763d25a9918881c77b18b15f0642b2773813fe58a31ca63aa
3
+ metadata.gz: 297aab67c7235a6a0100db0b857b15d7897beeaa58aa8a8572b4b13df5ccba28
4
+ data.tar.gz: a5694c1b122ed3c135790b04e837d88be9baad28c2b5b8cb594311339dc89c34
5
5
  SHA512:
6
- metadata.gz: 680f7ffeea26ec3fec0884883a3dbd46989ea14cf76d7daf29457032cbb5ba68f0fe8391f3e6edcd244689bc8f5e791b82a68b703e64eeb802190ebcfff5c001
7
- data.tar.gz: 7ac2a6f5988916d62745312ef1f3bc16fab09a424204d0643d0dcddce5b5e916afe9f879847fc9256f77db6dd6330ee555d355b76461ed59667d4c450e1b615f
6
+ metadata.gz: 6c68dfdb65de4636f3cbc85703f6224e7ff7bcf9edd77cdf92aa67de47fd7086ceb2b1c4b91317a944f0acd618c03c8caa9c3f2c5e76f004b4e0fe779adc7085
7
+ data.tar.gz: 994aa8228e557a3baef8612653dc5d6d30377b2c405eb6f35508acd0c953865d23bd2da0f73be5492ea70c538fdb9ba2ef3fcc4f0f1253e7b6ee9cce2ea8e35a
data/README.md CHANGED
@@ -1,4 +1,4 @@
1
- [![Test](https://github.com/enspirit/webspicy/actions/workflows/integration.yml/badge.svg?branch=master)](https://github.com/enspirit/webspicy/actions/workflows/integration.yml)
1
+ [![Test](https://github.com/enspirit/webspicy/actions/workflows/integration.yml/badge.svg)](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
- def merge_maps(h1, h2)
146
- h1.merge(h2) do |k,v1,v2|
147
- case v1
148
- when Hash then merge_maps(v1, v2)
149
- when Array then v1 + v2
150
- else v2
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
@@ -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'
@@ -10,7 +10,7 @@ module Webspicy
10
10
  @reporter << Reporter::SuccessOrNot.new
11
11
  end
12
12
 
13
- def run_scope
13
+ def run_scope(all = true)
14
14
  scope.each_specification_file do |spec_file|
15
15
  @specification = load_specification(spec_file)
16
16
  reporter.spec_file_done
@@ -51,16 +51,14 @@ module Webspicy
51
51
  end
52
52
 
53
53
  def before_service
54
- @spec_file_line_printed = false
54
+ @test_case_seen = false
55
55
  end
56
56
 
57
57
  def before_test_case
58
- unless @spec_file_line_printed
59
- io.puts spec_file_line(spec_file)
60
- io.puts
61
- io.flush
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
- unless @spec_file_line_printed
90
- io.puts
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
- @spec_file_errors = []
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
- @spec_files_count = 0
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
- @spec_files_count = 0
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
@@ -58,7 +58,8 @@ module Webspicy
58
58
  :after_each_done,
59
59
  :after_all,
60
60
  :after_all_done,
61
- :report
61
+ :report,
62
+ :clear
62
63
  ]
63
64
 
64
65
  HOOKS.each do |meth|
@@ -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
- reporter.before_all
78
- hooks.fire_before_all(self)
79
- reporter.before_all_done
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
- reporter.after_all
94
- hooks.fire_after_all(self)
95
- reporter.after_all_done
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)
@@ -1,8 +1,8 @@
1
1
  module Webspicy
2
2
  module Version
3
3
  MAJOR = 0
4
- MINOR = 20
5
- TINY = 22
4
+ MINOR = 21
5
+ TINY = 0
6
6
  end
7
7
  VERSION = "#{Version::MAJOR}.#{Version::MINOR}.#{Version::TINY}"
8
8
  end
@@ -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.20.22
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: 2021-12-17 00:00:00.000000000 Z
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.1.4
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!