2022-08-24 11:10:50 +08:00
|
|
|
package files
|
|
|
|
|
|
|
|
import (
|
2023-08-25 23:42:14 +08:00
|
|
|
"bufio"
|
2024-04-09 16:36:09 +08:00
|
|
|
"fmt"
|
2022-09-09 18:10:41 +08:00
|
|
|
"github.com/spf13/afero"
|
2024-03-06 11:56:07 +08:00
|
|
|
"io"
|
2023-04-27 21:31:52 +08:00
|
|
|
"net/http"
|
2022-08-24 11:10:50 +08:00
|
|
|
"os"
|
2023-04-26 22:18:16 +08:00
|
|
|
"os/user"
|
2022-09-09 18:10:41 +08:00
|
|
|
"path/filepath"
|
2023-04-26 22:18:16 +08:00
|
|
|
"strconv"
|
2024-04-11 18:32:11 +08:00
|
|
|
"strings"
|
2022-09-09 18:10:41 +08:00
|
|
|
"sync"
|
2022-08-24 11:10:50 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
func IsSymlink(mode os.FileMode) bool {
|
|
|
|
return mode&os.ModeSymlink != 0
|
|
|
|
}
|
|
|
|
|
2024-04-09 14:28:09 +08:00
|
|
|
func IsBlockDevice(mode os.FileMode) bool {
|
|
|
|
return mode&os.ModeDevice != 0 && mode&os.ModeCharDevice == 0
|
|
|
|
}
|
|
|
|
|
2022-08-30 17:59:59 +08:00
|
|
|
func GetMimeType(path string) string {
|
2023-04-27 21:31:52 +08:00
|
|
|
file, err := os.Open(path)
|
2022-08-30 17:59:59 +08:00
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
2023-04-27 21:31:52 +08:00
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
buffer := make([]byte, 512)
|
|
|
|
_, err = file.Read(buffer)
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
mimeType := http.DetectContentType(buffer)
|
|
|
|
return mimeType
|
2022-08-30 17:59:59 +08:00
|
|
|
}
|
2022-08-31 16:00:51 +08:00
|
|
|
|
|
|
|
func GetSymlink(path string) string {
|
|
|
|
linkPath, err := os.Readlink(path)
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return linkPath
|
|
|
|
}
|
2022-09-09 11:20:02 +08:00
|
|
|
|
2023-04-26 22:18:16 +08:00
|
|
|
func GetUsername(uid uint32) string {
|
|
|
|
usr, err := user.LookupId(strconv.Itoa(int(uid)))
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return usr.Username
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetGroup(gid uint32) string {
|
|
|
|
usr, err := user.LookupGroupId(strconv.Itoa(int(gid)))
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return usr.Name
|
|
|
|
}
|
|
|
|
|
2022-09-09 18:10:41 +08:00
|
|
|
func ScanDir(fs afero.Fs, path string, dirMap *sync.Map, wg *sync.WaitGroup) {
|
|
|
|
afs := &afero.Afero{Fs: fs}
|
|
|
|
files, _ := afs.ReadDir(path)
|
|
|
|
for _, f := range files {
|
|
|
|
if f.IsDir() {
|
|
|
|
wg.Add(1)
|
|
|
|
go ScanDir(fs, filepath.Join(path, f.Name()), dirMap, wg)
|
|
|
|
} else {
|
|
|
|
if f.Size() > 0 {
|
|
|
|
dirMap.Store(filepath.Join(path, f.Name()), float64(f.Size()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
defer wg.Done()
|
|
|
|
}
|
|
|
|
|
2022-09-09 11:20:02 +08:00
|
|
|
const dotCharacter = 46
|
|
|
|
|
|
|
|
func IsHidden(path string) bool {
|
|
|
|
return path[0] == dotCharacter
|
|
|
|
}
|
2023-08-25 23:42:14 +08:00
|
|
|
|
|
|
|
func ReadFileByLine(filename string, page, pageSize int) ([]string, bool, error) {
|
2023-11-23 11:00:08 +08:00
|
|
|
if !NewFileOp().Stat(filename) {
|
|
|
|
return nil, true, nil
|
|
|
|
}
|
2023-08-25 23:42:14 +08:00
|
|
|
file, err := os.Open(filename)
|
|
|
|
if err != nil {
|
|
|
|
return nil, false, err
|
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
|
2024-03-06 11:56:07 +08:00
|
|
|
reader := bufio.NewReaderSize(file, 8192)
|
2023-08-25 23:42:14 +08:00
|
|
|
|
|
|
|
var lines []string
|
|
|
|
currentLine := 0
|
|
|
|
startLine := (page - 1) * pageSize
|
|
|
|
endLine := startLine + pageSize
|
|
|
|
|
2024-03-06 11:56:07 +08:00
|
|
|
for {
|
|
|
|
line, _, err := reader.ReadLine()
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
2023-08-25 23:42:14 +08:00
|
|
|
if currentLine >= startLine && currentLine < endLine {
|
2024-03-06 11:56:07 +08:00
|
|
|
lines = append(lines, string(line))
|
2023-08-25 23:42:14 +08:00
|
|
|
}
|
|
|
|
currentLine++
|
|
|
|
if currentLine >= endLine {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
isEndOfFile := currentLine < endLine
|
|
|
|
|
|
|
|
return lines, isEndOfFile, nil
|
|
|
|
}
|
2024-04-09 16:36:09 +08:00
|
|
|
|
|
|
|
func GetParentMode(path string) (os.FileMode, error) {
|
|
|
|
absPath, err := filepath.Abs(path)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
fileInfo, err := os.Stat(absPath)
|
|
|
|
if err == nil {
|
|
|
|
return fileInfo.Mode(), nil
|
|
|
|
}
|
|
|
|
if !os.IsNotExist(err) {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
parentDir := filepath.Dir(absPath)
|
|
|
|
if parentDir == absPath {
|
|
|
|
return 0, fmt.Errorf("no existing directory found in the path: %s", path)
|
|
|
|
}
|
|
|
|
absPath = parentDir
|
|
|
|
}
|
|
|
|
}
|
2024-04-11 18:32:11 +08:00
|
|
|
|
|
|
|
func IsInvalidChar(name string) bool {
|
|
|
|
if strings.Contains(name, "&") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|