2013-12-10 05:27:09 +08:00
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
2023-09-27 20:40:51 +08:00
|
|
|
"bytes"
|
|
|
|
"crypto/sha256"
|
2013-12-10 05:27:09 +08:00
|
|
|
"errors"
|
2023-09-27 20:40:51 +08:00
|
|
|
"fmt"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/glog"
|
2013-12-10 05:27:09 +08:00
|
|
|
"os"
|
2020-07-17 13:38:55 +08:00
|
|
|
"os/user"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2019-12-29 04:28:58 +08:00
|
|
|
"time"
|
2013-12-10 05:27:09 +08:00
|
|
|
)
|
|
|
|
|
2023-09-27 20:40:51 +08:00
|
|
|
const maxFilenameLength = 255
|
|
|
|
|
2013-12-10 05:27:09 +08:00
|
|
|
func TestFolderWritable(folder string) (err error) {
|
|
|
|
fileInfo, err := os.Stat(folder)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !fileInfo.IsDir() {
|
|
|
|
return errors.New("Not a valid folder!")
|
|
|
|
}
|
|
|
|
perm := fileInfo.Mode().Perm()
|
|
|
|
glog.V(0).Infoln("Folder", folder, "Permission:", perm)
|
|
|
|
if 0200&perm != 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return errors.New("Not writable!")
|
|
|
|
}
|
2014-04-10 00:44:58 +08:00
|
|
|
|
2016-07-03 12:53:03 +08:00
|
|
|
func GetFileSize(file *os.File) (size int64, err error) {
|
|
|
|
var fi os.FileInfo
|
|
|
|
if fi, err = file.Stat(); err == nil {
|
|
|
|
size = fi.Size()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2019-06-03 17:26:31 +08:00
|
|
|
|
|
|
|
func FileExists(filename string) bool {
|
|
|
|
|
|
|
|
_, err := os.Stat(filename)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
|
|
|
|
}
|
2019-12-29 04:28:58 +08:00
|
|
|
|
2022-08-25 15:19:08 +08:00
|
|
|
func FolderExists(folder string) bool {
|
|
|
|
|
|
|
|
fileInfo, err := os.Stat(folder)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return fileInfo.IsDir()
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-12-29 04:28:58 +08:00
|
|
|
func CheckFile(filename string) (exists, canRead, canWrite bool, modTime time.Time, fileSize int64) {
|
|
|
|
exists = true
|
|
|
|
fi, err := os.Stat(filename)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
exists = false
|
|
|
|
return
|
|
|
|
}
|
2020-06-08 23:49:57 +08:00
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("check %s: %v", filename, err)
|
|
|
|
return
|
|
|
|
}
|
2019-12-29 04:28:58 +08:00
|
|
|
if fi.Mode()&0400 != 0 {
|
|
|
|
canRead = true
|
|
|
|
}
|
|
|
|
if fi.Mode()&0200 != 0 {
|
|
|
|
canWrite = true
|
|
|
|
}
|
|
|
|
modTime = fi.ModTime()
|
|
|
|
fileSize = fi.Size()
|
|
|
|
return
|
|
|
|
}
|
2020-07-17 13:38:55 +08:00
|
|
|
|
|
|
|
func ResolvePath(path string) string {
|
|
|
|
|
2021-02-23 08:50:56 +08:00
|
|
|
if !strings.Contains(path, "~") {
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
|
2020-07-17 13:38:55 +08:00
|
|
|
usr, _ := user.Current()
|
|
|
|
dir := usr.HomeDir
|
|
|
|
|
|
|
|
if path == "~" {
|
|
|
|
// In case of "~", which won't be caught by the "else if"
|
|
|
|
path = dir
|
|
|
|
} else if strings.HasPrefix(path, "~/") {
|
|
|
|
// Use strings.HasPrefix so we don't match paths like
|
|
|
|
// "/something/~/something/"
|
|
|
|
path = filepath.Join(dir, path[2:])
|
|
|
|
}
|
|
|
|
|
|
|
|
return path
|
|
|
|
}
|
2021-09-25 16:18:44 +08:00
|
|
|
|
|
|
|
func FileNameBase(filename string) string {
|
|
|
|
lastDotIndex := strings.LastIndex(filename, ".")
|
|
|
|
if lastDotIndex < 0 {
|
|
|
|
return filename
|
|
|
|
}
|
|
|
|
return filename[:lastDotIndex]
|
|
|
|
}
|
2022-02-05 03:14:04 +08:00
|
|
|
|
2023-09-27 20:40:51 +08:00
|
|
|
func ToShortFileName(path string) string {
|
|
|
|
fileName := filepath.Base(path)
|
|
|
|
if fileNameBytes := []byte(fileName); len(fileNameBytes) > maxFilenameLength {
|
|
|
|
shaStr := fmt.Sprintf("%x", sha256.Sum256(fileNameBytes))
|
|
|
|
fileNameBase := FileNameBase(fileName)
|
|
|
|
fileExt := fileName[len(fileNameBase):]
|
|
|
|
fileNameBaseBates := bytes.ToValidUTF8([]byte(fileNameBase)[:maxFilenameLength-len([]byte(fileExt))-8], []byte{})
|
|
|
|
shortFileName := string(fileNameBaseBates) + shaStr[len(shaStr)-8:]
|
|
|
|
return filepath.Join(filepath.Dir(path), shortFileName) + fileExt
|
|
|
|
}
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
|
2022-02-05 03:14:04 +08:00
|
|
|
// Copied from os.WriteFile(), adding file sync.
|
|
|
|
// see https://github.com/golang/go/issues/20599
|
|
|
|
func WriteFile(name string, data []byte, perm os.FileMode) error {
|
|
|
|
f, err := os.OpenFile(name, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, perm)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = f.Write(data)
|
|
|
|
if err1 := f.Sync(); err1 != nil && err == nil {
|
|
|
|
err = err1
|
|
|
|
}
|
|
|
|
if err1 := f.Close(); err1 != nil && err == nil {
|
|
|
|
err = err1
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|