| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 | // Copyright (C) 2016 The Syncthing Authors.//// This Source Code Form is subject to the terms of the Mozilla Public// License, v. 2.0. If a copy of the MPL was not distributed with this file,// You can obtain one at https://mozilla.org/MPL/2.0/.package fsimport (	"context"	"errors"	"io"	"os"	"path/filepath"	"strings"	"time")// The Filesystem interface abstracts access to the file system.type Filesystem interface {	Chmod(name string, mode FileMode) error	Lchown(name string, uid, gid int) error	Chtimes(name string, atime time.Time, mtime time.Time) error	Create(name string) (File, error)	CreateSymlink(target, name string) error	DirNames(name string) ([]string, error)	Lstat(name string) (FileInfo, error)	Mkdir(name string, perm FileMode) error	MkdirAll(name string, perm FileMode) error	Open(name string) (File, error)	OpenFile(name string, flags int, mode FileMode) (File, error)	ReadSymlink(name string) (string, error)	Remove(name string) error	RemoveAll(name string) error	Rename(oldname, newname string) error	Stat(name string) (FileInfo, error)	SymlinksSupported() bool	Walk(name string, walkFn WalkFunc) error	// If setup fails, returns non-nil error, and if afterwards a fatal (!)	// error occurs, sends that error on the channel. Afterwards this watch	// can be considered stopped.	Watch(path string, ignore Matcher, ctx context.Context, ignorePerms bool) (<-chan Event, <-chan error, error)	Hide(name string) error	Unhide(name string) error	Glob(pattern string) ([]string, error)	Roots() ([]string, error)	Usage(name string) (Usage, error)	Type() FilesystemType	URI() string	SameFile(fi1, fi2 FileInfo) bool}// The File interface abstracts access to a regular file, being a somewhat// smaller interface than os.Filetype File interface {	io.Closer	io.Reader	io.ReaderAt	io.Seeker	io.Writer	io.WriterAt	Name() string	Truncate(size int64) error	Stat() (FileInfo, error)	Sync() error}// The FileInfo interface is almost the same as os.FileInfo, but with the// Sys method removed (as we don't want to expose whatever is underlying)// and with a couple of convenience methods added.type FileInfo interface {	// Standard things present in os.FileInfo	Name() string	Mode() FileMode	Size() int64	ModTime() time.Time	IsDir() bool	// Extensions	IsRegular() bool	IsSymlink() bool	Owner() int	Group() int}// FileMode is similar to os.FileModetype FileMode uint32func (fm FileMode) String() string {	return os.FileMode(fm).String()}// Usage represents filesystem space usagetype Usage struct {	Free  int64	Total int64}type Matcher interface {	ShouldIgnore(name string) bool	SkipIgnoredDirs() bool}type MatchResult interface {	IsIgnored() bool}type Event struct {	Name string	Type EventType}type EventType intconst (	NonRemove EventType = 1 + iota	Remove	Mixed // Should probably not be necessary to be used in filesystem interface implementation)// Merge returns Mixed, except if evType and other are the same and not Mixed.func (evType EventType) Merge(other EventType) EventType {	return evType | other}func (evType EventType) String() string {	switch {	case evType == NonRemove:		return "non-remove"	case evType == Remove:		return "remove"	case evType == Mixed:		return "mixed"	default:		panic("bug: Unknown event type")	}}var ErrWatchNotSupported = errors.New("watching is not supported")// Equivalents from os package.const ModePerm = FileMode(os.ModePerm)const ModeSetgid = FileMode(os.ModeSetgid)const ModeSetuid = FileMode(os.ModeSetuid)const ModeSticky = FileMode(os.ModeSticky)const ModeSymlink = FileMode(os.ModeSymlink)const ModeType = FileMode(os.ModeType)const PathSeparator = os.PathSeparatorconst OptAppend = os.O_APPENDconst OptCreate = os.O_CREATEconst OptExclusive = os.O_EXCLconst OptReadOnly = os.O_RDONLYconst OptReadWrite = os.O_RDWRconst OptSync = os.O_SYNCconst OptTruncate = os.O_TRUNCconst OptWriteOnly = os.O_WRONLY// SkipDir is used as a return value from WalkFuncs to indicate that// the directory named in the call is to be skipped. It is not returned// as an error by any function.var SkipDir = filepath.SkipDir// IsExist is the equivalent of os.IsExistvar IsExist = os.IsExist// IsExist is the equivalent of os.ErrExistvar ErrExist = os.ErrExist// IsNotExist is the equivalent of os.IsNotExistvar IsNotExist = os.IsNotExist// ErrNotExist is the equivalent of os.ErrNotExistvar ErrNotExist = os.ErrNotExist// IsPermission is the equivalent of os.IsPermissionvar IsPermission = os.IsPermission// IsPathSeparator is the equivalent of os.IsPathSeparatorvar IsPathSeparator = os.IsPathSeparatortype Option func(Filesystem)func NewFilesystem(fsType FilesystemType, uri string, opts ...Option) Filesystem {	var fs Filesystem	switch fsType {	case FilesystemTypeBasic:		fs = newBasicFilesystem(uri, opts...)	case FilesystemTypeFake:		fs = newFakeFilesystem(uri, opts...)	default:		l.Debugln("Unknown filesystem", fsType, uri)		fs = &errorFilesystem{			fsType: fsType,			uri:    uri,			err:    errors.New("filesystem with type " + fsType.String() + " does not exist."),		}	}	if l.ShouldDebug("walkfs") {		return NewWalkFilesystem(&logFilesystem{fs})	}	if l.ShouldDebug("fs") {		return &logFilesystem{NewWalkFilesystem(fs)}	}	return NewWalkFilesystem(fs)}// IsInternal returns true if the file, as a path relative to the folder// root, represents an internal file that should always be ignored. The file// path must be clean (i.e., in canonical shortest form).func IsInternal(file string) bool {	// fs cannot import config, so we hard code .stfolder here (config.DefaultMarkerName)	internals := []string{".stfolder", ".stignore", ".stversions"}	for _, internal := range internals {		if file == internal {			return true		}		if IsParent(file, internal) {			return true		}	}	return false}// Canonicalize checks that the file path is valid and returns it in the "canonical" form:// - /foo/bar -> foo/bar// - / -> "."func Canonicalize(file string) (string, error) {	pathSep := string(PathSeparator)	if strings.HasPrefix(file, pathSep+pathSep) {		// The relative path may pretend to be an absolute path within		// the root, but the double path separator on Windows implies		// something else and is out of spec.		return "", ErrNotRelative	}	// The relative path should be clean from internal dotdots and similar	// funkyness.	file = filepath.Clean(file)	// It is not acceptable to attempt to traverse upwards.	switch file {	case "..":		return "", ErrNotRelative	}	if strings.HasPrefix(file, ".."+pathSep) {		return "", ErrNotRelative	}	if strings.HasPrefix(file, pathSep) {		if file == pathSep {			return ".", nil		}		return file[1:], nil	}	return file, nil}
 |