mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-07 16:42:27 +01:00
2e12119a9e
Natural sorting is needed for sort_by_label_natural() and sort_by_label_natural_desc()
functions in MetricsQL - see https://github.com/VictoriaMetrics/VictoriaMetrics/issues/6192
and https://github.com/VictoriaMetrics/VictoriaMetrics/pull/6256
Natural sorting will be also used by `| sort ...` pipe in VictoriaLogs -
see https://docs.victoriametrics.com/victorialogs/logsql/#sort-pipe
(cherry picked from commit 707f3a69db
)
112 lines
1.9 KiB
Go
112 lines
1.9 KiB
Go
package stringsutil
|
|
|
|
import (
|
|
"math"
|
|
)
|
|
|
|
// LessNatural returns true if a is less than b using natural sort comparison.
|
|
//
|
|
// See https://en.wikipedia.org/wiki/Natural_sort_order
|
|
func LessNatural(a, b string) bool {
|
|
isReverse := false
|
|
for {
|
|
if len(a) > len(b) {
|
|
a, b = b, a
|
|
isReverse = !isReverse
|
|
}
|
|
|
|
// Skip common prefix except of decimal digits
|
|
i := 0
|
|
for i < len(a) {
|
|
cA := a[i]
|
|
cB := b[i]
|
|
|
|
if cA >= '0' && cA <= '9' {
|
|
if cB >= '0' && cB <= '9' {
|
|
break
|
|
}
|
|
return !isReverse
|
|
}
|
|
if cB >= '0' && cB <= '9' {
|
|
return isReverse
|
|
}
|
|
if cA != cB {
|
|
// This should work properly for utf8 bytes in the middle of encoded unicode char, since:
|
|
// - utf8 bytes for multi-byte chars are bigger than decimal digit chars
|
|
// - sorting of utf8-encoded strings works properly thanks to utf8 properties
|
|
if isReverse {
|
|
return cB < cA
|
|
}
|
|
return cA < cB
|
|
}
|
|
|
|
i++
|
|
}
|
|
a = a[i:]
|
|
b = b[i:]
|
|
if len(a) == 0 {
|
|
if isReverse {
|
|
return false
|
|
}
|
|
return len(b) > 0
|
|
}
|
|
|
|
// Collect digit prefixes for a and b and then compare them.
|
|
|
|
iA := 1
|
|
nA := uint64(a[0] - '0')
|
|
for iA < len(a) {
|
|
c := a[iA]
|
|
if c < '0' || c > '9' {
|
|
break
|
|
}
|
|
if nA > (math.MaxUint64-9)/10 {
|
|
// Too big integer. Fall back to string comparison
|
|
if isReverse {
|
|
return b < a
|
|
}
|
|
return a < b
|
|
}
|
|
nA *= 10
|
|
nA += uint64(c - '0')
|
|
iA++
|
|
}
|
|
|
|
iB := 1
|
|
nB := uint64(b[0] - '0')
|
|
for iB < len(b) {
|
|
c := b[iB]
|
|
if c < '0' || c > '9' {
|
|
break
|
|
}
|
|
if nB > (math.MaxUint64-9)/10 {
|
|
// Too big integer. Fall back to string comparison
|
|
if isReverse {
|
|
return b < a
|
|
}
|
|
return a < b
|
|
}
|
|
nB *= 10
|
|
nB += uint64(c - '0')
|
|
iB++
|
|
}
|
|
|
|
if nA != nB {
|
|
if isReverse {
|
|
return nB < nA
|
|
}
|
|
return nA < nB
|
|
}
|
|
|
|
if iA != iB {
|
|
if isReverse {
|
|
return iB < iA
|
|
}
|
|
return iA < iB
|
|
}
|
|
|
|
a = a[iA:]
|
|
b = b[iB:]
|
|
}
|
|
}
|