finitio 0.7.0.pre.rc4 → 0.10.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/CHANGELOG.md +42 -0
- data/Gemfile +1 -11
- data/Gemfile.lock +101 -54
- data/README.md +89 -13
- data/finitio.gemspec +9 -157
- data/lib/finitio.rb +7 -5
- data/lib/finitio/generation.rb +106 -0
- data/lib/finitio/generation/ad_type.rb +10 -0
- data/lib/finitio/generation/alias_type.rb +9 -0
- data/lib/finitio/generation/any_type.rb +11 -0
- data/lib/finitio/generation/builtin_type.rb +9 -0
- data/lib/finitio/generation/hash_based_type.rb +15 -0
- data/lib/finitio/generation/heuristic.rb +8 -0
- data/lib/finitio/generation/heuristic/constant.rb +30 -0
- data/lib/finitio/generation/heuristic/random.rb +52 -0
- data/lib/finitio/generation/rel_based_type.rb +13 -0
- data/lib/finitio/generation/seq_type.rb +13 -0
- data/lib/finitio/generation/set_type.rb +13 -0
- data/lib/finitio/generation/sub_type.rb +9 -0
- data/lib/finitio/generation/union_type.rb +10 -0
- data/lib/finitio/inference.rb +51 -0
- data/lib/finitio/json_schema.rb +16 -0
- data/lib/finitio/json_schema/ad_type.rb +11 -0
- data/lib/finitio/json_schema/alias_type.rb +9 -0
- data/lib/finitio/json_schema/any_type.rb +9 -0
- data/lib/finitio/json_schema/builtin_type.rb +27 -0
- data/lib/finitio/json_schema/hash_based_type.rb +25 -0
- data/lib/finitio/json_schema/rel_based_type.rb +13 -0
- data/lib/finitio/json_schema/seq_type.rb +12 -0
- data/lib/finitio/json_schema/set_type.rb +13 -0
- data/lib/finitio/json_schema/struct_type.rb +12 -0
- data/lib/finitio/json_schema/sub_type.rb +10 -0
- data/lib/finitio/json_schema/union_type.rb +11 -0
- data/lib/finitio/support.rb +18 -0
- data/lib/finitio/support/attribute.rb +8 -0
- data/lib/finitio/support/compilation.rb +18 -18
- data/lib/finitio/support/contract.rb +10 -2
- data/lib/finitio/support/fetch_scope.rb +19 -0
- data/lib/finitio/support/heading.rb +42 -1
- data/lib/finitio/syntax.rb +1 -1
- data/lib/finitio/syntax/lexer.citrus +1 -1
- data/lib/finitio/syntax/type.rb +2 -0
- data/lib/finitio/syntax/type/high_order_type_instantiation.rb +29 -0
- data/lib/finitio/syntax/type/high_order_vars.rb +16 -0
- data/lib/finitio/syntax/type/type_def.rb +11 -1
- data/lib/finitio/syntax/types.citrus +14 -1
- data/lib/finitio/system.rb +18 -4
- data/lib/finitio/type.rb +19 -0
- data/lib/finitio/type/ad_type.rb +9 -1
- data/lib/finitio/type/alias_type.rb +8 -0
- data/lib/finitio/type/any_type.rb +12 -0
- data/lib/finitio/type/builtin_type.rb +4 -0
- data/lib/finitio/type/collection_type.rb +15 -0
- data/lib/finitio/type/heading_based_type.rb +17 -0
- data/lib/finitio/type/high_order_type.rb +39 -0
- data/lib/finitio/type/multi_relation_type.rb +4 -0
- data/lib/finitio/type/multi_tuple_type.rb +4 -0
- data/lib/finitio/type/proxy_type.rb +10 -20
- data/lib/finitio/type/relation_type.rb +4 -0
- data/lib/finitio/type/seq_type.rb +1 -1
- data/lib/finitio/type/struct_type.rb +8 -0
- data/lib/finitio/type/sub_type.rb +8 -0
- data/lib/finitio/type/tuple_type.rb +4 -0
- data/lib/finitio/type/union_type.rb +19 -0
- data/lib/finitio/version.rb +2 -2
- data/spec/finitio/test_system.rb +0 -11
- data/spec/generation/test_generation.rb +169 -0
- data/spec/heading/test_looks_similar.rb +45 -0
- data/spec/heading/test_suppremum.rb +56 -0
- data/spec/inference/test_inference.rb +42 -0
- data/spec/json_schema/test_ad_type.rb +20 -0
- data/spec/json_schema/test_alias_type.rb +15 -0
- data/spec/json_schema/test_any_type.rb +11 -0
- data/spec/json_schema/test_builtin_type.rb +51 -0
- data/spec/json_schema/test_multi_relation_type.rb +58 -0
- data/spec/json_schema/test_multi_tuple_type.rb +50 -0
- data/spec/json_schema/test_relation_type.rb +30 -0
- data/spec/json_schema/test_seq_type.rb +18 -0
- data/spec/json_schema/test_set_type.rb +19 -0
- data/spec/json_schema/test_struct_type.rb +18 -0
- data/spec/json_schema/test_sub_type.rb +17 -0
- data/spec/json_schema/test_tuple_type.rb +26 -0
- data/spec/json_schema/test_union_type.rb +17 -0
- data/spec/regression/test_heading_extra_are_proxy_resolved.rb +41 -0
- data/spec/spec_helper.rb +32 -6
- data/spec/support/test_compare_attrs.rb +67 -0
- data/spec/syntax/test_compile.rb +57 -0
- data/spec/system/fixtures/system.fio +2 -0
- data/spec/{finitio → system/fixtures}/with-duplicates.fio +2 -1
- data/spec/system/test_check_and_warn.rb +55 -0
- data/spec/type/ad_type/test_initialize.rb +1 -8
- data/spec/type/relation_type/test_suppremum.rb +104 -0
- data/spec/type/seq_type/test_suppremum.rb +54 -0
- data/spec/type/set_type/test_suppremum.rb +54 -0
- data/spec/type/test_suppremum.rb +49 -0
- data/spec/type/test_unconstrained.rb +150 -0
- data/spec/type/tuple_type/test_suppremum.rb +119 -0
- data/spec/type/union_type/test_suppremum.rb +51 -0
- data/tasks/test.rake +1 -1
- metadata +203 -17
- data/spec/type/proxy_type/test_delegation.rb +0 -37
- data/spec/type/proxy_type/test_resolve.rb +0 -29
data/lib/finitio.rb
CHANGED
@@ -16,16 +16,18 @@ module Finitio
|
|
16
16
|
|
17
17
|
ANY_TYPE = AnyType.new
|
18
18
|
|
19
|
+
LOCK = Mutex.new
|
20
|
+
|
19
21
|
STDLIB_PATHS = [
|
20
22
|
File.expand_path('../finitio/stdlib', __FILE__)
|
21
23
|
]
|
22
24
|
|
23
25
|
MEMOIZED_SYSTEMS = {}
|
24
26
|
|
25
|
-
MEMOIZATION_SEMAPHORE = Mutex.new
|
26
|
-
|
27
27
|
def stdlib_path(path)
|
28
|
-
|
28
|
+
LOCK.synchronize {
|
29
|
+
STDLIB_PATHS << path
|
30
|
+
}
|
29
31
|
end
|
30
32
|
|
31
33
|
def parse(source)
|
@@ -33,7 +35,7 @@ module Finitio
|
|
33
35
|
end
|
34
36
|
|
35
37
|
def system(source)
|
36
|
-
|
38
|
+
LOCK.synchronize {
|
37
39
|
_system(source)
|
38
40
|
}
|
39
41
|
end
|
@@ -48,7 +50,7 @@ module Finitio
|
|
48
50
|
private :_system
|
49
51
|
|
50
52
|
def clear_saved_systems!
|
51
|
-
|
53
|
+
LOCK.synchronize {
|
52
54
|
MEMOIZED_SYSTEMS.clear
|
53
55
|
}
|
54
56
|
end
|
@@ -0,0 +1,106 @@
|
|
1
|
+
require 'securerandom'
|
2
|
+
require_relative 'generation/heuristic'
|
3
|
+
module Finitio
|
4
|
+
#
|
5
|
+
# This class helps generating random data for a given Finitio type.
|
6
|
+
#
|
7
|
+
# Example
|
8
|
+
#
|
9
|
+
# type = Finitio.system(<<-FIO)
|
10
|
+
# .Integer
|
11
|
+
# FIO
|
12
|
+
# gen = Finitio::Generation.new
|
13
|
+
# gen.call(type)
|
14
|
+
#
|
15
|
+
# Random data is generated for most ruby builtin types, tuples and all sorts
|
16
|
+
# of collections (seq, set, relations).
|
17
|
+
#
|
18
|
+
# You can fine-tune the random data generation through the following means
|
19
|
+
# (the first one wins):
|
20
|
+
#
|
21
|
+
# * By passing generators by type name at construction:
|
22
|
+
#
|
23
|
+
# Finition::Generation.new({
|
24
|
+
# generators: {
|
25
|
+
# "Name" => ->(type, g, world) { "Bernard" }
|
26
|
+
# }
|
27
|
+
# })
|
28
|
+
#
|
29
|
+
# * Through `examples` metadata put on type definitions:
|
30
|
+
#
|
31
|
+
# /- examples: {"Bernard"} -/
|
32
|
+
# Name = .String
|
33
|
+
#
|
34
|
+
class Generation
|
35
|
+
|
36
|
+
DEFAULT_OPTIONS = {
|
37
|
+
|
38
|
+
heuristic: Heuristic::Random.new,
|
39
|
+
|
40
|
+
collection_size: 1..10,
|
41
|
+
|
42
|
+
generators: {
|
43
|
+
"Date" => ->(t,g,w) { g.heuristic.call(Date, g, w) },
|
44
|
+
"Time" => ->(t,g,w) { g.heuristic.call(Time, g, w) },
|
45
|
+
"DateTime" => ->(t,g,w) { g.heuristic.call(DateTime, g, w) }
|
46
|
+
}
|
47
|
+
|
48
|
+
}
|
49
|
+
|
50
|
+
def initialize(options = {})
|
51
|
+
@options = DEFAULT_OPTIONS.merge(options){|k,v1,v2|
|
52
|
+
v1.is_a?(Hash) ? v1.merge(v2) : v2
|
53
|
+
}
|
54
|
+
end
|
55
|
+
attr_reader :options
|
56
|
+
|
57
|
+
def heuristic
|
58
|
+
options[:heuristic]
|
59
|
+
end
|
60
|
+
|
61
|
+
def generators
|
62
|
+
options[:generators]
|
63
|
+
end
|
64
|
+
|
65
|
+
def flip_coin
|
66
|
+
flip_one_out_of(2) == 1
|
67
|
+
end
|
68
|
+
|
69
|
+
def flip_one_out_of(n)
|
70
|
+
case n
|
71
|
+
when Integer then Kernel.rand(n)
|
72
|
+
when Range then Kernel.rand(n)
|
73
|
+
when Enumerable then n[flip_one_out_of(n.size)]
|
74
|
+
else
|
75
|
+
raise "Cannot flip one on #{n}"
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
def collection_size
|
80
|
+
size = options[:collection_size]
|
81
|
+
size = size.is_a?(Proc) ? size.call : size
|
82
|
+
flip_one_out_of(size)
|
83
|
+
end
|
84
|
+
|
85
|
+
def call(type, world = nil)
|
86
|
+
if gen = generators[type.name]
|
87
|
+
gen.call(type, self, world)
|
88
|
+
elsif exs = type.metadata && type.metadata[:examples]
|
89
|
+
flip_one_out_of(exs)
|
90
|
+
else
|
91
|
+
type.generate_data(self, world)
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
end # class Generation
|
96
|
+
end # module Finitio
|
97
|
+
require_relative 'generation/any_type'
|
98
|
+
require_relative 'generation/builtin_type'
|
99
|
+
require_relative 'generation/seq_type'
|
100
|
+
require_relative 'generation/set_type'
|
101
|
+
require_relative 'generation/hash_based_type'
|
102
|
+
require_relative 'generation/rel_based_type'
|
103
|
+
require_relative 'generation/union_type'
|
104
|
+
require_relative 'generation/alias_type'
|
105
|
+
require_relative 'generation/sub_type'
|
106
|
+
require_relative 'generation/ad_type'
|
@@ -0,0 +1,11 @@
|
|
1
|
+
module Finitio
|
2
|
+
class AnyType
|
3
|
+
|
4
|
+
def generate_data(generator, world = nil)
|
5
|
+
candidates = [NilClass, String, Integer, Float]
|
6
|
+
type = generator.flip_one_out_of(candidates)
|
7
|
+
generator.heuristic.call(type, generator, world)
|
8
|
+
end
|
9
|
+
|
10
|
+
end # class AnyType
|
11
|
+
end # module Finitio
|
@@ -0,0 +1,15 @@
|
|
1
|
+
module Finitio
|
2
|
+
module HashBasedType
|
3
|
+
|
4
|
+
def generate_data(generator, world = nil)
|
5
|
+
tuple = {}
|
6
|
+
heading.each do |a|
|
7
|
+
if a.required or generator.flip_coin
|
8
|
+
tuple[a.name.to_sym] = generator.call(a.type, world)
|
9
|
+
end
|
10
|
+
end
|
11
|
+
tuple
|
12
|
+
end
|
13
|
+
|
14
|
+
end # class HashBasedType
|
15
|
+
end # module Finitio
|
@@ -0,0 +1,30 @@
|
|
1
|
+
module Finitio
|
2
|
+
class Generation
|
3
|
+
class Heuristic
|
4
|
+
class Constant < Heuristic
|
5
|
+
|
6
|
+
CONSTANTS = {
|
7
|
+
NilClass => nil,
|
8
|
+
TrueClass => true,
|
9
|
+
FalseClass => false,
|
10
|
+
Integer => 99,
|
11
|
+
Float => 99.99,
|
12
|
+
String => "Hello world",
|
13
|
+
Date => Date.today,
|
14
|
+
Time => Time.now,
|
15
|
+
DateTime => DateTime.now,
|
16
|
+
}
|
17
|
+
|
18
|
+
def call(ruby_type, generator, world = nil)
|
19
|
+
CONSTANTS.fetch(ruby_type) do
|
20
|
+
CONSTANTS.each_pair do |clazz, value|
|
21
|
+
return value if clazz >= ruby_type
|
22
|
+
end
|
23
|
+
throw :unfound
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
end # class Constant
|
28
|
+
end # class Heuristic
|
29
|
+
end # class Generation
|
30
|
+
end # module Finition
|
@@ -0,0 +1,52 @@
|
|
1
|
+
module Finitio
|
2
|
+
class Generation
|
3
|
+
class Heuristic
|
4
|
+
class Random < Heuristic
|
5
|
+
|
6
|
+
RANDOMERS = {
|
7
|
+
NilClass => nil,
|
8
|
+
|
9
|
+
TrueClass => true,
|
10
|
+
|
11
|
+
FalseClass => false,
|
12
|
+
|
13
|
+
Integer => ->(_,g,_) {
|
14
|
+
g.flip_one_out_of(1_000_000)
|
15
|
+
},
|
16
|
+
|
17
|
+
Float => ->(_,g,_) {
|
18
|
+
g.flip_one_out_of(1_000_000)
|
19
|
+
},
|
20
|
+
|
21
|
+
String => ->(_,g,_) {
|
22
|
+
(1..3).map{ SecureRandom.hex(6) }.join(" ")
|
23
|
+
},
|
24
|
+
|
25
|
+
Date => ->(_,g,_) {
|
26
|
+
Time.at(rand * Time.now.to_i).to_date
|
27
|
+
},
|
28
|
+
|
29
|
+
Time => ->(_,g,_) {
|
30
|
+
Time.at(rand * Time.now.to_i)
|
31
|
+
},
|
32
|
+
|
33
|
+
DateTime => ->(_,g,_) {
|
34
|
+
Time.at(rand * Time.now.to_i).to_datetime
|
35
|
+
}
|
36
|
+
}
|
37
|
+
|
38
|
+
def call(ruby_type, generator, world = nil)
|
39
|
+
r = RANDOMERS.fetch(ruby_type) do
|
40
|
+
pair = RANDOMERS.find do |clazz, value|
|
41
|
+
clazz >= ruby_type
|
42
|
+
end
|
43
|
+
throw :unfound unless pair
|
44
|
+
pair.last
|
45
|
+
end
|
46
|
+
r.is_a?(Proc) ? r.call(ruby_type, generator, world) : r
|
47
|
+
end
|
48
|
+
|
49
|
+
end # class Random
|
50
|
+
end # class Heuristic
|
51
|
+
end # class Generation
|
52
|
+
end # module Finition
|
@@ -0,0 +1,13 @@
|
|
1
|
+
module Finitio
|
2
|
+
module RelBasedType
|
3
|
+
|
4
|
+
def generate_data(generator, world = nil)
|
5
|
+
coll = []
|
6
|
+
generator.collection_size.times do
|
7
|
+
coll << generator.call(tuple_type, world)
|
8
|
+
end
|
9
|
+
coll.uniq
|
10
|
+
end
|
11
|
+
|
12
|
+
end # module RelBasedType
|
13
|
+
end # module Finitio
|
@@ -0,0 +1,51 @@
|
|
1
|
+
module Finitio
|
2
|
+
class Inference
|
3
|
+
|
4
|
+
def initialize(system, options = {})
|
5
|
+
@system = system
|
6
|
+
@options = options
|
7
|
+
end
|
8
|
+
attr_reader :system
|
9
|
+
|
10
|
+
def call(input)
|
11
|
+
infer_type(input)
|
12
|
+
end
|
13
|
+
|
14
|
+
private
|
15
|
+
|
16
|
+
def infer_type(value)
|
17
|
+
case value
|
18
|
+
when Hash
|
19
|
+
attrs = value.map{|k,v|
|
20
|
+
Attribute.new(k.to_sym, infer_type(v))
|
21
|
+
}
|
22
|
+
heading = Heading.new(attrs)
|
23
|
+
TupleType.new(heading)
|
24
|
+
when Array
|
25
|
+
infered = value.inject(nil){|sup, value|
|
26
|
+
value_type = infer_type(value)
|
27
|
+
sup.nil? ? value_type : sup.suppremum(value_type)
|
28
|
+
}
|
29
|
+
SeqType.new(infered.nil? ? ANY_TYPE : infered)
|
30
|
+
else
|
31
|
+
found = self.system.types.values.find{|t|
|
32
|
+
try_dress(t, value)
|
33
|
+
}
|
34
|
+
found ? found : ANY_TYPE
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
def try_dress(t, value)
|
39
|
+
raise "Type expected, got #{t}" unless t.is_a?(Type)
|
40
|
+
t.dress(value)
|
41
|
+
true
|
42
|
+
rescue Finitio::Error => ex
|
43
|
+
#puts %Q{[#{ex.class}] #{ex.message}}
|
44
|
+
nil
|
45
|
+
rescue => ex
|
46
|
+
puts %Q{[#{ex.class}] #{ex.message}\n#{ex.backtrace.join("\n")}}
|
47
|
+
nil
|
48
|
+
end
|
49
|
+
|
50
|
+
end # class Inference
|
51
|
+
end # module Finitio
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module Finitio
|
2
|
+
module JsonSchema
|
3
|
+
class Error < Finitio::Error; end
|
4
|
+
end # module JsonSchema
|
5
|
+
end # module Finitio
|
6
|
+
require_relative 'json_schema/ad_type'
|
7
|
+
require_relative 'json_schema/any_type'
|
8
|
+
require_relative 'json_schema/alias_type'
|
9
|
+
require_relative 'json_schema/builtin_type'
|
10
|
+
require_relative 'json_schema/hash_based_type'
|
11
|
+
require_relative 'json_schema/rel_based_type'
|
12
|
+
require_relative 'json_schema/seq_type'
|
13
|
+
require_relative 'json_schema/set_type'
|
14
|
+
require_relative 'json_schema/struct_type'
|
15
|
+
require_relative 'json_schema/sub_type'
|
16
|
+
require_relative 'json_schema/union_type'
|