diff options
Diffstat (limited to 'vendor/gopkg.in/yaml.v3/sorter.go')
| -rw-r--r-- | vendor/gopkg.in/yaml.v3/sorter.go | 134 |
1 files changed, 134 insertions, 0 deletions
diff --git a/vendor/gopkg.in/yaml.v3/sorter.go b/vendor/gopkg.in/yaml.v3/sorter.go new file mode 100644 index 0000000..9210ece --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/sorter.go | |||
| @@ -0,0 +1,134 @@ | |||
| 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 | } | ||
