133 lines
3.2 KiB
Go
133 lines
3.2 KiB
Go
// Copyright (C) 2019 Storj Labs, Inc.
|
|
// See LICENSE for copying information.
|
|
|
|
package fpath
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
|
|
"github.com/zeebo/errs"
|
|
)
|
|
|
|
// IsRoot returns whether path is the root directory
|
|
func IsRoot(path string) bool {
|
|
abs, err := filepath.Abs(path)
|
|
if err == nil {
|
|
path = abs
|
|
}
|
|
|
|
return filepath.Dir(path) == path
|
|
}
|
|
|
|
// ApplicationDir returns best base directory for specific OS
|
|
func ApplicationDir(subdir ...string) string {
|
|
for i := range subdir {
|
|
if runtime.GOOS == "windows" || runtime.GOOS == "darwin" {
|
|
subdir[i] = strings.Title(subdir[i])
|
|
} else {
|
|
subdir[i] = strings.ToLower(subdir[i])
|
|
}
|
|
}
|
|
var appdir string
|
|
home := os.Getenv("HOME")
|
|
|
|
switch runtime.GOOS {
|
|
case "windows":
|
|
// Windows standards: https://msdn.microsoft.com/en-us/library/windows/apps/hh465094.aspx?f=255&MSPPError=-2147217396
|
|
for _, env := range []string{"AppData", "AppDataLocal", "UserProfile", "Home"} {
|
|
val := os.Getenv(env)
|
|
if val != "" {
|
|
appdir = val
|
|
break
|
|
}
|
|
}
|
|
case "darwin":
|
|
// Mac standards: https://developer.apple.com/library/archive/documentation/FileManagement/Conceptual/FileSystemProgrammingGuide/MacOSXDirectories/MacOSXDirectories.html
|
|
appdir = filepath.Join(home, "Library", "Application Support")
|
|
case "linux":
|
|
fallthrough
|
|
default:
|
|
// Linux standards: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
|
|
appdir = os.Getenv("XDG_DATA_HOME")
|
|
if appdir == "" && home != "" {
|
|
appdir = filepath.Join(home, ".local", "share")
|
|
}
|
|
}
|
|
return filepath.Join(append([]string{appdir}, subdir...)...)
|
|
}
|
|
|
|
// IsValidSetupDir checks if directory is valid for setup configuration
|
|
func IsValidSetupDir(name string) (ok bool, err error) {
|
|
_, err = os.Stat(name)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return true, err
|
|
}
|
|
return false, err
|
|
}
|
|
|
|
/* #nosec G304 */ // The function limits later on paths having a config file
|
|
f, err := os.Open(name)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
defer func() {
|
|
err = errs.Combine(err, f.Close())
|
|
}()
|
|
|
|
for {
|
|
var filenames []string
|
|
filenames, err = f.Readdirnames(100)
|
|
if err == io.EOF {
|
|
// nothing more
|
|
return true, nil
|
|
} else if err != nil {
|
|
// something went wrong
|
|
return false, err
|
|
}
|
|
|
|
for _, filename := range filenames {
|
|
if filename == "config.yaml" {
|
|
return false, nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// IsWritable determines if a directory is writeable
|
|
func IsWritable(filepath string) (bool, error) {
|
|
info, err := os.Stat(filepath)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !info.IsDir() {
|
|
return false, fmt.Errorf("path %s is not a directory", filepath)
|
|
}
|
|
|
|
// Check if the user bit is enabled in file permission
|
|
if info.Mode().Perm()&0200 == 0 {
|
|
return false, fmt.Errorf("write permission bit is not set on this file for user")
|
|
}
|
|
|
|
// Test if user can create file
|
|
// There is no OS cross-compatible method for
|
|
// determining if a user has write permissions on a folder.
|
|
// We can test by attempting to create a file in the folder.
|
|
testFile := path.Join(filepath, ".perm")
|
|
file, err := os.Create(testFile) // For read access.
|
|
if err != nil {
|
|
return false, fmt.Errorf("write permission bit is not set on this file for user")
|
|
}
|
|
|
|
_ = file.Close()
|
|
_ = os.Remove(testFile)
|
|
|
|
return true, nil
|
|
}
|