| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194 | // 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	Chtimes(name string, atime time.Time, mtime time.Time) error	Create(name string) (File, error)	CreateSymlink(name, target 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	Watch(path string, ignore Matcher, ctx context.Context, ignorePerms bool) (<-chan Event, 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}// 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}// FileMode is similar to os.FileModetype FileMode uint32// Usage represents filesystem space usagetype Usage struct {	Free  int64	Total int64}type Matcher interface {	ShouldIgnore(name string) 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)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// IsNotExist is the equivalent of os.IsNotExistvar IsNotExist = os.IsNotExist// IsPermission is the equivalent of os.IsPermissionvar IsPermission = os.IsPermission// IsPathSeparator is the equivalent of os.IsPathSeparatorvar IsPathSeparator = os.IsPathSeparatorfunc NewFilesystem(fsType FilesystemType, uri string) Filesystem {	var fs Filesystem	switch fsType {	case FilesystemTypeBasic:		fs = NewWalkFilesystem(newBasicFilesystem(uri))	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("filesystem") {		fs = &logFilesystem{fs}	}	return 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"}	pathSep := string(PathSeparator)	for _, internal := range internals {		if file == internal {			return true		}		if strings.HasPrefix(file, internal+pathSep) {			return true		}	}	return false}
 |