2018-02-07 19:02:18 +01:00
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
2018-06-25 16:31:44 +02:00
|
|
|
"bytes"
|
2018-02-07 19:02:18 +01:00
|
|
|
"fmt"
|
|
|
|
"os"
|
2018-06-25 16:31:44 +02:00
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"unicode"
|
|
|
|
"unicode/utf8"
|
2018-02-07 19:02:18 +01:00
|
|
|
|
2018-06-29 16:50:05 +02:00
|
|
|
"github.com/fatih/color"
|
2018-02-07 19:02:18 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// Log outputs a log message.
|
|
|
|
func Log(msg string, v ...interface{}) {
|
2018-06-29 16:50:05 +02:00
|
|
|
fmt.Printf(" %s\n", color.CyanString(msg, v...))
|
2018-02-07 19:02:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Fatal error
|
|
|
|
func Fatal(err error) {
|
2018-06-29 16:50:05 +02:00
|
|
|
fmt.Fprintf(os.Stderr, "\n %s %s\n\n", color.RedString("Error:"), err)
|
2018-02-07 19:02:18 +01:00
|
|
|
os.Exit(1)
|
|
|
|
}
|
2018-06-25 16:31:44 +02:00
|
|
|
|
2018-06-29 16:38:13 +02:00
|
|
|
// Finds the ansi escape sequences (like colors)
|
|
|
|
// Taken from: https://github.com/chalk/ansi-regex/blob/d9d806ecb45d899cf43408906a4440060c5c50e5/index.js
|
|
|
|
var ansiEscapes = regexp.MustCompile(`[\x1B\x9B][[\]()#;?]*` +
|
|
|
|
`(?:(?:(?:[a-zA-Z\d]*(?:;[a-zA-Z\\d]*)*)?\x07)` +
|
|
|
|
`|(?:(?:\d{1,4}(?:;\d{0,4})*)?[\dA-PRZcf-ntqry=><~]))`)
|
2018-06-25 16:31:44 +02:00
|
|
|
|
|
|
|
func EscapeAwareRuneCountInString(s string) int {
|
|
|
|
n := utf8.RuneCountInString(s)
|
2018-06-29 16:38:13 +02:00
|
|
|
for _, sm := range ansiEscapes.FindAllString(s, -1) {
|
2018-06-25 16:31:44 +02:00
|
|
|
n -= utf8.RuneCountInString(sm)
|
|
|
|
}
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
func RightPad(str string, length int) string {
|
|
|
|
return str + strings.Repeat(" ", length-EscapeAwareRuneCountInString(str))
|
|
|
|
}
|
|
|
|
|
|
|
|
// WrapString wraps the given string within lim width in characters.
|
|
|
|
//
|
|
|
|
// Wrapping is currently naive and only happens at white-space. A future
|
|
|
|
// version of the library will implement smarter wrapping. This means that
|
|
|
|
// pathological cases can dramatically reach past the limit, such as a very
|
|
|
|
// long word.
|
|
|
|
// This is taken from: https://github.com/mitchellh/go-wordwrap/tree/f253961a26562056904822f2a52d4692347db1bd
|
|
|
|
func WrapString(s string, lim uint) string {
|
|
|
|
// Initialize a buffer with a slightly larger size to account for breaks
|
|
|
|
init := make([]byte, 0, len(s))
|
|
|
|
buf := bytes.NewBuffer(init)
|
|
|
|
|
|
|
|
var current uint
|
|
|
|
var wordBuf, spaceBuf bytes.Buffer
|
|
|
|
|
|
|
|
for _, char := range s {
|
|
|
|
if char == '\n' {
|
|
|
|
if wordBuf.Len() == 0 {
|
|
|
|
if current+uint(spaceBuf.Len()) > lim {
|
|
|
|
current = 0
|
|
|
|
} else {
|
|
|
|
current += uint(spaceBuf.Len())
|
|
|
|
spaceBuf.WriteTo(buf)
|
|
|
|
}
|
|
|
|
spaceBuf.Reset()
|
|
|
|
} else {
|
|
|
|
current += uint(spaceBuf.Len() + wordBuf.Len())
|
|
|
|
spaceBuf.WriteTo(buf)
|
|
|
|
spaceBuf.Reset()
|
|
|
|
wordBuf.WriteTo(buf)
|
|
|
|
wordBuf.Reset()
|
|
|
|
}
|
|
|
|
buf.WriteRune(char)
|
|
|
|
current = 0
|
|
|
|
} else if unicode.IsSpace(char) {
|
|
|
|
if spaceBuf.Len() == 0 || wordBuf.Len() > 0 {
|
|
|
|
current += uint(spaceBuf.Len() + wordBuf.Len())
|
|
|
|
spaceBuf.WriteTo(buf)
|
|
|
|
spaceBuf.Reset()
|
|
|
|
wordBuf.WriteTo(buf)
|
|
|
|
wordBuf.Reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
spaceBuf.WriteRune(char)
|
|
|
|
} else {
|
|
|
|
|
|
|
|
wordBuf.WriteRune(char)
|
|
|
|
|
|
|
|
if current+uint(spaceBuf.Len()+wordBuf.Len()) > lim && uint(wordBuf.Len()) < lim {
|
|
|
|
buf.WriteRune('\n')
|
|
|
|
current = 0
|
|
|
|
spaceBuf.Reset()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if wordBuf.Len() == 0 {
|
|
|
|
if current+uint(spaceBuf.Len()) <= lim {
|
|
|
|
spaceBuf.WriteTo(buf)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
spaceBuf.WriteTo(buf)
|
|
|
|
wordBuf.WriteTo(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf.String()
|
|
|
|
}
|