2017-12-29 16:03:30 +08:00
|
|
|
|
// 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.
|
|
|
|
|
|
2018-03-29 13:46:05 +08:00
|
|
|
|
// 文件管理.
|
2017-11-23 10:21:28 +08:00
|
|
|
|
package gfile
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"os"
|
|
|
|
|
"io"
|
|
|
|
|
"io/ioutil"
|
|
|
|
|
"sort"
|
|
|
|
|
"fmt"
|
|
|
|
|
"time"
|
|
|
|
|
"strings"
|
|
|
|
|
"bytes"
|
|
|
|
|
"os/exec"
|
|
|
|
|
"errors"
|
|
|
|
|
"os/user"
|
|
|
|
|
"runtime"
|
2018-05-03 16:12:01 +08:00
|
|
|
|
"path/filepath"
|
|
|
|
|
"gitee.com/johng/gf/g/util/gregx"
|
|
|
|
|
"gitee.com/johng/gf/g/container/gtype"
|
2017-11-23 10:21:28 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// 封装了常用的文件操作方法,如需更详细的文件控制,请查看官方os包
|
|
|
|
|
|
|
|
|
|
// 文件分隔符
|
2018-04-19 14:58:25 +08:00
|
|
|
|
const (
|
|
|
|
|
Separator = string(filepath.Separator)
|
|
|
|
|
)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
|
2018-05-03 16:12:01 +08:00
|
|
|
|
// 源码的main包所在目录,仅仅会设置一次
|
|
|
|
|
var mainPkgPath = gtype.NewInterface()
|
|
|
|
|
|
2017-11-23 10:21:28 +08:00
|
|
|
|
// 给定文件的绝对路径创建文件
|
|
|
|
|
func Mkdir(path string) error {
|
|
|
|
|
err := os.MkdirAll(path, os.ModePerm)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 给定文件的绝对路径创建文件
|
|
|
|
|
func Create(path string) error {
|
|
|
|
|
f, err := os.Create(path)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
f.Close()
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 打开文件
|
|
|
|
|
func Open(path string) (*os.File, error) {
|
2018-01-11 16:06:42 +08:00
|
|
|
|
f, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE, 0666)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return f, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 打开文件
|
|
|
|
|
func OpenWithFlag(path string, flag int) (*os.File, error) {
|
2018-01-11 16:06:42 +08:00
|
|
|
|
f, err := os.OpenFile(path, flag, 0666)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
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 IsFile(path string) bool {
|
|
|
|
|
return !IsDir(path)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 获取文件或目录信息
|
|
|
|
|
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()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 文件大小(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 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) []string {
|
|
|
|
|
f, err := os.Open(path)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
list, err := f.Readdirnames(-1)
|
|
|
|
|
f.Close()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
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) {
|
|
|
|
|
Mkdir(dir)
|
|
|
|
|
}
|
|
|
|
|
// 创建/打开文件
|
|
|
|
|
f, err := os.OpenFile(path, flag, perm)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
defer f.Close()
|
|
|
|
|
n, err := f.Write(data)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
} else if n < len(data) {
|
|
|
|
|
return io.ErrShortWrite
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// (文本)写入文件内容
|
|
|
|
|
func PutContents(path string, content string) error {
|
2018-01-11 16:06:42 +08:00
|
|
|
|
return putContents(path, []byte(content), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0666)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// (文本)追加内容到文件末尾
|
|
|
|
|
func PutContentsAppend(path string, content string) error {
|
2018-01-11 16:06:42 +08:00
|
|
|
|
return putContents(path, []byte(content), os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// (二进制)写入文件内容
|
|
|
|
|
func PutBinContents(path string, content []byte) error {
|
2018-01-11 16:06:42 +08:00
|
|
|
|
return putContents(path, content, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0666)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// (二进制)追加内容到文件末尾
|
|
|
|
|
func PutBinContentsAppend(path string, content []byte) error {
|
2018-01-11 16:06:42 +08:00
|
|
|
|
return putContents(path, content, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666)
|
2017-11-23 10:21:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 获取当前执行文件的绝对路径
|
|
|
|
|
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
|
2018-05-03 16:12:01 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 获取入口函数文件所在目录(main包文件目录),仅对源码开发环境有效(即仅对生成该可执行文件的系统下有效)
|
|
|
|
|
func MainPkgPath() string {
|
|
|
|
|
path := mainPkgPath.Val()
|
|
|
|
|
if path != nil {
|
|
|
|
|
return path.(string)
|
|
|
|
|
}
|
|
|
|
|
f := ""
|
|
|
|
|
for i := 1; i < 10000; i++ {
|
|
|
|
|
if _, file, _, ok := runtime.Caller(i); ok {
|
|
|
|
|
// 不包含go源码路径
|
|
|
|
|
if !gregx.IsMatchString("^" + runtime.GOROOT(), file) {
|
|
|
|
|
f = file
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if f != "" {
|
|
|
|
|
p := Dir(f)
|
|
|
|
|
mainPkgPath.Set(p)
|
|
|
|
|
return p
|
|
|
|
|
}
|
|
|
|
|
return ""
|
2017-11-23 10:21:28 +08:00
|
|
|
|
}
|