mirror of
https://github.com/syncthing/syncthing.git
synced 2024-11-17 02:48:57 -07:00
3e24d82513
Most notably, it now detects all-lowercase files and returns these as-is. The tests have been expanded with two cases and are now used as a benchmark (admittedly a rather trivial one). name old time/op new time/op delta UnicodeLowercaseMaybeChange-8 4.59µs ± 2% 4.57µs ± 1% ~ (p=0.197 n=10+10) UnicodeLowercaseNoChange-8 3.26µs ± 1% 3.09µs ± 1% -5.27% (p=0.000 n=9+10)
46 lines
1.0 KiB
Go
46 lines
1.0 KiB
Go
// Copyright (C) 2017 The Syncthing Authors.
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
package fs
|
|
|
|
import (
|
|
"strings"
|
|
"unicode"
|
|
"unicode/utf8"
|
|
)
|
|
|
|
func UnicodeLowercase(s string) string {
|
|
i := firstCaseChange(s)
|
|
if i == -1 {
|
|
return s
|
|
}
|
|
|
|
var rs strings.Builder
|
|
// WriteRune always reserves utf8.UTFMax bytes for non-ASCII runes,
|
|
// even if it doesn't need all that space. Overallocate now to prevent
|
|
// it from ever triggering a reallocation.
|
|
rs.Grow(utf8.UTFMax - 1 + len(s))
|
|
rs.WriteString(s[:i])
|
|
|
|
for _, r := range s[i:] {
|
|
rs.WriteRune(unicode.ToLower(unicode.ToUpper(r)))
|
|
}
|
|
return rs.String()
|
|
}
|
|
|
|
// Byte index of the first rune r s.t. lower(upper(r)) != r.
|
|
func firstCaseChange(s string) int {
|
|
for i, r := range s {
|
|
if r <= unicode.MaxASCII && (r < 'A' || r > 'Z') {
|
|
continue
|
|
}
|
|
if unicode.ToLower(unicode.ToUpper(r)) != r {
|
|
return i
|
|
}
|
|
}
|
|
return -1
|
|
}
|