util.jsonschema: Rename types for improved readability

This commit is contained in:
Kim Alvefur 2021-03-18 23:57:03 +01:00
parent d18097fb67
commit c1706af956
3 changed files with 56 additions and 30 deletions

View file

@ -1,5 +1,18 @@
local record lib local record lib
encode : function (any) : string encode : function (any) : string
decode : function (string) : any, string decode : function (string) : any, string
enum json_type_name
"null"
"boolean"
"object"
"array"
"number"
"string"
"integer"
end
type null_type = (nil)
null : null_type
end end
return lib return lib

View file

@ -8,18 +8,19 @@
-- https://json-schema.org/draft/2020-12/json-schema-validation.html -- https://json-schema.org/draft/2020-12/json-schema-validation.html
-- --
local record schema_t local json = require"util.json"
enum type_e local null = json.null;
"null"
"boolean"
"object"
"array"
"number"
"string"
"integer"
end
type : type_e local type json_type_name = json.json_type_name
-- json_type_name here is non-standard
local type schema_t = boolean | json_type_name | json_schema_object
local record json_schema_object
type json_type_name = json.json_type_name
type schema_object = json_schema_object
type : json_type_name
enum : { any } enum : { any }
const : any const : any
@ -56,7 +57,7 @@ local record schema_t
minContains : integer minContains : integer
-- objects -- objects
properties : { string : schema_t | type_e } properties : { string : schema_t }
maxProperties : integer maxProperties : integer
minProperties : integer minProperties : integer
required : { string } required : { string }
@ -87,26 +88,30 @@ local record schema_t
deprecated : boolean deprecated : boolean
readOnly : boolean readOnly : boolean
writeOnly : boolean writeOnly : boolean
-- methods
validate : function ( schema_t, any) : boolean
end end
local type_e = schema_t.type_e
-- TODO validator function per schema property -- TODO validator function per schema property
local type_validators : { type_e : function (schema_t, any) : boolean } = {} local type_validators : { json_type_name : function (schema_t, any) : boolean } = {}
local function simple_validate(schema : type_e, data : any) : boolean local function simple_validate(schema : json_type_name, data : any) : boolean
if schema == "object" and data is table then if schema == "object" and data is table then
return type(data) == "table" and (next(data)==nil or type((next(data, nil))) == "string") return type(data) == "table" and (next(data)==nil or type((next(data, nil))) == "string")
elseif schema == "array" and data is table then elseif schema == "array" and data is table then
return type(data) == "table" and (next(data)==nil or type((next(data, nil))) == "number") return type(data) == "table" and (next(data)==nil or type((next(data, nil))) == "number")
elseif schema == "integer" then elseif schema == "integer" then
return math.type(data) == schema return math.type(data) == schema
elseif schema == "null" then
return data == null
else else
return type(data) == schema return type(data) == schema
end end
end end
type_validators.string = function (schema : schema_t, data : any) : boolean type_validators.string = function (schema : json_schema_object, data : any) : boolean
-- XXX this is measured in byte, while JSON measures in ... bork -- XXX this is measured in byte, while JSON measures in ... bork
-- TODO use utf8.len? -- TODO use utf8.len?
if data is string then if data is string then
@ -121,7 +126,7 @@ type_validators.string = function (schema : schema_t, data : any) : boolean
return false return false
end end
type_validators.number = function (schema : schema_t, data : number) : boolean type_validators.number = function (schema : json_schema_object, data : number) : boolean
if schema.multipleOf and data % schema.multipleOf ~= 0 then if schema.multipleOf and data % schema.multipleOf ~= 0 then
return false return false
end end
@ -147,14 +152,14 @@ end
type_validators.integer = type_validators.number type_validators.integer = type_validators.number
local function validate(schema : schema_t | type_e | boolean, data : any) : boolean local function validate(schema : schema_t, data : any) : boolean
if schema is boolean then if schema is boolean then
return schema return schema
end end
if schema is type_e then if schema is json_type_name then
return simple_validate(schema, data) return simple_validate(schema, data)
end end
if schema is schema_t then if schema is json_schema_object then
if schema.allOf then if schema.allOf then
for _, sub in ipairs(schema.allOf) do for _, sub in ipairs(schema.allOf) do
if not validate(sub, data) then if not validate(sub, data) then
@ -228,7 +233,7 @@ local function validate(schema : schema_t | type_e | boolean, data : any) : bool
end end
end end
type_validators.table = function (schema : schema_t, data : any) : boolean type_validators.table = function (schema : json_schema_object, data : any) : boolean
if data is table then if data is table then
if schema.maxItems and #data > schema.maxItems then if schema.maxItems and #data > schema.maxItems then
@ -248,12 +253,12 @@ type_validators.table = function (schema : schema_t, data : any) : boolean
end end
if schema.properties then if schema.properties then
local additional : schema_t | boolean = schema.additionalProperties or true local additional : schema_t = schema.additionalProperties or true
for k, v in pairs(data) do for k, v in pairs(data) do
if schema.propertyNames and not validate(schema.propertyNames, k) then if schema.propertyNames and not validate(schema.propertyNames, k) then
return false return false
end end
local s = schema.properties[k as string] or additional as schema_t local s = schema.properties[k as string] or additional
if not validate(s, v) then if not validate(s, v) then
return false return false
end end
@ -345,7 +350,6 @@ type_validators.array = function (schema : schema_t, data : any) : boolean
return false return false
end end
return { json_schema_object.validate = validate;
validate = validate;
schema_t = schema_t; return json_schema_object;
}

View file

@ -1,6 +1,11 @@
local schema_t = {xml_t = {}} local json = require("util.json")
local null = json.null;
local type_e = schema_t.type_e local json_type_name = json.json_type_name
local schema_t = {}
local json_schema_object = {xml_t = {}}
local type_validators = {} local type_validators = {}
@ -11,6 +16,8 @@ local function simple_validate(schema, data)
return type(data) == "table" and (next(data) == nil or type((next(data, nil))) == "number") return type(data) == "table" and (next(data) == nil or type((next(data, nil))) == "number")
elseif schema == "integer" then elseif schema == "integer" then
return math.type(data) == schema return math.type(data) == schema
elseif schema == "null" then
return data == null
else else
return type(data) == schema return type(data) == schema
end end
@ -253,4 +260,6 @@ type_validators.array = function(schema, data)
return false return false
end end
return {validate = validate; schema_t = schema_t} json_schema_object.validate = validate;
return json_schema_object