package common import ( "bufio" "fmt" "io" "io/ioutil" "os" "os/signal" "strings" "syscall" ) var ( GoPath = os.Getenv("GOPATH") ) func TrapSignal(cb func()) { c := make(chan os.Signal, 1) signal.Notify(c, os.Interrupt, syscall.SIGTERM) go func() { for sig := range c { fmt.Printf("captured %v, exiting...\n", sig) if cb != nil { cb() } os.Exit(1) } }() select {} } func Exit(s string) { fmt.Printf(s + "\n") os.Exit(1) } func EnsureDir(dir string, mode os.FileMode) error { if _, err := os.Stat(dir); os.IsNotExist(err) { err := os.MkdirAll(dir, mode) if err != nil { return fmt.Errorf("Could not create directory %v. %v", dir, err) } } return nil } func IsDirEmpty(name string) (bool, error) { f, err := os.Open(name) if err != nil { if os.IsNotExist(err) { return true, err } // Otherwise perhaps a permission // error or some other error. return false, err } defer f.Close() _, err = f.Readdirnames(1) // Or f.Readdir(1) if err == io.EOF { return true, nil } return false, err // Either not empty or error, suits both cases } func FileExists(filePath string) bool { _, err := os.Stat(filePath) return !os.IsNotExist(err) } func ReadFile(filePath string) ([]byte, error) { return ioutil.ReadFile(filePath) } func MustReadFile(filePath string) []byte { fileBytes, err := ioutil.ReadFile(filePath) if err != nil { Exit(Fmt("MustReadFile failed: %v", err)) return nil } return fileBytes } func WriteFile(filePath string, contents []byte, mode os.FileMode) error { return ioutil.WriteFile(filePath, contents, mode) } func MustWriteFile(filePath string, contents []byte, mode os.FileMode) { err := WriteFile(filePath, contents, mode) if err != nil { Exit(Fmt("MustWriteFile failed: %v", err)) } } // WriteFileAtomic writes newBytes to temp and atomically moves to filePath // when everything else succeeds. func WriteFileAtomic(filePath string, newBytes []byte, mode os.FileMode) error { f, err := ioutil.TempFile("", "") if err != nil { return err } _, err = f.Write(newBytes) if err == nil { err = f.Sync() } if closeErr := f.Close(); err == nil { err = closeErr } if permErr := os.Chmod(f.Name(), mode); err == nil { err = permErr } if err == nil { err = os.Rename(f.Name(), filePath) } // any err should result in full cleanup if err != nil { os.Remove(f.Name()) } return err } //-------------------------------------------------------------------------------- func Tempfile(prefix string) (*os.File, string) { file, err := ioutil.TempFile("", prefix) if err != nil { PanicCrisis(err) } return file, file.Name() } func Tempdir(prefix string) (*os.File, string) { tempDir := os.TempDir() + "/" + prefix + RandStr(12) err := EnsureDir(tempDir, 0700) if err != nil { panic(Fmt("Error creating temp dir: %v", err)) } dir, err := os.Open(tempDir) if err != nil { panic(Fmt("Error opening temp dir: %v", err)) } return dir, tempDir } //-------------------------------------------------------------------------------- func Prompt(prompt string, defaultValue string) (string, error) { fmt.Print(prompt) reader := bufio.NewReader(os.Stdin) line, err := reader.ReadString('\n') if err != nil { return defaultValue, err } else { line = strings.TrimSpace(line) if line == "" { return defaultValue, nil } return line, nil } }