afero/mem/file.go

360 lines
7.4 KiB
Go
Raw Permalink Normal View History

// Copyright © 2015 Steve Francia <spf@spf13.com>.
2014-10-28 17:29:28 +03:00
// Copyright 2013 tsuru authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package mem
2014-10-28 17:29:28 +03:00
import (
"bytes"
"errors"
2014-10-28 17:29:28 +03:00
"io"
2022-07-15 13:27:01 +03:00
"io/fs"
2014-10-28 17:29:28 +03:00
"os"
2015-12-09 23:37:04 +03:00
"path/filepath"
2015-03-22 03:24:08 +03:00
"sync"
2014-10-28 17:29:28 +03:00
"sync/atomic"
2020-12-08 21:57:36 +03:00
"time"
2022-07-15 13:27:01 +03:00
"github.com/spf13/afero/internal/common"
2014-10-28 17:29:28 +03:00
)
2015-12-10 21:00:25 +03:00
const FilePathSeparator = string(filepath.Separator)
2022-07-15 13:27:01 +03:00
var _ fs.ReadDirFile = &File{}
type File struct {
// atomic requires 64-bit alignment for struct field access
at int64
readDirCount int64
closed bool
readOnly bool
fileData *FileData
}
func NewFileHandle(data *FileData) *File {
return &File{fileData: data}
}
func NewReadOnlyFileHandle(data *FileData) *File {
return &File{fileData: data, readOnly: true}
}
func (f File) Data() *FileData {
return f.fileData
}
type FileData struct {
sync.Mutex
2015-12-10 21:00:25 +03:00
name string
data []byte
memDir Dir
dir bool
mode os.FileMode
modtime time.Time
2020-01-22 17:25:31 +03:00
uid int
gid int
2015-12-10 21:00:25 +03:00
}
func (d *FileData) Name() string {
d.Lock()
defer d.Unlock()
return d.name
}
func CreateFile(name string) *FileData {
return &FileData{name: name, mode: os.ModeTemporary, modtime: time.Now()}
2015-12-10 21:00:25 +03:00
}
func CreateDir(name string) *FileData {
2022-01-10 13:50:49 +03:00
return &FileData{name: name, memDir: &DirMap{}, dir: true, modtime: time.Now()}
2015-12-10 21:00:25 +03:00
}
func ChangeFileName(f *FileData, newname string) {
f.Lock()
f.name = newname
f.Unlock()
2015-12-10 21:00:25 +03:00
}
func SetMode(f *FileData, mode os.FileMode) {
f.Lock()
f.mode = mode
f.Unlock()
2015-12-10 21:00:25 +03:00
}
func SetModTime(f *FileData, mtime time.Time) {
f.Lock()
setModTime(f, mtime)
f.Unlock()
}
func setModTime(f *FileData, mtime time.Time) {
f.modtime = mtime
}
2020-01-22 17:25:31 +03:00
func SetUID(f *FileData, uid int) {
f.Lock()
f.uid = uid
f.Unlock()
}
func SetGID(f *FileData, gid int) {
f.Lock()
2020-12-08 21:57:36 +03:00
f.gid = gid
2020-01-22 17:25:31 +03:00
f.Unlock()
}
func GetFileInfo(f *FileData) *FileInfo {
return &FileInfo{f}
}
func (f *File) Open() error {
2014-11-01 06:38:54 +03:00
atomic.StoreInt64(&f.at, 0)
atomic.StoreInt64(&f.readDirCount, 0)
f.fileData.Lock()
2014-11-01 06:38:54 +03:00
f.closed = false
f.fileData.Unlock()
2014-11-01 06:38:54 +03:00
return nil
2014-10-28 17:29:28 +03:00
}
func (f *File) Close() error {
f.fileData.Lock()
2014-10-28 17:29:28 +03:00
f.closed = true
2016-01-04 21:34:24 +03:00
if !f.readOnly {
setModTime(f.fileData, time.Now())
2016-01-04 21:34:24 +03:00
}
f.fileData.Unlock()
2014-10-28 17:29:28 +03:00
return nil
}
func (f *File) Name() string {
return f.fileData.Name()
2014-10-28 17:29:28 +03:00
}
func (f *File) Stat() (os.FileInfo, error) {
return &FileInfo{f.fileData}, nil
2014-10-28 17:29:28 +03:00
}
func (f *File) Sync() error {
return nil
}
func (f *File) Readdir(count int) (res []os.FileInfo, err error) {
if !f.fileData.dir {
return nil, &os.PathError{Op: "readdir", Path: f.fileData.name, Err: errors.New("not a dir")}
}
var outLength int64
2014-10-28 17:29:28 +03:00
f.fileData.Lock()
files := f.fileData.memDir.Files()[f.readDirCount:]
2014-10-28 17:29:28 +03:00
if count > 0 {
if len(files) < count {
outLength = int64(len(files))
} else {
outLength = int64(count)
}
if len(files) == 0 {
err = io.EOF
}
} else {
outLength = int64(len(files))
2014-10-28 17:29:28 +03:00
}
f.readDirCount += outLength
f.fileData.Unlock()
2014-10-28 17:29:28 +03:00
res = make([]os.FileInfo, outLength)
for i := range res {
res[i] = &FileInfo{files[i]}
2014-10-28 17:29:28 +03:00
}
return res, err
2014-10-28 17:29:28 +03:00
}
func (f *File) Readdirnames(n int) (names []string, err error) {
2014-10-28 17:29:28 +03:00
fi, err := f.Readdir(n)
names = make([]string, len(fi))
for i, f := range fi {
2015-12-09 23:37:04 +03:00
_, names[i] = filepath.Split(f.Name())
2014-10-28 17:29:28 +03:00
}
return names, err
}
2022-07-15 13:27:01 +03:00
// Implements fs.ReadDirFile
func (f *File) ReadDir(n int) ([]fs.DirEntry, error) {
fi, err := f.Readdir(n)
if err != nil {
return nil, err
}
di := make([]fs.DirEntry, len(fi))
for i, f := range fi {
di[i] = common.FileInfoDirEntry{FileInfo: f}
}
return di, nil
}
func (f *File) Read(b []byte) (n int, err error) {
f.fileData.Lock()
defer f.fileData.Unlock()
2022-07-14 14:06:04 +03:00
if f.closed {
2014-10-28 17:29:28 +03:00
return 0, ErrFileClosed
}
if len(b) > 0 && int(f.at) == len(f.fileData.data) {
2015-03-22 03:24:08 +03:00
return 0, io.EOF
}
if int(f.at) > len(f.fileData.data) {
return 0, io.ErrUnexpectedEOF
}
if len(f.fileData.data)-int(f.at) >= len(b) {
2014-10-28 17:29:28 +03:00
n = len(b)
} else {
n = len(f.fileData.data) - int(f.at)
2014-10-28 17:29:28 +03:00
}
copy(b, f.fileData.data[f.at:f.at+int64(n)])
2014-10-28 17:29:28 +03:00
atomic.AddInt64(&f.at, int64(n))
return
}
func (f *File) ReadAt(b []byte, off int64) (n int, err error) {
2018-11-20 21:34:41 +03:00
prev := atomic.LoadInt64(&f.at)
2014-10-28 17:29:28 +03:00
atomic.StoreInt64(&f.at, off)
2018-11-20 21:34:41 +03:00
n, err = f.Read(b)
atomic.StoreInt64(&f.at, prev)
return
2014-10-28 17:29:28 +03:00
}
func (f *File) Truncate(size int64) error {
2022-07-14 14:06:04 +03:00
if f.closed {
2014-10-28 17:29:28 +03:00
return ErrFileClosed
}
2016-01-04 21:34:24 +03:00
if f.readOnly {
2017-01-10 01:53:20 +03:00
return &os.PathError{Op: "truncate", Path: f.fileData.name, Err: errors.New("file handle is read only")}
2016-01-04 21:34:24 +03:00
}
2014-10-28 17:29:28 +03:00
if size < 0 {
return ErrOutOfRange
}
f.fileData.Lock()
defer f.fileData.Unlock()
if size > int64(len(f.fileData.data)) {
diff := size - int64(len(f.fileData.data))
2023-02-23 12:14:33 +03:00
f.fileData.data = append(f.fileData.data, bytes.Repeat([]byte{0o0}, int(diff))...)
2014-10-28 17:29:28 +03:00
} else {
f.fileData.data = f.fileData.data[0:size]
2014-10-28 17:29:28 +03:00
}
setModTime(f.fileData, time.Now())
2014-10-28 17:29:28 +03:00
return nil
}
func (f *File) Seek(offset int64, whence int) (int64, error) {
2022-07-14 14:06:04 +03:00
if f.closed {
2014-10-28 17:29:28 +03:00
return 0, ErrFileClosed
}
switch whence {
case io.SeekStart:
2014-10-28 17:29:28 +03:00
atomic.StoreInt64(&f.at, offset)
case io.SeekCurrent:
atomic.AddInt64(&f.at, offset)
case io.SeekEnd:
atomic.StoreInt64(&f.at, int64(len(f.fileData.data))+offset)
2014-10-28 17:29:28 +03:00
}
return f.at, nil
}
func (f *File) Write(b []byte) (n int, err error) {
2022-07-14 14:06:04 +03:00
if f.closed {
return 0, ErrFileClosed
}
if f.readOnly {
2017-01-10 01:53:20 +03:00
return 0, &os.PathError{Op: "write", Path: f.fileData.name, Err: errors.New("file handle is read only")}
}
2014-10-28 17:29:28 +03:00
n = len(b)
cur := atomic.LoadInt64(&f.at)
f.fileData.Lock()
defer f.fileData.Unlock()
diff := cur - int64(len(f.fileData.data))
2014-10-28 17:29:28 +03:00
var tail []byte
if n+int(cur) < len(f.fileData.data) {
tail = f.fileData.data[n+int(cur):]
2014-10-28 17:29:28 +03:00
}
if diff > 0 {
2023-02-23 12:14:33 +03:00
f.fileData.data = append(f.fileData.data, append(bytes.Repeat([]byte{0o0}, int(diff)), b...)...)
f.fileData.data = append(f.fileData.data, tail...)
2014-10-28 17:29:28 +03:00
} else {
f.fileData.data = append(f.fileData.data[:cur], b...)
f.fileData.data = append(f.fileData.data, tail...)
2014-10-28 17:29:28 +03:00
}
setModTime(f.fileData, time.Now())
2014-10-28 17:29:28 +03:00
atomic.AddInt64(&f.at, int64(n))
2014-10-28 17:29:28 +03:00
return
}
func (f *File) WriteAt(b []byte, off int64) (n int, err error) {
2014-10-28 17:29:28 +03:00
atomic.StoreInt64(&f.at, off)
return f.Write(b)
}
func (f *File) WriteString(s string) (ret int, err error) {
2014-10-28 17:29:28 +03:00
return f.Write([]byte(s))
}
func (f *File) Info() *FileInfo {
return &FileInfo{f.fileData}
2014-10-28 17:29:28 +03:00
}
type FileInfo struct {
*FileData
2014-10-28 17:29:28 +03:00
}
// Implements os.FileInfo
func (s *FileInfo) Name() string {
s.Lock()
_, name := filepath.Split(s.name)
s.Unlock()
return name
}
2023-02-23 12:14:33 +03:00
func (s *FileInfo) Mode() os.FileMode {
s.Lock()
defer s.Unlock()
return s.mode
}
2023-02-23 12:14:33 +03:00
func (s *FileInfo) ModTime() time.Time {
s.Lock()
defer s.Unlock()
return s.modtime
}
2023-02-23 12:14:33 +03:00
func (s *FileInfo) IsDir() bool {
s.Lock()
defer s.Unlock()
return s.dir
}
func (s *FileInfo) Sys() interface{} { return nil }
func (s *FileInfo) Size() int64 {
2014-11-01 06:39:14 +03:00
if s.IsDir() {
return int64(42)
}
s.Lock()
defer s.Unlock()
return int64(len(s.data))
2014-11-01 06:39:14 +03:00
}
var (
ErrFileClosed = errors.New("File is closed")
2022-07-14 14:06:04 +03:00
ErrOutOfRange = errors.New("out of range")
ErrTooLarge = errors.New("too large")
ErrFileNotFound = os.ErrNotExist
ErrFileExists = os.ErrExist
ErrDestinationExists = os.ErrExist
)