json-schema 0.1.13 → 0.1.14
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.
- data/README.textile +105 -11
- data/lib/json-schema.rb +2 -0
- data/lib/json-schema/attributes/additionalitems.rb +23 -0
- data/lib/json-schema/attributes/additionalproperties.rb +39 -0
- data/lib/json-schema/attributes/dependencies.rb +28 -0
- data/lib/json-schema/attributes/disallow.rb +11 -0
- data/lib/json-schema/attributes/divisibleby.rb +16 -0
- data/lib/json-schema/attributes/enum.rb +24 -0
- data/lib/json-schema/attributes/extends.rb +14 -0
- data/lib/json-schema/attributes/format.rb +105 -0
- data/lib/json-schema/attributes/items.rb +25 -0
- data/lib/json-schema/attributes/maximum.rb +15 -0
- data/lib/json-schema/attributes/maxitems.rb +12 -0
- data/lib/json-schema/attributes/maxlength.rb +14 -0
- data/lib/json-schema/attributes/minimum.rb +15 -0
- data/lib/json-schema/attributes/minitems.rb +12 -0
- data/lib/json-schema/attributes/minlength.rb +14 -0
- data/lib/json-schema/attributes/pattern.rb +15 -0
- data/lib/json-schema/attributes/patternproperties.rb +23 -0
- data/lib/json-schema/attributes/properties.rb +23 -0
- data/lib/json-schema/attributes/ref.rb +53 -0
- data/lib/json-schema/attributes/type.rb +71 -0
- data/lib/json-schema/attributes/uniqueitems.rb +16 -0
- data/lib/json-schema/schema.rb +19 -2
- data/lib/json-schema/validator.rb +82 -578
- data/lib/json-schema/validators/draft3.rb +38 -0
- data/test/test_extended_schema.rb +68 -0
- data/test/{test_jsonschema.rb → test_jsonschema_draft3.rb} +59 -5
- metadata +30 -6
data/README.textile
CHANGED
@@ -2,27 +2,40 @@ h1. Ruby JSON Schema Validator
|
|
2
2
|
|
3
3
|
This library is intended to provide Ruby with an interface for validating JSON objects against a JSON schema conforming to "JSON Schema Draft 3":http://tools.ietf.org/html/draft-zyp-json-schema-03.
|
4
4
|
|
5
|
-
h2.
|
5
|
+
h2. Dependencies
|
6
6
|
|
7
|
-
|
7
|
+
The JSON::Schema library depends on the ruby JSON gem (json). Support for YAJL and other Ruby parsers is planned.
|
8
|
+
|
9
|
+
h2. Installation
|
10
|
+
|
11
|
+
From rubygems.org:
|
8
12
|
|
9
13
|
<pre>
|
10
14
|
gem install json-schema
|
11
15
|
</pre>
|
12
16
|
|
13
|
-
|
17
|
+
From the git repo:
|
14
18
|
|
15
19
|
<pre>
|
16
20
|
$ gem build json-schema.gemspec
|
17
|
-
$ gem install json-schema-0.1.
|
21
|
+
$ gem install json-schema-0.1.14.gem
|
18
22
|
</pre>
|
19
23
|
|
24
|
+
|
25
|
+
h2. Basic Usage
|
26
|
+
|
27
|
+
Two base validation methods exist: <code>validate</code> and <code>validate!</code>. The first returns a boolean on whether a validation attempt passes and the latter will throw a <code>JSON::Schema::ValidationError</code> with an appropriate message/trace on where the validation failed.
|
28
|
+
|
29
|
+
Both methods take two arguments, which can be either a JSON string, a file containing JSON, or a Ruby object representing JSON data. The first argument to these methods is always the schema, the second is always the data to validate.
|
30
|
+
|
31
|
+
By default, the validator uses (and only supports) the "JSON Schema Draft 3":http://tools.ietf.org/html/draft-zyp-json-schema-03 specification for validation; however, the user is free to specify additional specifications or extend existing ones.
|
32
|
+
|
33
|
+
h3. Validate Ruby objects against a Ruby schema
|
34
|
+
|
20
35
|
<pre>
|
21
36
|
require 'rubygems'
|
22
37
|
require 'json-schema'
|
23
38
|
|
24
|
-
JSON::Validator.validate('schema.json', 'data.json')
|
25
|
-
|
26
39
|
schema = {
|
27
40
|
"type" => "object",
|
28
41
|
"properties" => {
|
@@ -35,20 +48,102 @@ data = {
|
|
35
48
|
}
|
36
49
|
|
37
50
|
JSON::Validator.validate(schema, data)
|
51
|
+
</pre>
|
52
|
+
|
53
|
+
h3. Validate a JSON string against a JSON schema file
|
54
|
+
|
55
|
+
<pre>
|
56
|
+
require 'rubygems'
|
57
|
+
require 'json-schema'
|
58
|
+
|
59
|
+
JSON::Validator.validate('schema.json', "{'a' : 5}")
|
60
|
+
</pre>
|
61
|
+
|
62
|
+
h3. Validate a list of objects against a schema that represents the individual objects
|
63
|
+
|
64
|
+
<pre>
|
65
|
+
require 'rubygems'
|
66
|
+
require 'json-schema'
|
67
|
+
|
68
|
+
data = ['user','user','user']
|
69
|
+
JSON::Validator.validate('user.json', data, :list => true)
|
70
|
+
</pre>
|
38
71
|
|
39
|
-
|
40
|
-
JSON::Validator.validate(schema, data, :list => true)
|
72
|
+
h3. Catch a validation error and print it out
|
41
73
|
|
74
|
+
<pre>
|
75
|
+
require 'rubygems'
|
76
|
+
require 'json-schema'
|
77
|
+
|
78
|
+
schema = {
|
79
|
+
"type" => "object",
|
80
|
+
"properties" => {
|
81
|
+
"a" => {"type" => "integer", "required" => true}
|
82
|
+
}
|
83
|
+
}
|
84
|
+
|
42
85
|
data = {
|
43
86
|
"a" => "taco"
|
44
87
|
}
|
45
88
|
|
46
89
|
begin
|
47
|
-
JSON::Validator.
|
48
|
-
rescue ValidationError
|
90
|
+
JSON::Validator.validate!(schema, data)
|
91
|
+
rescue JSON::Schema::ValidationError
|
49
92
|
puts $!.message
|
50
93
|
end
|
51
94
|
</pre>
|
95
|
+
|
96
|
+
h3. Extend an existing schema and validating against it
|
97
|
+
|
98
|
+
For this example, we are going to extend the "JSON Schema Draft 3":http://tools.ietf.org/html/draft-zyp-json-schema-03 specification by adding a 'bitwise-and' property for validation.
|
99
|
+
|
100
|
+
<pre>
|
101
|
+
require 'rubygems'
|
102
|
+
require 'json-schema'
|
103
|
+
|
104
|
+
class BitwiseAndAttribute < JSON::Schema::Attribute
|
105
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
106
|
+
if data.is_a?(Integer) && data & current_schema.schema['bitwise-and'].to_i == 0
|
107
|
+
message = "The property '#{build_fragment(fragments)}' did not evaluate to true when bitwise-AND'd with #{current_schema.schema['bitwise-or']}"
|
108
|
+
raise JSON::Schema::ValidationError.new(message, fragments, current_schema)
|
109
|
+
end
|
110
|
+
end
|
111
|
+
end
|
112
|
+
|
113
|
+
class ExtendedSchema < JSON::Schema::Validator
|
114
|
+
def initialize
|
115
|
+
super
|
116
|
+
extend_schema_definition("http://json-schema.org/draft-03/schema#")
|
117
|
+
@attributes["bitwise-and"] = BitwiseAndAttribute
|
118
|
+
@uri = URI.parse("http://test.com/test.json")
|
119
|
+
end
|
120
|
+
|
121
|
+
JSON::Validator.register_validator(self.new)
|
122
|
+
end
|
123
|
+
|
124
|
+
schema = {
|
125
|
+
"$schema" => "http://test.com/test.json",
|
126
|
+
"properties" => {
|
127
|
+
"a" => {
|
128
|
+
"bitwise-and" => 1
|
129
|
+
},
|
130
|
+
"b" => {
|
131
|
+
"type" => "string"
|
132
|
+
}
|
133
|
+
}
|
134
|
+
}
|
135
|
+
|
136
|
+
data = {
|
137
|
+
"a" => 0
|
138
|
+
}
|
139
|
+
|
140
|
+
data = {"a" => 1, "b" => "taco"}
|
141
|
+
JSON::Validator.validate(schema,data) # => true
|
142
|
+
data = {"a" => 1, "b" => 5}
|
143
|
+
JSON::Validator.validate(schema,data) # => false
|
144
|
+
data = {"a" => 0, "b" => "taco"}
|
145
|
+
JSON::Validator.validate(schema,data) # => false
|
146
|
+
</pre>
|
52
147
|
|
53
148
|
|
54
149
|
h2. Notes
|
@@ -56,7 +151,6 @@ h2. Notes
|
|
56
151
|
The following core schema attributes are not implemented:
|
57
152
|
|
58
153
|
* default - This library aims to solely be a validator and does not modify an object it is validating.
|
59
|
-
* $schema - Support for this will come later, possibly with the ability to validate against other JSON Schema draft versions
|
60
154
|
|
61
155
|
The 'format' attribute is only validated for the following values:
|
62
156
|
|
data/lib/json-schema.rb
CHANGED
@@ -2,4 +2,6 @@ $LOAD_PATH.unshift "#{File.dirname(__FILE__)}/json-schema"
|
|
2
2
|
|
3
3
|
require 'schema'
|
4
4
|
require 'validator'
|
5
|
+
Dir[File.join(File.dirname(__FILE__), "json-schema/attributes/*")].each {|file| require file }
|
6
|
+
Dir[File.join(File.dirname(__FILE__), "json-schema/validators/*")].each {|file| require file }
|
5
7
|
require 'uri/file'
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class AdditionalItemsAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if data.is_a?(Array) && current_schema.schema['items'].is_a?(Array)
|
6
|
+
if current_schema.schema['additionalItems'] == false && current_schema.schema['items'].length != data.length
|
7
|
+
message = "The property '#{build_fragment(fragments)}' contains additional array elements outside of the schema when none are allowed"
|
8
|
+
raise ValidationError.new(message, fragments, current_schema)
|
9
|
+
elsif current_schema.schema['additionalItems'].is_a?(Hash)
|
10
|
+
schema = JSON::Schema.new(current_schema.schema['additionalItems'],current_schema.uri,validator)
|
11
|
+
data.each_with_index do |item,i|
|
12
|
+
if i >= current_schema.schema['items'].length
|
13
|
+
fragments << i.to_s
|
14
|
+
schema.validate(item, fragments)
|
15
|
+
fragments.pop
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class AdditionalPropertiesAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if data.is_a?(Hash)
|
6
|
+
extra_properties = data.keys
|
7
|
+
|
8
|
+
if current_schema.schema['properties']
|
9
|
+
extra_properties = extra_properties - current_schema.schema['properties'].keys
|
10
|
+
end
|
11
|
+
|
12
|
+
if current_schema.schema['patternProperties']
|
13
|
+
current_schema.schema['patternProperties'].each_key do |key|
|
14
|
+
r = Regexp.new(key)
|
15
|
+
extras_clone = extra_properties.clone
|
16
|
+
extras_clone.each do |prop|
|
17
|
+
if r.match(prop)
|
18
|
+
extra_properties = extra_properties - [prop]
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
if current_schema.schema['additionalProperties'] == false && !extra_properties.empty?
|
25
|
+
message = "The property '#{build_fragment(fragments)}' contains additional properties outside of the schema when none are allowed"
|
26
|
+
raise ValidationError.new(message, fragments, current_schema)
|
27
|
+
elsif current_schema.schema['additionalProperties'].is_a?(Hash)
|
28
|
+
extra_properties.each do |key|
|
29
|
+
schema = JSON::Schema.new(current_schema.schema['additionalProperties'],current_schema.uri,validator)
|
30
|
+
fragments << key
|
31
|
+
schema.validate(data[key],fragments)
|
32
|
+
fragments.pop
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class DependenciesAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if data.is_a?(Hash)
|
6
|
+
current_schema.schema['dependencies'].each do |property,dependency_value|
|
7
|
+
if data.has_key?(property)
|
8
|
+
if dependency_value.is_a?(String) && !data.has_key?(dependency_value)
|
9
|
+
message = "The property '#{build_fragment(fragments)}' has a property '#{property}' that depends on a missing property '#{dependency_value}'"
|
10
|
+
raise ValidationError.new(message, fragments, current_schema)
|
11
|
+
elsif dependency_value.is_a?(Array)
|
12
|
+
dependency_value.each do |value|
|
13
|
+
if !data.has_key?(value)
|
14
|
+
message = "The property '#{build_fragment(fragments)}' has a property '#{property}' that depends on a missing property '#{value}'"
|
15
|
+
raise ValidationError.new(message, fragments, current_schema)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
else
|
19
|
+
schema = JSON::Schema.new(dependency_value,current_schema.uri,validator)
|
20
|
+
schema.validate(data, fragments)
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,11 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class DisallowAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if validator.attributes['type']
|
6
|
+
validator.attributes['type'].validate(current_schema, data, fragments, validator, {:disallow => true})
|
7
|
+
end
|
8
|
+
end
|
9
|
+
end
|
10
|
+
end
|
11
|
+
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class DivisibleByAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if data.is_a?(Numeric)
|
6
|
+
if current_schema.schema['divisibleBy'] == 0 ||
|
7
|
+
current_schema.schema['divisibleBy'] == 0.0 ||
|
8
|
+
(BigDecimal.new(data.to_s) % BigDecimal.new(current_schema.schema['divisibleBy'].to_s)).to_f != 0
|
9
|
+
message = "The property '#{build_fragment(fragments)}' was not divisible by #{current_schema.schema['divisibleBy']}"
|
10
|
+
raise ValidationError.new(message, fragments, current_schema)
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class EnumAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if !current_schema.schema['enum'].include?(data)
|
6
|
+
message = "The property '#{build_fragment(fragments)}' did not match one of the following values:"
|
7
|
+
current_schema.schema['enum'].each {|val|
|
8
|
+
if val.is_a?(NilClass)
|
9
|
+
message += " null,"
|
10
|
+
elsif val.is_a?(Array)
|
11
|
+
message += " (array),"
|
12
|
+
elsif val.is_a?(Hash)
|
13
|
+
message += " (object),"
|
14
|
+
else
|
15
|
+
message += " #{val.to_s},"
|
16
|
+
end
|
17
|
+
}
|
18
|
+
message.chop!
|
19
|
+
raise ValidationError.new(message, fragments, current_schema)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,14 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class ExtendsAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
schemas = current_schema.schema['extends']
|
6
|
+
schemas = [schemas] if !schemas.is_a?(Array)
|
7
|
+
schemas.each do |s|
|
8
|
+
schema = JSON::Schema.new(s,current_schema.uri,validator)
|
9
|
+
schema.validate(data, fragments)
|
10
|
+
end
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
@@ -0,0 +1,105 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class FormatAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
case current_schema.schema['format']
|
6
|
+
|
7
|
+
# Timestamp in restricted ISO-8601 YYYY-MM-DDThh:mm:ssZ
|
8
|
+
when 'date-time'
|
9
|
+
error_message = "The property '#{build_fragment(fragments)}' must be a string and be a date/time in the ISO-8601 format of YYYY-MM-DDThh:mm:ssZ"
|
10
|
+
raise ValidationError.new(error_message, fragments, current_schema) if !data.is_a?(String)
|
11
|
+
r = Regexp.new('^\d\d\d\d-\d\d-\d\dT(\d\d):(\d\d):(\d\d)Z$')
|
12
|
+
if (m = r.match(data))
|
13
|
+
parts = data.split("T")
|
14
|
+
begin
|
15
|
+
Date.parse(parts[0])
|
16
|
+
rescue Exception
|
17
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
18
|
+
end
|
19
|
+
begin
|
20
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[1].to_i > 23
|
21
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[2].to_i > 59
|
22
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[3].to_i > 59
|
23
|
+
rescue Exception
|
24
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
25
|
+
end
|
26
|
+
else
|
27
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
28
|
+
end
|
29
|
+
|
30
|
+
# Date in the format of YYYY-MM-DD
|
31
|
+
when 'date'
|
32
|
+
error_message = "The property '#{build_fragment(fragments)}' must be a string and be a date in the format of YYYY-MM-DD"
|
33
|
+
raise ValidationError.new(error_message, fragments, current_schema) if !data.is_a?(String)
|
34
|
+
r = Regexp.new('^\d\d\d\d-\d\d-\d\d$')
|
35
|
+
if (m = r.match(data))
|
36
|
+
begin
|
37
|
+
Date.parse(data)
|
38
|
+
rescue Exception
|
39
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
40
|
+
end
|
41
|
+
else
|
42
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
43
|
+
end
|
44
|
+
|
45
|
+
# Time in the format of HH:MM:SS
|
46
|
+
when 'time'
|
47
|
+
error_message = "The property '#{build_fragment(fragments)}' must be a string and be a time in the format of hh:mm:ss"
|
48
|
+
raise ValidationError.new(error_message, fragments, current_schema) if !data.is_a?(String)
|
49
|
+
r = Regexp.new('^(\d\d):(\d\d):(\d\d)$')
|
50
|
+
if (m = r.match(data))
|
51
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[1].to_i > 23
|
52
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[2].to_i > 59
|
53
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[3].to_i > 59
|
54
|
+
else
|
55
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
56
|
+
end
|
57
|
+
|
58
|
+
# IPv4 in dotted-quad format
|
59
|
+
when 'ip-address', 'ipv4'
|
60
|
+
error_message = "The property '#{build_fragment(fragments)}' must be a string and be a valid IPv4 address"
|
61
|
+
raise ValidationError.new(error_message, fragments, current_schema) if !data.is_a?(String)
|
62
|
+
r = Regexp.new('^(\d+){1,3}\.(\d+){1,3}\.(\d+){1,3}\.(\d+){1,3}$')
|
63
|
+
if (m = r.match(data))
|
64
|
+
1.upto(4) do |x|
|
65
|
+
raise ValidationError.new(error_message, fragments, current_schema) if m[x].to_i > 255
|
66
|
+
end
|
67
|
+
else
|
68
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
69
|
+
end
|
70
|
+
|
71
|
+
# IPv6 in standard format (including abbreviations)
|
72
|
+
when 'ipv6'
|
73
|
+
error_message = "The property '#{build_fragment(fragments)}' must be a string and be a valid IPv6 address"
|
74
|
+
raise ValidationError.new(error_message, fragments, current_schema) if !data.is_a?(String)
|
75
|
+
r = Regexp.new('^[a-f0-9:]+$')
|
76
|
+
if (m = r.match(data))
|
77
|
+
# All characters are valid, now validate structure
|
78
|
+
parts = data.split(":")
|
79
|
+
raise ValidationError.new(error_message, fragments, current_schema) if parts.length > 8
|
80
|
+
condensed_zeros = false
|
81
|
+
parts.each do |part|
|
82
|
+
if part.length == 0
|
83
|
+
raise ValidationError.new(error_message, fragments, current_schema) if condensed_zeros
|
84
|
+
condensed_zeros = true
|
85
|
+
end
|
86
|
+
raise ValidationError.new(error_message, fragments, current_schema) if part.length > 4
|
87
|
+
end
|
88
|
+
else
|
89
|
+
raise ValidationError.new(error_message, fragments, current_schema)
|
90
|
+
end
|
91
|
+
|
92
|
+
# Milliseconds since the epoch. Must be an integer or a float
|
93
|
+
when 'utc-millisec'
|
94
|
+
error_message = "The property '#{build_fragment(fragments)}' must be an integer or a float"
|
95
|
+
raise ValidationError.new(error_message, fragments, current_schema) if (!data.is_a?(Numeric))
|
96
|
+
|
97
|
+
# Must be a string
|
98
|
+
when 'regex','color','style','phone','uri','email','host-name'
|
99
|
+
error_message = "The property '#{build_fragment(fragments)}' must be a string"
|
100
|
+
raise ValidationError.new(error_message, fragments, current_schema) if (!data.is_a?(String))
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
module JSON
|
2
|
+
class Schema
|
3
|
+
class ItemsAttribute < Attribute
|
4
|
+
def self.validate(current_schema, data, fragments, validator, options = {})
|
5
|
+
if data.is_a?(Array)
|
6
|
+
if current_schema.schema['items'].is_a?(Hash)
|
7
|
+
data.each_with_index do |item,i|
|
8
|
+
schema = JSON::Schema.new(current_schema.schema['items'],current_schema.uri,validator)
|
9
|
+
fragments << i.to_s
|
10
|
+
schema.validate(item,fragments)
|
11
|
+
fragments.pop
|
12
|
+
end
|
13
|
+
elsif current_schema.schema['items'].is_a?(Array)
|
14
|
+
current_schema.schema['items'].each_with_index do |item_schema,i|
|
15
|
+
schema = JSON::Schema.new(item_schema,current_schema.uri,validator)
|
16
|
+
fragments << i.to_s
|
17
|
+
schema.validate(data[i],fragments)
|
18
|
+
fragments.pop
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|