forked from grafana/dashboard-spec
-
Notifications
You must be signed in to change notification settings - Fork 0
/
spec.go
149 lines (136 loc) · 4.01 KB
/
spec.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
144
145
146
147
148
149
package main
import (
"encoding/json"
"log"
"sort"
)
// OpenAPI 3.0 spec document.
type Spec struct {
Version string `json:"openapi"`
Info struct {
Title string
Version string
}
Components struct {
Schemas map[string]Schema
}
}
// OpenAPI 3.0 schema.
type Schema struct {
AllOf []*Schema
Default interface{}
Description string
Items *Schema
Properties map[string]*Schema
ReadOnly bool
Required []string
Title string
Type string
}
// Used for the purpose of flattening the properties of a schema. The location
// field makes it possible to reconstruct later. This facilitates generating
// setter/appender methods for deeply nested properties.
type FlatSchema struct {
Name string
Location []string
Schema *Schema
}
// Return a schema's default value as JSON.
func (s Schema) DefaultJSON() string {
b, err := json.Marshal(s.Default)
if err != nil {
log.Fatalln(err)
}
return string(b)
}
// If title is set, it's assumed it carries more meaning than the property name
// itself. And therefore more suitable for humans. This is useful for naming
// arguments and functions.
func (s Schema) HumanName(name string) string {
if s.Title != "" {
return s.Title
} else {
return name
}
}
// Combined schemas from Properties and AllOf.
func (s Schema) AllProperties() map[string]*Schema {
ap := s.Properties
if ap == nil {
ap = map[string]*Schema{}
}
for _, aos := range s.AllOf {
for n, subsc := range aos.Properties {
ap[n] = subsc
}
}
return ap
}
// Returns all top-level properties except objects and arrays of objects. These
// are intended to be used as arguments for the schema object's constructor.
func (s Schema) TopLevelSimpleProperties() map[string]*Schema {
p := map[string]*Schema{}
for n, s := range s.AllProperties() {
if !s.ReadOnly && s.Type != "object" &&
(s.Type != "array" || s.Type == "array" && s.Items != nil && s.Items.Type != "object") {
p[n] = s
}
}
return p
}
// Returns all properties that are readOnly and have a default property. It's
// intended that these are set, but not explicitly configurable. For example, a
// panel's "type" field.
func (s Schema) ReadOnlyWithDefaultProperties() []FlatSchema {
return flatten(&s, func(s *Schema) bool {
return s.ReadOnly && s.Default != nil
})
}
// Returns all top-level object properties. It's anticipated that these have
// setter methods nested inside their parent schema object.
func (s Schema) TopLevelObjectProperties() map[string]*Schema {
p := map[string]*Schema{}
for n, s := range s.AllProperties() {
if !s.ReadOnly && s.Type == "object" {
p[n] = s
}
}
return p
}
// Returns all nested properties except arrays of objects. It's anticipated
// that the parent schema object is a top-level object property and that the
// properties returned here will be arguments in the parent's setter method.
func (s Schema) NestedSimpleProperties() []FlatSchema {
return flatten(&s, func(s *Schema) bool {
return !s.ReadOnly && s.Type != "object" &&
(s.Type != "array" || s.Type == "array" && s.Items != nil && s.Items.Type != "object")
})
}
// Returns nested properties that are arrays of objects. It's anticipated that
// these are used to create appender methods for constructing those objects and
// appending them.
func (s Schema) NestedComplexArrayProperties() []FlatSchema {
return flatten(&s, func(s *Schema) bool {
return !s.ReadOnly && s.Type == "array" && s.Items != nil && s.Items.Type == "object"
})
}
// Recursively flattens nested properties.
func flatten(s *Schema, filter func(*Schema) bool) (fs []FlatSchema) {
var flatten func(*Schema, []string)
flatten = func(s *Schema, locationPrefix []string) {
for n, s := range s.AllProperties() {
if filter(s) {
fs = append(fs, FlatSchema{
Name: n,
Location: append(locationPrefix, n),
Schema: s,
})
} else if s.Type == "object" {
flatten(s, append(locationPrefix, n))
}
}
}
flatten(s, []string{})
sort.SliceStable(fs, func(i, j int) bool { return fs[i].Name < fs[j].Name })
return fs
}