| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383 | // 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"	"fmt"	"io"	"io/fs"	"os"	"path/filepath"	"strings"	"time"	"github.com/syncthing/syncthing/lib/ignore/ignoreresult"	"github.com/syncthing/syncthing/lib/protocol")type XattrFilter interface {	Permit(string) bool	GetMaxSingleEntrySize() int	GetMaxTotalSize() int}// The Filesystem interface abstracts access to the file system.type Filesystem interface {	Chmod(name string, mode FileMode) error	Lchown(name string, uid, gid string) error // uid/gid as strings; numeric on POSIX, SID on Windows, like in os/user package	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	Options() []Option	SameFile(fi1, fi2 FileInfo) bool	PlatformData(name string, withOwnership, withXattrs bool, xattrFilter XattrFilter) (protocol.PlatformData, error)	GetXattr(name string, xattrFilter XattrFilter) ([]protocol.Xattr, error)	SetXattr(path string, xattrs []protocol.Xattr, xattrFilter XattrFilter) error}type wrappingFilesystem interface {	// Used for unwrapping things	underlying() (Filesystem, 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	Sys() interface{}	// Extensions	IsRegular() bool	IsSymlink() bool	Owner() int	Group() int	InodeChangeTime() time.Time // may be zero if not supported}// 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  uint64	Total uint64}type Matcher interface {	Match(name string) ignoreresult.R}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 evType {	case NonRemove:		return "non-remove"	case Remove:		return "remove"	case Mixed:		return "mixed"	default:		panic("bug: Unknown event type")	}}var (	ErrWatchNotSupported  = errors.New("watching is not supported")	ErrXattrsNotSupported = errors.New("extended attributes are not supported on this platform"))// Equivalents from os package.const (	ModePerm      = FileMode(os.ModePerm)	ModeSetgid    = FileMode(os.ModeSetgid)	ModeSetuid    = FileMode(os.ModeSetuid)	ModeSticky    = FileMode(os.ModeSticky)	ModeSymlink   = FileMode(os.ModeSymlink)	ModeType      = FileMode(os.ModeType)	PathSeparator = os.PathSeparator	OptAppend     = os.O_APPEND	OptCreate     = os.O_CREATE	OptExclusive  = os.O_EXCL	OptReadOnly   = os.O_RDONLY	OptReadWrite  = os.O_RDWR	OptSync       = os.O_SYNC	OptTruncate   = os.O_TRUNC	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 //nolint:errnamefunc IsExist(err error) bool {	return errors.Is(err, ErrExist)}// ErrExist is the equivalent of os.ErrExistvar ErrExist = fs.ErrExist// IsNotExist is the equivalent of os.IsNotExistfunc IsNotExist(err error) bool {	return errors.Is(err, ErrNotExist)}// ErrNotExist is the equivalent of os.ErrNotExistvar ErrNotExist = fs.ErrNotExist// IsPermission is the equivalent of os.IsPermissionfunc IsPermission(err error) bool {	return errors.Is(err, fs.ErrPermission)}// IsPathSeparator is the equivalent of os.IsPathSeparatorvar IsPathSeparator = os.IsPathSeparatorfunc NewFilesystem(fsType FilesystemType, uri string, opts ...Option) Filesystem {	var caseOpt Option	var mtimeOpt Option	i := 0	for _, opt := range opts {		if caseOpt != nil && mtimeOpt != nil {			break		}		switch opt.(type) {		case *OptionDetectCaseConflicts:			caseOpt = opt		case *optionMtime:			mtimeOpt = opt		default:			opts[i] = opt			i++		}	}	opts = opts[:i]	// Construct file system using the registered factory function	var fs Filesystem	var err error	filesystemFactoriesMutex.Lock()	fsFactory, factoryFound := filesystemFactories[fsType]	filesystemFactoriesMutex.Unlock()	if factoryFound {		fs, err = fsFactory(uri, opts...)	} else {		err = fmt.Errorf("File system type '%s' not recognized", fsType)	}	if err != nil {		fs = &errorFilesystem{			fsType: fsType,			uri:    uri,			err:    err,		}	}	// mtime handling should happen inside walking, as filesystem calls while	// walking should be mtime-resolved too	if mtimeOpt != nil {		fs = mtimeOpt.apply(fs)	}	fs = &metricsFS{next: fs}	layersAboveWalkFilesystem := 0	if caseOpt != nil {		// DirNames calls made to check the case of a name will also be		// attributed to the calling function.		layersAboveWalkFilesystem++	}	switch {	case l.ShouldDebug("walkfs"):		// A walkFilesystem is not a layer to skip, it embeds the underlying		// filesystem, passing calls directly trough. Except for calls made		// during walking, however those are truly originating in the walk		// filesystem.		fs = NewWalkFilesystem(newLogFilesystem(fs, layersAboveWalkFilesystem))	case l.ShouldDebug("fs"):		fs = newLogFilesystem(NewWalkFilesystem(fs), layersAboveWalkFilesystem)	default:		fs = NewWalkFilesystem(fs)	}	// Case handling is at the outermost layer to resolve all input names.	// Reason being is that the only names/paths that are potentially "wrong"	// come from outside the fs package. Any paths that result from filesystem	// operations itself already have the correct case. Thus there's e.g. no	// point to check the case on all the stating the walk filesystem does, it	// just adds overhead.	if caseOpt != nil {		fs = caseOpt.apply(fs)	}	return fs}// fs cannot import config or versioner, so we hard code .stfolder// (config.DefaultMarkerName) and .stversions (versioner.DefaultPath)var internals = []string{".stfolder", ".stignore", ".stversions"}// 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 {	for _, internal := range internals {		if file == internal {			return true		}		if IsParent(file, internal) {			return true		}	}	return false}var (	errPathInvalid           = errors.New("path is invalid")	errPathTraversingUpwards = errors.New("relative path traversing upwards (starting with ..)"))// 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) {	const 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 "", errPathInvalid	}	// The relative path should be clean from internal dotdots and similar	// funkiness.	file = filepath.Clean(file)	// It is not acceptable to attempt to traverse upwards.	if file == ".." {		return "", errPathTraversingUpwards	}	if strings.HasPrefix(file, ".."+pathSep) {		return "", errPathTraversingUpwards	}	if strings.HasPrefix(file, pathSep) {		if file == pathSep {			return ".", nil		}		return file[1:], nil	}	return file, nil}// unwrapFilesystem removes "wrapping" filesystems to expose the filesystem of the requested wrapper type T, if it exists.func unwrapFilesystem[T Filesystem](fs Filesystem) (T, bool) {	for {		if unwrapped, ok := fs.(T); ok {			return unwrapped, true		}		wrappingFs, ok := fs.(wrappingFilesystem)		if !ok {			var x T			return x, false		}		fs, ok = wrappingFs.underlying()		if !ok {			var x T			return x, false		}	}}// WriteFile writes data to the named file, creating it if necessary.// If the file does not exist, WriteFile creates it with permissions perm (before umask);// otherwise WriteFile truncates it before writing, without changing permissions.// Since Writefile requires multiple system calls to complete, a failure mid-operation// can leave the file in a partially written state.func WriteFile(fs Filesystem, name string, data []byte, perm FileMode) error {	f, err := fs.OpenFile(name, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, perm)	if err != nil {		return err	}	_, err = f.Write(data)	if err1 := f.Close(); err1 != nil && err == nil {		err = err1	}	return err}
 |