mirror of
https://github.com/safing/portmaster
synced 2025-09-02 18:49:14 +00:00
63 lines
1.7 KiB
Go
63 lines
1.7 KiB
Go
package utils
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/fs"
|
|
"log/slog"
|
|
"os"
|
|
"runtime"
|
|
)
|
|
|
|
const isWindows = runtime.GOOS == "windows"
|
|
|
|
// EnsureDirectory ensures that the given directory exists and that is has the given permissions set.
|
|
// If path is a file, it is deleted and a directory created.
|
|
func EnsureDirectory(path string, perm FSPermission) error {
|
|
if !perm.IsExecPermission() {
|
|
slog.Warn("utils: setting not executable permission for directory", "dir", path)
|
|
}
|
|
|
|
// open path
|
|
f, err := os.Stat(path)
|
|
if err == nil {
|
|
// file exists
|
|
if f.IsDir() {
|
|
// directory exists, check permissions
|
|
if isWindows {
|
|
// Ignore windows permission error. For none admin users it will always fail.
|
|
_ = SetFilePermission(path, perm)
|
|
return nil
|
|
} else if f.Mode().Perm() != perm.AsUnixPermission() {
|
|
return SetFilePermission(path, perm)
|
|
}
|
|
return nil
|
|
}
|
|
err = os.Remove(path)
|
|
if err != nil {
|
|
return fmt.Errorf("could not remove file %s to place dir: %w", path, err)
|
|
}
|
|
}
|
|
// file does not exist (or has been deleted)
|
|
if err == nil || errors.Is(err, fs.ErrNotExist) {
|
|
err = os.MkdirAll(path, perm.AsUnixPermission())
|
|
if err != nil {
|
|
return fmt.Errorf("could not create dir %s: %w", path, err)
|
|
}
|
|
// Set permissions.
|
|
err = SetFilePermission(path, perm)
|
|
// Ignore windows permission error. For none admin users it will always fail.
|
|
if !isWindows {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
// other error opening path
|
|
return fmt.Errorf("failed to access %s: %w", path, err)
|
|
}
|
|
|
|
// PathExists returns whether the given path (file or dir) exists.
|
|
func PathExists(path string) bool {
|
|
_, err := os.Stat(path)
|
|
return err == nil || errors.Is(err, fs.ErrExist)
|
|
}
|