Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
kardolus
GitHub Repository: kardolus/chatgpt-cli
Path: blob/main/vendor/gopkg.in/yaml.v3/sorter.go
2872 views
1
//
2
// Copyright (c) 2011-2019 Canonical Ltd
3
//
4
// Licensed under the Apache License, Version 2.0 (the "License");
5
// you may not use this file except in compliance with the License.
6
// You may obtain a copy of the License at
7
//
8
// http://www.apache.org/licenses/LICENSE-2.0
9
//
10
// Unless required by applicable law or agreed to in writing, software
11
// distributed under the License is distributed on an "AS IS" BASIS,
12
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
// See the License for the specific language governing permissions and
14
// limitations under the License.
15
16
package yaml
17
18
import (
19
"reflect"
20
"unicode"
21
)
22
23
type keyList []reflect.Value
24
25
func (l keyList) Len() int { return len(l) }
26
func (l keyList) Swap(i, j int) { l[i], l[j] = l[j], l[i] }
27
func (l keyList) Less(i, j int) bool {
28
a := l[i]
29
b := l[j]
30
ak := a.Kind()
31
bk := b.Kind()
32
for (ak == reflect.Interface || ak == reflect.Ptr) && !a.IsNil() {
33
a = a.Elem()
34
ak = a.Kind()
35
}
36
for (bk == reflect.Interface || bk == reflect.Ptr) && !b.IsNil() {
37
b = b.Elem()
38
bk = b.Kind()
39
}
40
af, aok := keyFloat(a)
41
bf, bok := keyFloat(b)
42
if aok && bok {
43
if af != bf {
44
return af < bf
45
}
46
if ak != bk {
47
return ak < bk
48
}
49
return numLess(a, b)
50
}
51
if ak != reflect.String || bk != reflect.String {
52
return ak < bk
53
}
54
ar, br := []rune(a.String()), []rune(b.String())
55
digits := false
56
for i := 0; i < len(ar) && i < len(br); i++ {
57
if ar[i] == br[i] {
58
digits = unicode.IsDigit(ar[i])
59
continue
60
}
61
al := unicode.IsLetter(ar[i])
62
bl := unicode.IsLetter(br[i])
63
if al && bl {
64
return ar[i] < br[i]
65
}
66
if al || bl {
67
if digits {
68
return al
69
} else {
70
return bl
71
}
72
}
73
var ai, bi int
74
var an, bn int64
75
if ar[i] == '0' || br[i] == '0' {
76
for j := i - 1; j >= 0 && unicode.IsDigit(ar[j]); j-- {
77
if ar[j] != '0' {
78
an = 1
79
bn = 1
80
break
81
}
82
}
83
}
84
for ai = i; ai < len(ar) && unicode.IsDigit(ar[ai]); ai++ {
85
an = an*10 + int64(ar[ai]-'0')
86
}
87
for bi = i; bi < len(br) && unicode.IsDigit(br[bi]); bi++ {
88
bn = bn*10 + int64(br[bi]-'0')
89
}
90
if an != bn {
91
return an < bn
92
}
93
if ai != bi {
94
return ai < bi
95
}
96
return ar[i] < br[i]
97
}
98
return len(ar) < len(br)
99
}
100
101
// keyFloat returns a float value for v if it is a number/bool
102
// and whether it is a number/bool or not.
103
func keyFloat(v reflect.Value) (f float64, ok bool) {
104
switch v.Kind() {
105
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
106
return float64(v.Int()), true
107
case reflect.Float32, reflect.Float64:
108
return v.Float(), true
109
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
110
return float64(v.Uint()), true
111
case reflect.Bool:
112
if v.Bool() {
113
return 1, true
114
}
115
return 0, true
116
}
117
return 0, false
118
}
119
120
// numLess returns whether a < b.
121
// a and b must necessarily have the same kind.
122
func numLess(a, b reflect.Value) bool {
123
switch a.Kind() {
124
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
125
return a.Int() < b.Int()
126
case reflect.Float32, reflect.Float64:
127
return a.Float() < b.Float()
128
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
129
return a.Uint() < b.Uint()
130
case reflect.Bool:
131
return !a.Bool() && b.Bool()
132
}
133
panic("not a number")
134
}
135
136