-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathschema_type.go
143 lines (123 loc) · 3.14 KB
/
schema_type.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
package jsonschema
import (
"fmt"
"reflect"
)
type SchemaType string
const (
SCHEMA_TYPE_STRING SchemaType = "string"
SCHEMA_TYPE_NUMBER SchemaType = "number"
SCHEMA_TYPE_NULL SchemaType = "null"
SCHEMA_TYPE_INTEGER SchemaType = "integer"
SCHEMA_TYPE_BOOLEAN SchemaType = "boolean"
SCHEMA_TYPE_ARRAY SchemaType = "array"
SCHEMA_TYPE_OBJECT SchemaType = "object"
)
func (self SchemaType) Valid() bool {
switch self {
case SCHEMA_TYPE_ARRAY:
fallthrough
case SCHEMA_TYPE_BOOLEAN:
fallthrough
case SCHEMA_TYPE_INTEGER:
fallthrough
case SCHEMA_TYPE_NULL:
fallthrough
case SCHEMA_TYPE_NUMBER:
fallthrough
case SCHEMA_TYPE_OBJECT:
fallthrough
case SCHEMA_TYPE_STRING:
return true
}
return false
}
func (self SchemaType) Match(value reflect.Value) bool {
kind := value.Kind()
switch self {
case SCHEMA_TYPE_ARRAY:
return kind == reflect.Array || kind == reflect.Slice
case SCHEMA_TYPE_BOOLEAN:
return kind == reflect.Bool
case SCHEMA_TYPE_INTEGER:
return kind == reflect.Int || (value.CanFloat() && float64(int64(value.Float())) == value.Float())
case SCHEMA_TYPE_NUMBER:
return kind == reflect.Int || kind == reflect.Float32 || kind == reflect.Float64
case SCHEMA_TYPE_OBJECT:
return kind == reflect.Map || kind == reflect.Struct
case SCHEMA_TYPE_STRING:
return kind == reflect.String
}
return kind == reflect.Invalid
}
// https://json-schema.org/understanding-json-schema/reference/type
func schemaType(key string) Keyword {
return Keyword{
Compile: func(ns *Namespace, ctx Context, config reflect.Value) []SchemaError {
errs := []SchemaError{}
switch config.Kind() {
case reflect.String:
if !SchemaType(config.String()).Valid() {
errs = append(errs, SchemaError{
Path: ctx.Path,
Keyword: key,
Message: `must be a valid "SchemaType"`,
})
}
break
case reflect.Array:
fallthrough
case reflect.Slice:
for i := 0; i < config.Len(); i++ {
index := config.Index(i).Elem()
if index.Kind() != reflect.String || !SchemaType(index.String()).Valid() {
errs = append(errs, SchemaError{
Path: fmt.Sprintf("%s/%s/%d", ctx.Path, key, i),
Keyword: key,
Message: `must be a valid "SchemaType"`,
})
break
}
}
break
default:
errs = append(errs, SchemaError{
Path: ctx.Path,
Keyword: key,
Message: `must be a "string" or "[]string"`,
})
}
return errs
},
Validate: func(ns *Namespace, ctx Context, config reflect.Value, value reflect.Value) []SchemaError {
errs := []SchemaError{}
types := []string{}
if !value.IsValid() {
return errs
}
if config.Kind() == reflect.String {
types = append(types, config.String())
} else {
for i := 0; i < config.Len(); i++ {
index := config.Index(i).Elem()
types = append(types, index.String())
}
}
for _, t := range types {
if SchemaType(t).Match(value) {
return errs
}
}
errs = append(errs, SchemaError{
Path: ctx.Path,
Keyword: key,
Message: fmt.Sprintf(
`"%s" should be one of %v`,
value.Kind().String(),
types,
),
})
return errs
},
}
}