gf/g/os/gfile/gfile.go

465 lines
10 KiB
Go
Raw Normal View History

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"
"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
// 源码的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) {
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) {
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 {
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 {
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 {
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 {
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
}
// 获取入口函数文件所在目录(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
}