prop_check 0.6.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.
@@ -0,0 +1,207 @@
1
+ require 'stringio'
2
+
3
+ require 'prop_check/property/configuration'
4
+ require 'prop_check/property/check_evaluator'
5
+ module PropCheck
6
+ class Property
7
+
8
+ def self.forall(**bindings, &block)
9
+
10
+ property = new(bindings)
11
+
12
+ return property.check(&block) if block_given?
13
+
14
+ property
15
+ end
16
+
17
+ def self.configuration
18
+ @configuration ||= Configuration.new
19
+ end
20
+
21
+ def self.configure
22
+ yield(configuration)
23
+ end
24
+
25
+ attr_reader :bindings, :condition
26
+
27
+ def initialize(**bindings)
28
+ raise ArgumentError, 'No bindings specified!' if bindings.empty?
29
+
30
+ @bindings = bindings
31
+ @condition = -> { true }
32
+ @config = self.class.configuration
33
+ end
34
+
35
+ def configuration
36
+ @config
37
+ end
38
+
39
+ def with_config(**config, &block)
40
+ @config = @config.merge(config)
41
+
42
+ return self.check(&block) if block_given?
43
+
44
+ self
45
+ end
46
+
47
+ def where(&new_condition)
48
+ original_condition = @condition.dup
49
+ @condition = -> { instance_exec(&original_condition) && instance_exec(&new_condition) }
50
+
51
+ self
52
+ end
53
+
54
+ def check(&block)
55
+ binding_generator = PropCheck::Generators.fixed_hash(bindings)
56
+
57
+ n_runs = 0
58
+ n_successful = 0
59
+
60
+ # Loop stops at first exception
61
+ attempts_enumerator(binding_generator).each do |generator_result|
62
+ n_runs += 1
63
+ check_attempt(generator_result, n_successful, &block)
64
+ n_successful += 1
65
+ end
66
+
67
+ ensure_not_exhausted!(n_runs)
68
+ end
69
+
70
+ private def ensure_not_exhausted!(n_runs)
71
+ return if n_runs >= @config.n_runs
72
+
73
+ raise GeneratorExhaustedError, """
74
+ Could not perform `n_runs = #{@config.n_runs}` runs,
75
+ (exhausted #{@config.max_generate_attempts} tries)
76
+ because too few generator results were adhering to
77
+ the `where` condition.
78
+
79
+ Try refining your generators instead.
80
+ """
81
+ end
82
+
83
+ private def check_attempt(generator_result, n_successful, &block)
84
+ CheckEvaluator.new(generator_result.root, &block).call
85
+
86
+ # immediately stop (without shrinnking) for when the app is asked
87
+ # to close by outside intervention
88
+ rescue SignalException, SystemExit
89
+ raise
90
+
91
+ # We want to capture _all_ exceptions (even low-level ones) here,
92
+ # so we can shrink to find their cause.
93
+ # don't worry: they all get reraised
94
+ rescue Exception => e
95
+ output, shrunken_result, shrunken_exception, n_shrink_steps = show_problem_output(e, generator_result, n_successful, &block)
96
+ output_string = output.is_a?(StringIO) ? output.string : e.message
97
+
98
+ e.define_singleton_method :prop_check_info do
99
+ {
100
+ original_input: generator_result.root,
101
+ original_exception_message: e.message,
102
+ shrunken_input: shrunken_result,
103
+ shrunken_exception: shrunken_exception,
104
+ n_successful: n_successful,
105
+ n_shrink_steps: n_shrink_steps
106
+ }
107
+ end
108
+
109
+ raise e, output_string, e.backtrace
110
+ end
111
+
112
+ private def attempts_enumerator(binding_generator)
113
+
114
+ rng = Random::DEFAULT
115
+ n_runs = 0
116
+ size = 1
117
+ (0...@config.max_generate_attempts)
118
+ .lazy
119
+ .map { binding_generator.generate(size, rng) }
120
+ .reject { |val| val.root == :"_PropCheck.filter_me" }
121
+ .select { |val| CheckEvaluator.new(val.root, &@condition).call }
122
+ .map do |result|
123
+ n_runs += 1
124
+ size += 1
125
+
126
+ result
127
+ end
128
+ .take_while { n_runs <= @config.n_runs }
129
+ end
130
+
131
+ private def show_problem_output(problem, generator_results, n_successful, &block)
132
+ output = @config.verbose ? STDOUT : StringIO.new
133
+ output = pre_output(output, n_successful, generator_results.root, problem)
134
+ shrunken_result, shrunken_exception, n_shrink_steps = shrink2(generator_results, output, &block)
135
+ output = post_output(output, n_shrink_steps, shrunken_result, shrunken_exception)
136
+
137
+ [output, shrunken_result, shrunken_exception, n_shrink_steps]
138
+ end
139
+
140
+ private def pre_output(output, n_successful, generated_root, problem)
141
+ output.puts ""
142
+ output.puts "(after #{n_successful} successful property test runs)"
143
+ output.puts "Failed on: "
144
+ output.puts "`#{print_roots(generated_root)}`"
145
+ output.puts ""
146
+ output.puts "Exception message:\n---\n#{problem}"
147
+ output.puts "---"
148
+ output.puts ""
149
+
150
+ output
151
+ end
152
+
153
+ private def post_output(output, n_shrink_steps, shrunken_result, shrunken_exception)
154
+ output.puts ''
155
+ output.puts "Shrunken input (after #{n_shrink_steps} shrink steps):"
156
+ output.puts "`#{print_roots(shrunken_result)}`"
157
+ output.puts ""
158
+ output.puts "Shrunken exception:\n---\n#{shrunken_exception}"
159
+ output.puts "---"
160
+ output.puts ""
161
+
162
+ output
163
+ end
164
+
165
+ private def print_roots(lazy_tree_hash)
166
+ lazy_tree_hash.map do |name, val|
167
+ "#{name} = #{val.inspect}"
168
+ end.join(", ")
169
+ end
170
+
171
+ private def shrink2(bindings_tree, io, &fun)
172
+ io.puts 'Shrinking...' if @config.verbose
173
+ problem_child = bindings_tree
174
+ siblings = problem_child.children.lazy
175
+ parent_siblings = nil
176
+ problem_exception = nil
177
+ shrink_steps = 0
178
+ (0..@config.max_shrink_steps).each do
179
+ begin
180
+ sibling = siblings.next
181
+ rescue StopIteration
182
+ break if parent_siblings.nil?
183
+
184
+ siblings = parent_siblings.lazy
185
+ parent_siblings = nil
186
+ next
187
+ end
188
+
189
+ shrink_steps += 1
190
+ io.print '.' if @config.verbose
191
+
192
+ begin
193
+ CheckEvaluator.new(sibling.root, &fun).call
194
+ rescue Exception => problem
195
+ problem_child = sibling
196
+ parent_siblings = siblings
197
+ siblings = problem_child.children.lazy
198
+ problem_exception = problem
199
+ end
200
+ end
201
+
202
+ io.puts "(Note: Exceeded #{@config.max_shrink_steps} shrinking steps, the maximum.)" if shrink_steps >= @config.max_shrink_steps
203
+
204
+ [problem_child.root, problem_exception, shrink_steps]
205
+ end
206
+ end
207
+ end
@@ -0,0 +1,45 @@
1
+ module PropCheck
2
+ class Property
3
+ ##
4
+ # A wrapper class that implements the 'Cloaker' concept
5
+ # which allows us to refer to variables set in 'bindings',
6
+ # while still being able to access things that are only in scope
7
+ # in the creator of '&block'.
8
+ #
9
+ # This allows us to bind the variables specified in `bindings`
10
+ # one way during checking and another way during shrinking.
11
+ class CheckEvaluator
12
+ include RSpec::Matchers if Object.const_defined?('RSpec')
13
+
14
+ def initialize(bindings, &block)
15
+ @caller = block.binding.receiver
16
+ @block = block
17
+ define_named_instance_methods(bindings)
18
+ end
19
+
20
+ def call
21
+ self.instance_exec(&@block)
22
+ end
23
+
24
+ private def define_named_instance_methods(results)
25
+ results.each do |name, result|
26
+ define_singleton_method(name) { result }
27
+ end
28
+ end
29
+
30
+ ##
31
+ # Dispatches to caller whenever something is not part of `bindings`.
32
+ # (No need to invoke this method manually)
33
+ def method_missing(method, *args, &block)
34
+ @caller.__send__(method, *args, &block) || super
35
+ end
36
+
37
+ ##
38
+ # Checks respond_to of caller whenever something is not part of `bindings`.
39
+ # (No need to invoke this method manually)
40
+ def respond_to_missing?(*args)
41
+ @caller.respond_to?(*args) || super
42
+ end
43
+ end
44
+ end
45
+ end
@@ -0,0 +1,14 @@
1
+ module PropCheck
2
+ class Property
3
+ Configuration = Struct.new(:verbose, :n_runs, :max_generate_attempts, :max_shrink_steps, keyword_init: true) do
4
+
5
+ def initialize(verbose: false, n_runs: 1_000, max_generate_attempts: 10_000, max_shrink_steps: 10_000)
6
+ super
7
+ end
8
+
9
+ def merge(other)
10
+ Configuration.new(**self.to_h.merge(other.to_h))
11
+ end
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,14 @@
1
+ module PropCheck
2
+ ##
3
+ # Integration with RSpec
4
+ module RSpec
5
+ # To make it available within examples
6
+ def self.extend_object(obj)
7
+ obj.define_method(:forall) do |*args, **kwargs, &block|
8
+ PropCheck::Property.forall(*args, **kwargs) do
9
+ instance_exec(self, &block)
10
+ end
11
+ end
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,3 @@
1
+ module PropCheck
2
+ VERSION = "0.6.0"
3
+ end
@@ -0,0 +1,40 @@
1
+
2
+ lib = File.expand_path("../lib", __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require "prop_check/version"
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "prop_check"
8
+ spec.version = PropCheck::VERSION
9
+ spec.authors = ["Qqwy/Wiebe-Marten Wijnja"]
10
+ spec.email = ["w-m@wmcode.nl"]
11
+
12
+ spec.summary = %q{PropCheck allows you to do property-based testing , including shrinking. (akin to Haskell's QuickCheck, Erlang's PropEr, Elixir's StreamData)}
13
+ spec.description = %q{PropCheck allows you to do property-based testing , including shrinking. (akin to Haskell's QuickCheck, Erlang's PropEr, Elixir's StreamData). This means that your test are run many times with different, autogenerated inputs, and as soon as a failing case is found, this input is simplified, in the end giving you back the simplest input that made the test fail.}
14
+ spec.homepage = "https://github.com/Qqwy/ruby-prop_check/"
15
+ spec.license = "MIT"
16
+
17
+ # Prevent pushing this gem to RubyGems.org. To allow pushes either set the 'allowed_push_host'
18
+ # to allow pushing to a single host or delete this section to allow pushing to any host.
19
+ if spec.respond_to?(:metadata)
20
+ spec.metadata["homepage_uri"] = spec.homepage
21
+ spec.metadata["source_code_uri"] = "https://github.com/Qqwy/ruby-prop_check/"
22
+ spec.metadata["changelog_uri"] = "https://github.com/Qqwy/ruby-prop_check/CHANGELOG.md"
23
+ else
24
+ raise "RubyGems 2.0 or newer is required to protect against " \
25
+ "public gem pushes."
26
+ end
27
+
28
+ # Specify which files should be added to the gem when it is released.
29
+ # The `git ls-files -z` loads the files in the RubyGem that have been added into git.
30
+ spec.files = Dir.chdir(File.expand_path('..', __FILE__)) do
31
+ `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
32
+ end
33
+ spec.bindir = "exe"
34
+ spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
35
+ spec.require_paths = ["lib"]
36
+
37
+ spec.add_development_dependency "bundler", "~> 2.0"
38
+ spec.add_development_dependency "rake", "~> 10.0"
39
+ spec.add_development_dependency "rspec", "~> 3.0"
40
+ end
metadata ADDED
@@ -0,0 +1,119 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: prop_check
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.6.0
5
+ platform: ruby
6
+ authors:
7
+ - Qqwy/Wiebe-Marten Wijnja
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2019-07-03 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: bundler
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '2.0'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '2.0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '10.0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '10.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rspec
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '3.0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '3.0'
55
+ description: PropCheck allows you to do property-based testing , including shrinking.
56
+ (akin to Haskell's QuickCheck, Erlang's PropEr, Elixir's StreamData). This means
57
+ that your test are run many times with different, autogenerated inputs, and as soon
58
+ as a failing case is found, this input is simplified, in the end giving you back
59
+ the simplest input that made the test fail.
60
+ email:
61
+ - w-m@wmcode.nl
62
+ executables: []
63
+ extensions: []
64
+ extra_rdoc_files: []
65
+ files:
66
+ - ".gitignore"
67
+ - ".rspec"
68
+ - ".rubocop.yml"
69
+ - ".tool-versions"
70
+ - ".travis.yml"
71
+ - CHANGELOG.md
72
+ - CODE_OF_CONDUCT.md
73
+ - Gemfile
74
+ - Gemfile.lock
75
+ - LICENSE.txt
76
+ - README.md
77
+ - Rakefile
78
+ - bin/console
79
+ - bin/setup
80
+ - lib/prop_check.rb
81
+ - lib/prop_check/generator.rb
82
+ - lib/prop_check/generators.rb
83
+ - lib/prop_check/helper.rb
84
+ - lib/prop_check/lazy_tree.rb
85
+ - lib/prop_check/property.rb
86
+ - lib/prop_check/property/check_evaluator.rb
87
+ - lib/prop_check/property/configuration.rb
88
+ - lib/prop_check/rspec.rb
89
+ - lib/prop_check/version.rb
90
+ - prop_check.gemspec
91
+ homepage: https://github.com/Qqwy/ruby-prop_check/
92
+ licenses:
93
+ - MIT
94
+ metadata:
95
+ homepage_uri: https://github.com/Qqwy/ruby-prop_check/
96
+ source_code_uri: https://github.com/Qqwy/ruby-prop_check/
97
+ changelog_uri: https://github.com/Qqwy/ruby-prop_check/CHANGELOG.md
98
+ post_install_message:
99
+ rdoc_options: []
100
+ require_paths:
101
+ - lib
102
+ required_ruby_version: !ruby/object:Gem::Requirement
103
+ requirements:
104
+ - - ">="
105
+ - !ruby/object:Gem::Version
106
+ version: '0'
107
+ required_rubygems_version: !ruby/object:Gem::Requirement
108
+ requirements:
109
+ - - ">="
110
+ - !ruby/object:Gem::Version
111
+ version: '0'
112
+ requirements: []
113
+ rubyforge_project:
114
+ rubygems_version: 2.7.6
115
+ signing_key:
116
+ specification_version: 4
117
+ summary: PropCheck allows you to do property-based testing , including shrinking.
118
+ (akin to Haskell's QuickCheck, Erlang's PropEr, Elixir's StreamData)
119
+ test_files: []