forked from kaptinlin/jsonschema
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpropertyNames.go
62 lines (53 loc) · 2.57 KB
/
propertyNames.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
package jsonschema
import (
"fmt"
"strings"
)
// EvaluatePropertyNames checks if every property name in the object conforms to the schema specified by the propertyNames attribute.
// According to the JSON Schema Draft 2020-12:
// - The "propertyNames" keyword must be a valid JSON Schema.
// - If the instance is an object, this keyword validates if every property name in the instance validates against the provided schema.
// - The property name that the schema is testing will always be a string.
// - Omitting this keyword has the same behavior as an empty schema.
//
// This method ensures that each property name in the object instance conforms to the constraints defined in the propertyNames schema.
// If a property name does not conform, it returns a EvaluationError detailing the issue with that specific property name.
//
// Reference: https://json-schema.org/draft/2020-12/json-schema-core#name-propertynames
func evaluatePropertyNames(schema *Schema, object map[string]interface{}, evaluatedProps map[string]bool, evaluatedItems map[int]bool, dynamicScope *DynamicScope) ([]*EvaluationResult, *EvaluationError) {
if schema.PropertyNames == nil {
// No propertyNames schema defined, equivalent to an empty schema, which means all property names are valid.
return nil, nil
}
invalid_properties := []string{}
results := []*EvaluationResult{}
if schema.PropertyNames != nil {
for propName := range object {
result, _, _ := schema.PropertyNames.evaluate(propName, dynamicScope)
if result != nil {
//nolint:errcheck
result.SetEvaluationPath(fmt.Sprintf("/propertyNames/%s", propName)).
SetSchemaLocation(schema.GetSchemaLocation(fmt.Sprintf("/propertyNames/%s", propName))).
SetInstanceLocation(fmt.Sprintf("/%s", propName))
}
results = append(results, result)
if !result.IsValid() {
invalid_properties = append(invalid_properties, propName)
}
}
}
if len(invalid_properties) == 1 {
return results, NewEvaluationError("propertyNames", "property_name_mismatch", "Property name {property} does not match the schema", map[string]interface{}{
"property": fmt.Sprintf("'%s'", invalid_properties[0]),
})
} else if len(invalid_properties) > 1 {
quotedProperties := make([]string, len(invalid_properties))
for i, prop := range invalid_properties {
quotedProperties[i] = fmt.Sprintf("'%s'", prop)
}
return results, NewEvaluationError("propertyNames", "property_names_mismatch", "Property names {properties} do not match the schema", map[string]interface{}{
"properties": strings.Join(quotedProperties, ", "),
})
}
return results, nil
}