forked from mirrors/gotosocial
9d0df426da
* feat: vendor minio client * feat: introduce storage package with s3 support * feat: serve s3 files directly this saves a lot of bandwith as the files are fetched from the object store directly * fix: use explicit local storage in tests * feat: integrate s3 storage with the main server * fix: add s3 config to cli tests * docs: explicitly set values in example config also adds license header to the storage package * fix: use better http status code on s3 redirect HTTP 302 Found is the best fit, as it signifies that the resource requested was found but not under its presumed URL 307/TemporaryRedirect would mean that this resource is usually located here, not in this case 303/SeeOther indicates that the redirection does not link to the requested resource but to another page * refactor: use context in storage driver interface
167 lines
3.7 KiB
Go
167 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
|
|
}
|