2016-11-15 16:23:20 +01:00
|
|
|
package goutil
|
|
|
|
|
|
|
|
import (
|
2018-07-01 16:03:26 +02:00
|
|
|
"bufio"
|
|
|
|
"errors"
|
2018-11-04 23:07:08 +01:00
|
|
|
"flag"
|
2018-07-01 16:03:26 +02:00
|
|
|
"fmt"
|
2016-11-15 16:23:20 +01:00
|
|
|
"io/ioutil"
|
2018-07-01 16:03:26 +02:00
|
|
|
"os"
|
2018-11-04 23:07:08 +01:00
|
|
|
"os/exec"
|
2016-11-19 01:08:56 +01:00
|
|
|
"path"
|
2016-11-15 16:23:20 +01:00
|
|
|
"path/filepath"
|
2018-11-04 23:07:08 +01:00
|
|
|
"runtime"
|
2016-11-19 00:55:13 +01:00
|
|
|
"strings"
|
2016-11-15 16:23:20 +01:00
|
|
|
)
|
|
|
|
|
2018-07-01 16:03:26 +02:00
|
|
|
func DecorateError(s string, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return errors.New(s + ": " + err.Error())
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-15 16:23:20 +01:00
|
|
|
func WriteFile(filename string, data string) error {
|
|
|
|
return ioutil.WriteFile(filename, []byte(data), 0644)
|
|
|
|
}
|
|
|
|
|
2018-11-04 23:07:08 +01:00
|
|
|
func AppendToFile(filename string, data string) error {
|
|
|
|
f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0644)
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("AppendToFile " + filename + ": " + err.Error())
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
_, err = f.WriteString(data)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-11-15 16:23:20 +01:00
|
|
|
func ReadFile(filename string) (string, error) {
|
|
|
|
data, err := ioutil.ReadFile(filename)
|
|
|
|
return string(data), err
|
|
|
|
}
|
|
|
|
|
|
|
|
func TrimExt(path string) string {
|
|
|
|
extension := filepath.Ext(path)
|
|
|
|
return path[0 : len(path)-len(extension)]
|
|
|
|
}
|
2016-11-19 00:55:13 +01:00
|
|
|
|
|
|
|
func ListFilesExt(dir string, ext string) []string {
|
|
|
|
list := make([]string, 0)
|
|
|
|
ext = strings.ToUpper(ext)
|
|
|
|
files, err := ioutil.ReadDir(dir)
|
|
|
|
if err == nil {
|
|
|
|
for _, file := range files {
|
|
|
|
if !file.IsDir() {
|
|
|
|
if strings.ToUpper(filepath.Ext(file.Name())) == ext {
|
2016-11-19 01:08:56 +01:00
|
|
|
list = append(list, path.Join(dir, file.Name()))
|
2016-11-19 00:55:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return list
|
|
|
|
}
|
2018-07-01 16:03:26 +02:00
|
|
|
|
2018-12-08 20:13:16 +01:00
|
|
|
func RecListFilesExt(dir string, ext string) []string {
|
|
|
|
list := make([]string, 0)
|
|
|
|
traverse := func(path string, info os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if info.IsDir() {
|
|
|
|
list = append(list, ListFilesExt(path, ext)...)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
filepath.Walk(dir, traverse)
|
|
|
|
return list
|
|
|
|
}
|
|
|
|
|
2018-07-01 16:03:26 +02:00
|
|
|
func PathExists(path string) bool {
|
|
|
|
if _, err := os.Stat(path); os.IsNotExist(err) {
|
|
|
|
return false
|
|
|
|
} else {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func AskFor(question string) (string, error) {
|
|
|
|
fmt.Print(question + ": ")
|
|
|
|
reader := bufio.NewReader(os.Stdin)
|
|
|
|
s, err := reader.ReadString('\n')
|
|
|
|
if err == nil {
|
|
|
|
return strings.TrimSpace(s), nil
|
|
|
|
} else {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func IntMin(x, y int) int {
|
|
|
|
if x < y {
|
|
|
|
return x
|
|
|
|
} else {
|
|
|
|
return y
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func IntMax(x, y int) int {
|
|
|
|
if x > y {
|
|
|
|
return x
|
|
|
|
} else {
|
|
|
|
return y
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func StrSliceAt(sl []string, i int) string {
|
|
|
|
if i < len(sl) {
|
|
|
|
return sl[i]
|
|
|
|
} else {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func StrSlice(sl []string, start int, end int) []string {
|
|
|
|
bound := func(i int) int { return IntMin(IntMax(0, i), len(sl)) }
|
|
|
|
start = bound(start)
|
|
|
|
end = bound(end)
|
|
|
|
return sl[start:end]
|
|
|
|
}
|
2018-11-04 23:07:08 +01:00
|
|
|
|
|
|
|
func OpenInDefaultApp(filename string, wait bool) error {
|
|
|
|
var cmd *exec.Cmd
|
2018-12-10 17:06:59 +01:00
|
|
|
goos := runtime.GOOS
|
|
|
|
switch goos {
|
2018-11-04 23:07:08 +01:00
|
|
|
case "windows":
|
|
|
|
cmd = exec.Command(filepath.Join(os.Getenv("SYSTEMROOT"), "System32", "rundll32.exe"), "url.dll,FileProtocolHandler", filename)
|
|
|
|
case "darwin":
|
|
|
|
cmd = exec.Command("open", filename)
|
|
|
|
default:
|
|
|
|
cmd = exec.Command("xdg-open", filename)
|
|
|
|
}
|
|
|
|
if wait {
|
2018-12-10 17:06:59 +01:00
|
|
|
err := cmd.Run()
|
|
|
|
if goos == "windows" {
|
|
|
|
AskFor("Press Enter when you're done")
|
|
|
|
}
|
|
|
|
return err
|
2018-11-04 23:07:08 +01:00
|
|
|
} else {
|
|
|
|
return cmd.Start()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type CommandFunc func(args []string) error
|
|
|
|
|
|
|
|
type CommandFlagsInit func(s *flag.FlagSet)
|
|
|
|
|
|
|
|
type CommandInitExecFunc func() (CommandFlagsInit, CommandFunc)
|
|
|
|
|
|
|
|
type Command struct {
|
|
|
|
cmd string
|
|
|
|
exec CommandFunc
|
|
|
|
desc string
|
|
|
|
flags *flag.FlagSet
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCommandWithFlags(cmd string, f CommandInitExecFunc, desc string) Command {
|
|
|
|
flagsInit, exec := f()
|
|
|
|
var flags *flag.FlagSet
|
|
|
|
if flagsInit != nil {
|
|
|
|
flags = flag.NewFlagSet(cmd, flag.ExitOnError)
|
|
|
|
flagsInit(flags)
|
|
|
|
}
|
|
|
|
c := Command{cmd, exec, desc, flags}
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCommand(cmd string, f CommandFunc, desc string) Command {
|
|
|
|
return NewCommandWithFlags(cmd, func() (CommandFlagsInit, CommandFunc) { return nil, f }, desc)
|
|
|
|
}
|
|
|
|
|
|
|
|
type commandCollection map[string]Command
|
|
|
|
|
2018-12-09 18:14:38 +01:00
|
|
|
func usageAndExit(cmds []Command) {
|
2018-12-09 23:57:43 +01:00
|
|
|
app := os.Args[0]
|
|
|
|
fmt.Println(app)
|
2018-12-09 10:06:27 +01:00
|
|
|
fmt.Println()
|
2018-11-04 23:07:08 +01:00
|
|
|
fmt.Println("Possible commands:")
|
|
|
|
for _, cmd := range cmds {
|
|
|
|
fmt.Println(" " + cmd.cmd)
|
|
|
|
fmt.Println(" \t" + cmd.desc)
|
|
|
|
}
|
|
|
|
fmt.Println()
|
2018-12-09 23:57:43 +01:00
|
|
|
fmt.Println("for detailed information type '" + app + " help COMMAND'")
|
2018-11-04 23:07:08 +01:00
|
|
|
os.Exit(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func Execute(possibleCommands []Command) error {
|
|
|
|
commands := commandCollection{}
|
|
|
|
for _, c := range possibleCommands {
|
|
|
|
commands[c.cmd] = c
|
|
|
|
}
|
|
|
|
arglen := len(os.Args)
|
|
|
|
if arglen <= 1 {
|
2018-12-09 18:14:38 +01:00
|
|
|
usageAndExit(possibleCommands)
|
2018-11-04 23:07:08 +01:00
|
|
|
}
|
|
|
|
cmdStr := os.Args[1]
|
|
|
|
if cmdStr == "help" {
|
|
|
|
if arglen == 2 {
|
2018-12-09 18:14:38 +01:00
|
|
|
usageAndExit(possibleCommands)
|
2018-11-04 23:07:08 +01:00
|
|
|
} else {
|
|
|
|
cmd, ok := commands[os.Args[2]]
|
|
|
|
if !ok {
|
2018-12-09 18:14:38 +01:00
|
|
|
usageAndExit(possibleCommands)
|
2018-11-04 23:07:08 +01:00
|
|
|
} else {
|
|
|
|
fmt.Println("Description of " + cmd.cmd + ":")
|
|
|
|
fmt.Println(" " + cmd.desc)
|
|
|
|
if cmd.flags != nil {
|
|
|
|
fmt.Println("Possible flags:")
|
|
|
|
cmd.flags.PrintDefaults()
|
|
|
|
} else {
|
|
|
|
fmt.Println("No flags")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cmd, ok := commands[cmdStr]
|
|
|
|
if !ok {
|
2018-12-09 18:14:38 +01:00
|
|
|
usageAndExit(possibleCommands)
|
2018-11-04 23:07:08 +01:00
|
|
|
}
|
|
|
|
if cmd.flags != nil {
|
|
|
|
cmd.flags.Parse(os.Args[2:])
|
|
|
|
return cmd.exec(cmd.flags.Args())
|
|
|
|
} else {
|
|
|
|
return cmd.exec(os.Args[2:])
|
|
|
|
}
|
|
|
|
}
|