serialfile.go 2.86 KB
Newer Older
1 2 3
package files

import (
4
	"fmt"
5
	"io"
rht's avatar
rht committed
6
	"io/ioutil"
7
	"os"
gatesvp's avatar
gatesvp committed
8
	fp "path/filepath"
9 10 11 12 13 14 15
	"syscall"
)

// serialFile implements File, and reads from a path on the OS filesystem.
// No more than one file will be opened at a time (directories will advance
// to the next file when NextFile() is called).
type serialFile struct {
16
	name    string
17 18
	path    string
	files   []os.FileInfo
19
	stat    os.FileInfo
rht's avatar
rht committed
20
	current *File
21 22
}

23
func NewSerialFile(name, path string, stat os.FileInfo) (File, error) {
24
	switch mode := stat.Mode(); {
rht's avatar
rht committed
25 26 27 28 29 30 31 32 33 34 35 36 37 38
	case mode.IsRegular():
		file, err := os.Open(path)
		if err != nil {
			return nil, err
		}
		return NewReaderFile(name, path, file, stat), nil
	case mode.IsDir():
		// for directories, stat all of the contents first, so we know what files to
		// open when NextFile() is called
		contents, err := ioutil.ReadDir(path)
		if err != nil {
			return nil, err
		}
		return &serialFile{name, path, contents, stat, nil}, nil
39
	case mode&os.ModeSymlink != 0:
40 41 42 43
		target, err := os.Readlink(path)
		if err != nil {
			return nil, err
		}
rht's avatar
rht committed
44
		return NewLinkFile(name, path, target, stat), nil
45
	default:
rht's avatar
rht committed
46
		return nil, fmt.Errorf("Unrecognized file type for %s: %s", name, mode.String())
47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71
	}
}

func (f *serialFile) IsDirectory() bool {
	// non-directories get created as a ReaderFile, so serialFiles should only
	// represent directories
	return true
}

func (f *serialFile) NextFile() (File, error) {
	// if a file was opened previously, close it
	err := f.Close()
	if err != nil {
		return nil, err
	}

	// if there aren't any files left in the root directory, we're done
	if len(f.files) == 0 {
		return nil, io.EOF
	}

	stat := f.files[0]
	f.files = f.files[1:]

	// open the next file
72
	fileName := fp.Join(f.name, stat.Name())
73
	filePath := fp.Join(f.path, stat.Name())
74

rht's avatar
rht committed
75 76 77 78
	// recursively call the constructor on the next file
	// if it's a regular file, we will open it as a ReaderFile
	// if it's a directory, files in it will be opened serially
	sf, err := NewSerialFile(fileName, filePath, stat)
79 80 81
	if err != nil {
		return nil, err
	}
82

rht's avatar
rht committed
83
	f.current = &sf
84

rht's avatar
rht committed
85
	return sf, nil
86 87 88
}

func (f *serialFile) FileName() string {
89 90 91 92
	return f.name
}

func (f *serialFile) FullPath() string {
93 94 95 96 97 98 99 100 101 102
	return f.path
}

func (f *serialFile) Read(p []byte) (int, error) {
	return 0, ErrNotReader
}

func (f *serialFile) Close() error {
	// close the current file if there is one
	if f.current != nil {
rht's avatar
rht committed
103
		err := (*f.current).Close()
104 105 106 107 108 109 110 111
		// ignore EINVAL error, the file might have already been closed
		if err != nil && err != syscall.EINVAL {
			return err
		}
	}

	return nil
}
112 113 114 115

func (f *serialFile) Stat() os.FileInfo {
	return f.stat
}
116 117

func (f *serialFile) Size() (int64, error) {
118 119
	if !f.stat.IsDir() {
		return f.stat.Size(), nil
120 121
	}

122 123 124 125
	var du int64
	err := fp.Walk(f.FileName(), func(p string, fi os.FileInfo, err error) error {
		if fi != nil && fi.Mode()&(os.ModeSymlink|os.ModeNamedPipe) == 0 {
			du += fi.Size()
126
		}
127 128 129
		return nil
	})
	return du, err
130
}