package spec import ( "bytes" "encoding/json" "reflect" "sort" ) type OrderSchemaItem struct { Name string Schema } type OrderSchemaItems []OrderSchemaItem func (items OrderSchemaItems) MarshalJSON() ([]byte, error) { buf := bytes.NewBuffer(nil) buf.WriteString("{") for i := range items { if i > 0 { buf.WriteString(",") } buf.WriteString("\"") buf.WriteString(items[i].Name) buf.WriteString("\":") bs, err := json.Marshal(&items[i].Schema) if err != nil { return nil, err } buf.Write(bs) } buf.WriteString("}") return buf.Bytes(), nil } func (items OrderSchemaItems) Len() int { return len(items) } func (items OrderSchemaItems) Swap(i, j int) { items[i], items[j] = items[j], items[i] } func (items OrderSchemaItems) Less(i, j int) (ret bool) { ii, oki := items[i].Extensions.GetString("x-order") ij, okj := items[j].Extensions.GetString("x-order") if oki { if okj { defer func() { if err := recover(); err != nil { defer func() { if err = recover(); err != nil { ret = items[i].Name < items[j].Name } }() ret = reflect.ValueOf(ii).String() < reflect.ValueOf(ij).String() } }() return reflect.ValueOf(ii).Int() < reflect.ValueOf(ij).Int() } return true } else if okj { return false } return items[i].Name < items[j].Name } type SchemaProperties map[string]Schema func (properties SchemaProperties) ToOrderedSchemaItems() OrderSchemaItems { items := make(OrderSchemaItems, 0, len(properties)) for k, v := range properties { items = append(items, OrderSchemaItem{ Name: k, Schema: v, }) } sort.Sort(items) return items } func (properties SchemaProperties) MarshalJSON() ([]byte, error) { if properties == nil { return []byte("null"), nil } return json.Marshal(properties.ToOrderedSchemaItems()) }