2020-05-11 20:57:46 +02:00
// 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"
)
2020-05-29 15:08:11 +02:00
// IsRoot returns whether path is the root directory.
2020-05-11 20:57:46 +02:00
func IsRoot ( path string ) bool {
abs , err := filepath . Abs ( path )
if err == nil {
path = abs
}
return filepath . Dir ( path ) == path
}
2020-05-29 15:08:11 +02:00
// ApplicationDir returns best base directory for specific OS.
2020-05-11 20:57:46 +02:00
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 ... ) ... )
}
2020-05-29 15:08:11 +02:00
// IsValidSetupDir checks if directory is valid for setup configuration.
2020-05-11 20:57:46 +02:00
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
}
}
}
}
2020-05-29 15:08:11 +02:00
// IsWritable determines if a directory is writeable.
2020-05-11 20:57:46 +02:00
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
}