435c0d9fc8
This PR switches the Nomad repository from using govendor to Go modules for managing dependencies. Aspects of the Nomad workflow remain pretty much the same. The usual Makefile targets should continue to work as they always did. The API submodule simply defers to the parent Nomad version on the repository, keeping the semantics of API versioning that currently exists.
168 lines
3.7 KiB
Go
168 lines
3.7 KiB
Go
package homedir
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
// DisableCache will disable caching of the home directory. Caching is enabled
|
|
// by default.
|
|
var DisableCache bool
|
|
|
|
var homedirCache string
|
|
var cacheLock sync.RWMutex
|
|
|
|
// Dir returns the home directory for the executing user.
|
|
//
|
|
// This uses an OS-specific method for discovering the home directory.
|
|
// An error is returned if a home directory cannot be detected.
|
|
func Dir() (string, error) {
|
|
if !DisableCache {
|
|
cacheLock.RLock()
|
|
cached := homedirCache
|
|
cacheLock.RUnlock()
|
|
if cached != "" {
|
|
return cached, nil
|
|
}
|
|
}
|
|
|
|
cacheLock.Lock()
|
|
defer cacheLock.Unlock()
|
|
|
|
var result string
|
|
var err error
|
|
if runtime.GOOS == "windows" {
|
|
result, err = dirWindows()
|
|
} else {
|
|
// Unix-like system, so just assume Unix
|
|
result, err = dirUnix()
|
|
}
|
|
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
homedirCache = result
|
|
return result, nil
|
|
}
|
|
|
|
// Expand expands the path to include the home directory if the path
|
|
// is prefixed with `~`. If it isn't prefixed with `~`, the path is
|
|
// returned as-is.
|
|
func Expand(path string) (string, error) {
|
|
if len(path) == 0 {
|
|
return path, nil
|
|
}
|
|
|
|
if path[0] != '~' {
|
|
return path, nil
|
|
}
|
|
|
|
if len(path) > 1 && path[1] != '/' && path[1] != '\\' {
|
|
return "", errors.New("cannot expand user-specific home dir")
|
|
}
|
|
|
|
dir, err := Dir()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return filepath.Join(dir, path[1:]), nil
|
|
}
|
|
|
|
// Reset clears the cache, forcing the next call to Dir to re-detect
|
|
// the home directory. This generally never has to be called, but can be
|
|
// useful in tests if you're modifying the home directory via the HOME
|
|
// env var or something.
|
|
func Reset() {
|
|
cacheLock.Lock()
|
|
defer cacheLock.Unlock()
|
|
homedirCache = ""
|
|
}
|
|
|
|
func dirUnix() (string, error) {
|
|
homeEnv := "HOME"
|
|
if runtime.GOOS == "plan9" {
|
|
// On plan9, env vars are lowercase.
|
|
homeEnv = "home"
|
|
}
|
|
|
|
// First prefer the HOME environmental variable
|
|
if home := os.Getenv(homeEnv); home != "" {
|
|
return home, nil
|
|
}
|
|
|
|
var stdout bytes.Buffer
|
|
|
|
// If that fails, try OS specific commands
|
|
if runtime.GOOS == "darwin" {
|
|
cmd := exec.Command("sh", "-c", `dscl -q . -read /Users/"$(whoami)" NFSHomeDirectory | sed 's/^[^ ]*: //'`)
|
|
cmd.Stdout = &stdout
|
|
if err := cmd.Run(); err == nil {
|
|
result := strings.TrimSpace(stdout.String())
|
|
if result != "" {
|
|
return result, nil
|
|
}
|
|
}
|
|
} else {
|
|
cmd := exec.Command("getent", "passwd", strconv.Itoa(os.Getuid()))
|
|
cmd.Stdout = &stdout
|
|
if err := cmd.Run(); err != nil {
|
|
// If the error is ErrNotFound, we ignore it. Otherwise, return it.
|
|
if err != exec.ErrNotFound {
|
|
return "", err
|
|
}
|
|
} else {
|
|
if passwd := strings.TrimSpace(stdout.String()); passwd != "" {
|
|
// username:password:uid:gid:gecos:home:shell
|
|
passwdParts := strings.SplitN(passwd, ":", 7)
|
|
if len(passwdParts) > 5 {
|
|
return passwdParts[5], nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If all else fails, try the shell
|
|
stdout.Reset()
|
|
cmd := exec.Command("sh", "-c", "cd && pwd")
|
|
cmd.Stdout = &stdout
|
|
if err := cmd.Run(); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
result := strings.TrimSpace(stdout.String())
|
|
if result == "" {
|
|
return "", errors.New("blank output when reading home directory")
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func dirWindows() (string, error) {
|
|
// First prefer the HOME environmental variable
|
|
if home := os.Getenv("HOME"); home != "" {
|
|
return home, nil
|
|
}
|
|
|
|
// Prefer standard environment variable USERPROFILE
|
|
if home := os.Getenv("USERPROFILE"); home != "" {
|
|
return home, nil
|
|
}
|
|
|
|
drive := os.Getenv("HOMEDRIVE")
|
|
path := os.Getenv("HOMEPATH")
|
|
home := drive + path
|
|
if drive == "" || path == "" {
|
|
return "", errors.New("HOMEDRIVE, HOMEPATH, or USERPROFILE are blank")
|
|
}
|
|
|
|
return home, nil
|
|
}
|