17453f10f5
cc #505
300 lines
6.1 KiB
Go
300 lines
6.1 KiB
Go
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"unicode"
|
|
|
|
"github.com/mattn/go-runewidth"
|
|
)
|
|
|
|
func isRoot(name string) bool { return filepath.Dir(name) == name }
|
|
|
|
func replaceTilde(s string) string {
|
|
if strings.HasPrefix(s, "~") {
|
|
s = strings.Replace(s, "~", gUser.HomeDir, 1)
|
|
}
|
|
return s
|
|
}
|
|
|
|
func runeSliceWidth(rs []rune) int {
|
|
w := 0
|
|
for _, r := range rs {
|
|
w += runewidth.RuneWidth(r)
|
|
}
|
|
return w
|
|
}
|
|
|
|
func runeSliceWidthRange(rs []rune, beg, end int) []rune {
|
|
curr := 0
|
|
b := 0
|
|
for i, r := range rs {
|
|
w := runewidth.RuneWidth(r)
|
|
switch {
|
|
case curr == beg:
|
|
b = i
|
|
case curr < beg && curr+w > beg:
|
|
b = i + 1
|
|
case curr == end:
|
|
return rs[b:i]
|
|
case curr > end:
|
|
return rs[b : i-1]
|
|
}
|
|
curr += w
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// This function is used to escape whitespaces and special characters with
|
|
// backlashes in a given string.
|
|
func escape(s string) string {
|
|
buf := make([]rune, 0, len(s))
|
|
for _, r := range s {
|
|
if unicode.IsSpace(r) || r == '\\' || r == ';' || r == '#' {
|
|
buf = append(buf, '\\')
|
|
}
|
|
buf = append(buf, r)
|
|
}
|
|
return string(buf)
|
|
}
|
|
|
|
// This function is used to remove backlashes that are used to escape
|
|
// whitespaces and special characters in a given string.
|
|
func unescape(s string) string {
|
|
esc := false
|
|
buf := make([]rune, 0, len(s))
|
|
for _, r := range s {
|
|
if esc {
|
|
if !unicode.IsSpace(r) && r != '\\' && r != ';' && r != '#' {
|
|
buf = append(buf, '\\')
|
|
}
|
|
buf = append(buf, r)
|
|
esc = false
|
|
continue
|
|
}
|
|
if r == '\\' {
|
|
esc = true
|
|
continue
|
|
}
|
|
esc = false
|
|
buf = append(buf, r)
|
|
}
|
|
if esc {
|
|
buf = append(buf, '\\')
|
|
}
|
|
return string(buf)
|
|
}
|
|
|
|
// This function splits the given string by whitespaces. It is aware of escaped
|
|
// whitespaces so that they are not splitted unintentionally.
|
|
func tokenize(s string) []string {
|
|
esc := false
|
|
var buf []rune
|
|
var toks []string
|
|
for _, r := range s {
|
|
if r == '\\' {
|
|
esc = true
|
|
buf = append(buf, r)
|
|
continue
|
|
}
|
|
if esc {
|
|
esc = false
|
|
buf = append(buf, r)
|
|
continue
|
|
}
|
|
if !unicode.IsSpace(r) {
|
|
buf = append(buf, r)
|
|
} else {
|
|
toks = append(toks, string(buf))
|
|
buf = nil
|
|
}
|
|
}
|
|
toks = append(toks, string(buf))
|
|
return toks
|
|
}
|
|
|
|
// This function splits the first word of a string delimited by whitespace from
|
|
// the rest. This is used to tokenize a string one by one without touching the
|
|
// rest. Whitespace on the left side of both the word and the rest are trimmed.
|
|
func splitWord(s string) (word, rest string) {
|
|
s = strings.TrimLeftFunc(s, unicode.IsSpace)
|
|
ind := len(s)
|
|
for i, c := range s {
|
|
if unicode.IsSpace(c) {
|
|
ind = i
|
|
break
|
|
}
|
|
}
|
|
word = s[0:ind]
|
|
rest = strings.TrimLeftFunc(s[ind:], unicode.IsSpace)
|
|
return
|
|
}
|
|
|
|
var reComment = regexp.MustCompile(`#.*$`)
|
|
var reTrailingSpace = regexp.MustCompile(`\s+$`)
|
|
|
|
func readPairs(r io.Reader) ([][]string, error) {
|
|
var pairs [][]string
|
|
s := bufio.NewScanner(r)
|
|
for s.Scan() {
|
|
line := s.Text()
|
|
|
|
line = reComment.ReplaceAllString(line, "")
|
|
line = reTrailingSpace.ReplaceAllString(line, "")
|
|
|
|
if line == "" {
|
|
continue
|
|
}
|
|
|
|
squote, dquote := false, false
|
|
pair := strings.FieldsFunc(line, func(r rune) bool {
|
|
if r == '\'' && !dquote {
|
|
squote = !squote
|
|
} else if r == '"' && !squote {
|
|
dquote = !dquote
|
|
}
|
|
return !squote && !dquote && unicode.IsSpace(r)
|
|
})
|
|
|
|
if len(pair) != 2 {
|
|
return nil, errors.New(fmt.Sprintf("expected pair but found: %s", s.Text()))
|
|
continue
|
|
}
|
|
|
|
for i := 0; i < len(pair); i++ {
|
|
squote, dquote := false, false
|
|
buf := make([]rune, 0, len(pair[i]))
|
|
for _, r := range pair[i] {
|
|
if r == '\'' && !dquote {
|
|
squote = !squote
|
|
continue
|
|
}
|
|
if r == '"' && !squote {
|
|
dquote = !dquote
|
|
continue
|
|
}
|
|
buf = append(buf, r)
|
|
}
|
|
pair[i] = string(buf)
|
|
}
|
|
|
|
pairs = append(pairs, pair)
|
|
}
|
|
|
|
return pairs, nil
|
|
}
|
|
|
|
// This function converts a size in bytes to a human readable form using metric
|
|
// suffixes (e.g. 1K = 1000). For values less than 10 the first significant
|
|
// digit is shown, otherwise it is hidden. Numbers are always rounded down.
|
|
// This should be fine for most human beings.
|
|
func humanize(size int64) string {
|
|
if size < 1000 {
|
|
return fmt.Sprintf("%dB", size)
|
|
}
|
|
|
|
suffix := []string{
|
|
"K", // kilo
|
|
"M", // mega
|
|
"G", // giga
|
|
"T", // tera
|
|
"P", // peta
|
|
"E", // exa
|
|
"Z", // zeta
|
|
"Y", // yotta
|
|
}
|
|
|
|
curr := float64(size) / 1000
|
|
for _, s := range suffix {
|
|
if curr < 10 {
|
|
return fmt.Sprintf("%.1f%s", curr-0.0499, s)
|
|
} else if curr < 1000 {
|
|
return fmt.Sprintf("%d%s", int(curr), s)
|
|
}
|
|
curr /= 1000
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
// This function compares two strings for natural sorting which takes into
|
|
// account values of numbers in strings. For example, '2' is less than '10',
|
|
// and similarly 'foo2bar' is less than 'foo10bar', but 'bar2bar' is greater
|
|
// than 'foo10bar'.
|
|
func naturalLess(s1, s2 string) bool {
|
|
lo1, lo2, hi1, hi2 := 0, 0, 0, 0
|
|
for {
|
|
if hi1 >= len(s1) {
|
|
return hi2 != len(s2)
|
|
}
|
|
|
|
if hi2 >= len(s2) {
|
|
return false
|
|
}
|
|
|
|
isDigit1 := isDigit(s1[hi1])
|
|
isDigit2 := isDigit(s2[hi2])
|
|
|
|
for lo1 = hi1; hi1 < len(s1) && isDigit(s1[hi1]) == isDigit1; hi1++ {
|
|
}
|
|
|
|
for lo2 = hi2; hi2 < len(s2) && isDigit(s2[hi2]) == isDigit2; hi2++ {
|
|
}
|
|
|
|
if s1[lo1:hi1] == s2[lo2:hi2] {
|
|
continue
|
|
}
|
|
|
|
if isDigit1 && isDigit2 {
|
|
num1, err1 := strconv.Atoi(s1[lo1:hi1])
|
|
num2, err2 := strconv.Atoi(s2[lo2:hi2])
|
|
|
|
if err1 == nil && err2 == nil {
|
|
return num1 < num2
|
|
}
|
|
}
|
|
|
|
return s1[lo1:hi1] < s2[lo2:hi2]
|
|
}
|
|
}
|
|
|
|
var reAltKey = regexp.MustCompile(`<a-(.)>`)
|
|
|
|
var reWord = regexp.MustCompile(`(\pL|\pN)+`)
|
|
var reWordBeg = regexp.MustCompile(`([^\pL\pN]|^)(\pL|\pN)`)
|
|
var reWordEnd = regexp.MustCompile(`(\pL|\pN)([^\pL\pN]|$)`)
|
|
|
|
func min(a, b int) int {
|
|
if a < b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
func max(a, b int) int {
|
|
if a > b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
func mod(a, b int) int {
|
|
return (a%b + b) % b
|
|
}
|
|
|
|
// We don't need no generic code
|
|
// We don't need no type control
|
|
// No dark templates in compiler
|
|
// Haskell leave them kids alone
|
|
// Hey Bjarne leave them kids alone
|
|
// All in all it's just another brick in the code
|
|
// All in all you're just another brick in the code
|
|
//
|
|
// -- Pink Trolled --
|