mirror of
https://gitee.com/johng/gf.git
synced 2024-12-04 05:07:44 +08:00
554 lines
12 KiB
Go
554 lines
12 KiB
Go
// Copyright 2017 gf Author(https://gitee.com/johng/gf). All Rights Reserved.
|
||
//
|
||
// This Source Code Form is subject to the terms of the MIT License.
|
||
// If a copy of the MIT was not distributed with this file,
|
||
// You can obtain one at https://gitee.com/johng/gf.
|
||
|
||
// 文件管理.
|
||
package gfile
|
||
|
||
import (
|
||
"os"
|
||
"io"
|
||
"io/ioutil"
|
||
"sort"
|
||
"fmt"
|
||
"time"
|
||
"strings"
|
||
"bytes"
|
||
"os/exec"
|
||
"errors"
|
||
"os/user"
|
||
"runtime"
|
||
"path/filepath"
|
||
"gitee.com/johng/gf/g/util/gregex"
|
||
"gitee.com/johng/gf/g/container/gtype"
|
||
"gitee.com/johng/gf/g/os/gfilepool"
|
||
)
|
||
|
||
// 封装了常用的文件操作方法,如需更详细的文件控制,请查看官方os包
|
||
|
||
|
||
// 文件分隔符
|
||
const (
|
||
Separator = string(filepath.Separator)
|
||
)
|
||
|
||
var (
|
||
// 源码的main包所在目录,仅仅会设置一次
|
||
mainPkgPath = gtype.NewString()
|
||
|
||
// 编译时的 GOROOT 数值
|
||
goRootOfBuild = gtype.NewString()
|
||
)
|
||
|
||
// 给定文件的绝对路径创建文件
|
||
func Mkdir(path string) error {
|
||
err := os.MkdirAll(path, os.ModePerm)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// 给定文件的绝对路径创建文件
|
||
func Create(path string) error {
|
||
dir := Dir(path)
|
||
if !Exists(dir) {
|
||
Mkdir(dir)
|
||
}
|
||
f, err := os.Create(path)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
f.Close()
|
||
return nil
|
||
}
|
||
|
||
// 打开文件
|
||
func Open(path string) (*os.File, error) {
|
||
f, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE, 0666)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
return f, nil
|
||
}
|
||
|
||
// 打开文件
|
||
func OpenWithFlag(path string, flag int) (*os.File, error) {
|
||
f, err := os.OpenFile(path, flag, 0666)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
return f, nil
|
||
}
|
||
|
||
// 判断所给路径文件/文件夹是否存在
|
||
func Exists(path string) bool {
|
||
_, err := os.Stat(path)
|
||
if err != nil {
|
||
if os.IsExist(err) {
|
||
return true
|
||
}
|
||
return false
|
||
}
|
||
return true
|
||
}
|
||
|
||
// 判断所给路径是否为文件夹
|
||
func IsDir(path string) bool {
|
||
s, err := os.Stat(path)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
return s.IsDir()
|
||
}
|
||
|
||
// 获取当前工作目录
|
||
func Pwd() string {
|
||
pwd, _ := filepath.Abs(filepath.Dir(os.Args[0]))
|
||
return pwd
|
||
}
|
||
|
||
// 判断所给路径是否为文件
|
||
func IsFile(path string) bool {
|
||
s, err := os.Stat(path)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
return !s.IsDir()
|
||
}
|
||
|
||
// 获取文件或目录信息
|
||
func Info(path string) *os.FileInfo {
|
||
info, err := os.Stat(path)
|
||
if err != nil {
|
||
return nil
|
||
}
|
||
return &info
|
||
}
|
||
|
||
// 修改时间(秒)
|
||
func MTime(path string) int64 {
|
||
f, e := os.Stat(path)
|
||
if e != nil {
|
||
return 0
|
||
}
|
||
return f.ModTime().Unix()
|
||
}
|
||
|
||
// 修改时间(毫秒)
|
||
func MTimeMillisecond(path string) int64 {
|
||
f, e := os.Stat(path)
|
||
if e != nil {
|
||
return 0
|
||
}
|
||
return int64(f.ModTime().Nanosecond()/1000000)
|
||
}
|
||
|
||
// 文件大小(bytes)
|
||
func Size(path string) int64 {
|
||
f, e := os.Stat(path)
|
||
if e != nil {
|
||
return 0
|
||
}
|
||
return f.Size()
|
||
}
|
||
|
||
// 格式化文件大小
|
||
func ReadableSize(path string) string {
|
||
return FormatSize(float64(Size(path)))
|
||
}
|
||
|
||
// 格式化文件大小
|
||
func FormatSize(raw float64) string {
|
||
var t float64 = 1024
|
||
var d float64 = 1
|
||
|
||
if raw < t {
|
||
return fmt.Sprintf("%.2fB", raw/d)
|
||
}
|
||
|
||
d *= 1024
|
||
t *= 1024
|
||
|
||
if raw < t {
|
||
return fmt.Sprintf("%.2fK", raw/d)
|
||
}
|
||
|
||
d *= 1024
|
||
t *= 1024
|
||
|
||
if raw < t {
|
||
return fmt.Sprintf("%.2fM", raw/d)
|
||
}
|
||
|
||
d *= 1024
|
||
t *= 1024
|
||
|
||
if raw < t {
|
||
return fmt.Sprintf("%.2fG", raw/d)
|
||
}
|
||
|
||
d *= 1024
|
||
t *= 1024
|
||
|
||
if raw < t {
|
||
return fmt.Sprintf("%.2fT", raw/d)
|
||
}
|
||
|
||
d *= 1024
|
||
t *= 1024
|
||
|
||
if raw < t {
|
||
return fmt.Sprintf("%.2fP", raw/d)
|
||
}
|
||
|
||
return "TooLarge"
|
||
}
|
||
|
||
// 文件移动/重命名
|
||
func Move(src string, dst string) error {
|
||
return os.Rename(src, dst)
|
||
}
|
||
|
||
|
||
// 文件移动/重命名
|
||
func Rename(src string, dst string) error {
|
||
return Move(src, dst)
|
||
}
|
||
|
||
// 文件复制
|
||
func Copy(src string, dst string) error {
|
||
srcFile, err := os.Open(src)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
dstFile, err := os.Create(dst)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
_, err = io.Copy(dstFile, srcFile)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
err = dstFile.Sync()
|
||
if err != nil {
|
||
return err
|
||
}
|
||
srcFile.Close()
|
||
dstFile.Close()
|
||
return nil
|
||
}
|
||
|
||
// 文件名正则匹配查找
|
||
func Glob(pattern string) ([]string, error) {
|
||
return filepath.Glob(pattern)
|
||
}
|
||
|
||
// 文件/目录删除
|
||
func Remove(path string) error {
|
||
return os.RemoveAll(path)
|
||
}
|
||
|
||
// 文件是否可
|
||
func IsReadable(path string) bool {
|
||
result := true
|
||
file, err := os.OpenFile(path, os.O_RDONLY, 0666)
|
||
if err != nil {
|
||
result = false
|
||
}
|
||
file.Close()
|
||
return result
|
||
}
|
||
|
||
// 文件是否可写
|
||
func IsWritable(path string) bool {
|
||
result := true
|
||
if IsDir(path) {
|
||
// 如果是目录,那么创建一个临时文件进行写入测试
|
||
tfile := strings.TrimRight(path, Separator) + Separator + string(time.Now().UnixNano())
|
||
err := Create(tfile)
|
||
if err != nil || !Exists(tfile){
|
||
result = false
|
||
} else {
|
||
Remove(tfile)
|
||
}
|
||
} else {
|
||
// 如果是文件,那么判断文件是否可打开
|
||
file, err := os.OpenFile(path, os.O_WRONLY, 0666)
|
||
if err != nil {
|
||
result = false
|
||
}
|
||
file.Close()
|
||
}
|
||
return result
|
||
}
|
||
|
||
// 修改文件/目录权限
|
||
func Chmod(path string, mode os.FileMode) error {
|
||
return os.Chmod(path, mode)
|
||
}
|
||
|
||
// 打开目录,并返回其下一级子目录名称列表,按照文件名称大小写进行排序,支持目录递归遍历。
|
||
// 当递归遍历时,结果集返回的是子级文件/目录的绝对路径,而不仅仅是一个名字
|
||
func ScanDir(path string, recursive ... bool) []string {
|
||
f, err := os.Open(path)
|
||
if err != nil {
|
||
return nil
|
||
}
|
||
|
||
list, err := f.Readdirnames(-1)
|
||
f.Close()
|
||
if err != nil {
|
||
return nil
|
||
}
|
||
// 是否递归遍历
|
||
if len(recursive) > 0 && recursive[0] && len(list) > 0 {
|
||
for k, v := range list {
|
||
p := fmt.Sprintf("%s%s%s", path, Separator, v)
|
||
list[k] = p
|
||
if IsDir(p) {
|
||
list = append(list, ScanDir(p, true)...)
|
||
}
|
||
}
|
||
}
|
||
// 默认按照字符串大小排序
|
||
sort.Slice(list, func(i, j int) bool { return list[i] < list[j] })
|
||
return list
|
||
}
|
||
|
||
// 将所给定的路径转换为绝对路径
|
||
// 并判断文件路径是否存在,如果文件不存在,那么返回空字符串
|
||
func RealPath(path string) string {
|
||
p, err := filepath.Abs(path)
|
||
if err != nil {
|
||
return ""
|
||
}
|
||
if !Exists(p) {
|
||
return ""
|
||
}
|
||
return p
|
||
}
|
||
|
||
// (文本)读取文件内容
|
||
func GetContents(path string) string {
|
||
return string(GetBinContents(path))
|
||
}
|
||
|
||
// (二进制)读取文件内容
|
||
func GetBinContents(path string) []byte {
|
||
data, err := ioutil.ReadFile(path)
|
||
if err != nil {
|
||
return nil
|
||
}
|
||
return data
|
||
}
|
||
|
||
// 写入文件内容
|
||
func putContents(path string, data []byte, flag int, perm os.FileMode) error {
|
||
// 支持目录递归创建
|
||
dir := Dir(path)
|
||
if !Exists(dir) {
|
||
if err := Mkdir(dir); err != nil {
|
||
return err
|
||
}
|
||
}
|
||
// 创建/打开文件,使用文件指针池,默认为60秒
|
||
f, err := gfilepool.OpenWithPool(path, flag, perm, 60000)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
defer f.Close()
|
||
if n, err := f.Write(data); err != nil {
|
||
return err
|
||
} else if n < len(data) {
|
||
return io.ErrShortWrite
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// Truncate
|
||
func Truncate(path string, size int) error {
|
||
return os.Truncate(path, int64(size))
|
||
}
|
||
|
||
// (文本)写入文件内容
|
||
func PutContents(path string, content string) error {
|
||
return putContents(path, []byte(content), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0666)
|
||
}
|
||
|
||
// (文本)追加内容到文件末尾
|
||
func PutContentsAppend(path string, content string) error {
|
||
return putContents(path, []byte(content), os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666)
|
||
}
|
||
|
||
// (二进制)写入文件内容
|
||
func PutBinContents(path string, content []byte) error {
|
||
return putContents(path, content, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0666)
|
||
}
|
||
|
||
// (二进制)追加内容到文件末尾
|
||
func PutBinContentsAppend(path string, content []byte) error {
|
||
return putContents(path, content, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666)
|
||
}
|
||
|
||
|
||
// 获取当前执行文件的绝对路径
|
||
func SelfPath() string {
|
||
p, _ := filepath.Abs(os.Args[0])
|
||
return p
|
||
}
|
||
|
||
// 获取当前执行文件的目录绝对路径
|
||
func SelfDir() string {
|
||
return filepath.Dir(SelfPath())
|
||
}
|
||
|
||
// 获取指定文件路径的文件名称
|
||
func Basename(path string) string {
|
||
return filepath.Base(path)
|
||
}
|
||
|
||
// 获取指定文件路径的目录地址绝对路径
|
||
func Dir(path string) string {
|
||
return filepath.Dir(path)
|
||
}
|
||
|
||
// 获取指定文件路径的文件扩展名
|
||
func Ext(path string) string {
|
||
return filepath.Ext(path)
|
||
}
|
||
|
||
// 获取用户主目录
|
||
func Home() (string, error) {
|
||
u, err := user.Current()
|
||
if nil == err {
|
||
return u.HomeDir, nil
|
||
}
|
||
if "windows" == runtime.GOOS {
|
||
return homeWindows()
|
||
}
|
||
return homeUnix()
|
||
}
|
||
|
||
func homeUnix() (string, error) {
|
||
if home := os.Getenv("HOME"); home != "" {
|
||
return home, nil
|
||
}
|
||
var stdout bytes.Buffer
|
||
cmd := exec.Command("sh", "-c", "eval echo ~$USER")
|
||
cmd.Stdout = &stdout
|
||
if err := cmd.Run(); err != nil {
|
||
return "", err
|
||
}
|
||
|
||
result := strings.TrimSpace(stdout.String())
|
||
if result == "" {
|
||
return "", errors.New("blank output when reading home directory")
|
||
}
|
||
|
||
return result, nil
|
||
}
|
||
|
||
func homeWindows() (string, error) {
|
||
drive := os.Getenv("HOMEDRIVE")
|
||
path := os.Getenv("HOMEPATH")
|
||
home := drive + path
|
||
if drive == "" || path == "" {
|
||
home = os.Getenv("USERPROFILE")
|
||
}
|
||
if home == "" {
|
||
return "", errors.New("HOMEDRIVE, HOMEPATH, and USERPROFILE are blank")
|
||
}
|
||
|
||
return home, nil
|
||
}
|
||
|
||
// 获得文件内容下一个指定字节的位置
|
||
func GetNextCharOffset(file *os.File, char string, start int64) int64 {
|
||
c := []byte(char)[0]
|
||
b := make([]byte, 1)
|
||
o := start
|
||
for {
|
||
_, err := file.ReadAt(b, o)
|
||
if err != nil {
|
||
return 0
|
||
}
|
||
if b[0] == c {
|
||
return o
|
||
}
|
||
o++
|
||
}
|
||
return 0
|
||
}
|
||
|
||
// 获得文件内容中两个offset之间的内容 [start, end)
|
||
func GetBinContentByTwoOffsets(file *os.File, start int64, end int64) []byte {
|
||
buffer := make([]byte, end - start)
|
||
if _, err := file.ReadAt(buffer, start); err != nil {
|
||
return nil
|
||
}
|
||
return buffer
|
||
}
|
||
|
||
// 获取入口函数文件所在目录(main包文件目录),
|
||
// **仅对源码开发环境有效(即仅对生成该可执行文件的系统下有效)**
|
||
func MainPkgPath() string {
|
||
path := mainPkgPath.Val()
|
||
if path != "" {
|
||
return path
|
||
}
|
||
f := ""
|
||
for i := 1; i < 10000; i++ {
|
||
if _, file, _, ok := runtime.Caller(i); ok {
|
||
if strings.EqualFold("<autogenerated>", file) {
|
||
// 如果是通过init包方法进入,那么无法得到准确的文件路径
|
||
f = ""
|
||
} else if !gregex.IsMatchString("^" + GoRootOfBuild(), file) {
|
||
// 不包含go源码路径
|
||
f = file
|
||
}
|
||
} else {
|
||
break
|
||
}
|
||
}
|
||
if f != "" {
|
||
p := Dir(f)
|
||
mainPkgPath.Set(p)
|
||
return p
|
||
}
|
||
return ""
|
||
}
|
||
|
||
// 编译时环境的GOROOT数值
|
||
func GoRootOfBuild() string {
|
||
if v := goRootOfBuild.Val(); v != "" {
|
||
return v
|
||
}
|
||
firstEntry := ""
|
||
for i := 0; i < 10000; i++ {
|
||
if _, file, _, ok := runtime.Caller(i); ok {
|
||
firstEntry = file
|
||
} else {
|
||
break
|
||
}
|
||
}
|
||
if len(firstEntry) > 0 {
|
||
sep := "/"
|
||
array := strings.Split(firstEntry, sep)
|
||
if len(array) == 1 {
|
||
sep = "\\"
|
||
array = strings.Split(firstEntry, sep)
|
||
}
|
||
root := strings.Join(array[0 : len(array) - 3], sep)
|
||
goRootOfBuild.Set(root)
|
||
return root
|
||
}
|
||
return ""
|
||
}
|
||
|
||
// 系统临时目录
|
||
func TempDir() string {
|
||
return os.TempDir()
|
||
} |