You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

208 lines
6.3 KiB

3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
  1. package mount
  2. import (
  3. "context"
  4. "math/rand"
  5. "os"
  6. "path"
  7. "path/filepath"
  8. "sync/atomic"
  9. "time"
  10. "github.com/hanwen/go-fuse/v2/fuse"
  11. "google.golang.org/grpc"
  12. "github.com/seaweedfs/seaweedfs/weed/filer"
  13. "github.com/seaweedfs/seaweedfs/weed/mount/meta_cache"
  14. "github.com/seaweedfs/seaweedfs/weed/pb"
  15. "github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
  16. "github.com/seaweedfs/seaweedfs/weed/pb/mount_pb"
  17. "github.com/seaweedfs/seaweedfs/weed/storage/types"
  18. "github.com/seaweedfs/seaweedfs/weed/util"
  19. "github.com/seaweedfs/seaweedfs/weed/util/chunk_cache"
  20. "github.com/seaweedfs/seaweedfs/weed/util/grace"
  21. "github.com/seaweedfs/seaweedfs/weed/wdclient"
  22. "github.com/hanwen/go-fuse/v2/fs"
  23. )
  24. type Option struct {
  25. filerIndex int32 // align memory for atomic read/write
  26. FilerAddresses []pb.ServerAddress
  27. MountDirectory string
  28. GrpcDialOption grpc.DialOption
  29. FilerMountRootPath string
  30. Collection string
  31. Replication string
  32. TtlSec int32
  33. DiskType types.DiskType
  34. ChunkSizeLimit int64
  35. ConcurrentWriters int
  36. CacheDir string
  37. CacheSizeMB int64
  38. CacheDirWrite string
  39. DataCenter string
  40. Umask os.FileMode
  41. Quota int64
  42. DisableXAttr bool
  43. MountUid uint32
  44. MountGid uint32
  45. MountMode os.FileMode
  46. MountCtime time.Time
  47. MountMtime time.Time
  48. MountParentInode uint64
  49. VolumeServerAccess string // how to access volume servers
  50. Cipher bool // whether encrypt data on volume server
  51. UidGidMapper *meta_cache.UidGidMapper
  52. uniqueCacheDir string
  53. uniqueCacheTempPageDir string
  54. }
  55. type WFS struct {
  56. // https://dl.acm.org/doi/fullHtml/10.1145/3310148
  57. // follow https://github.com/hanwen/go-fuse/blob/master/fuse/api.go
  58. fuse.RawFileSystem
  59. mount_pb.UnimplementedSeaweedMountServer
  60. fs.Inode
  61. option *Option
  62. metaCache *meta_cache.MetaCache
  63. stats statsCache
  64. chunkCache *chunk_cache.TieredChunkCache
  65. signature int32
  66. concurrentWriters *util.LimitedConcurrentExecutor
  67. inodeToPath *InodeToPath
  68. fhmap *FileHandleToInode
  69. dhmap *DirectoryHandleToInode
  70. fuseServer *fuse.Server
  71. IsOverQuota bool
  72. }
  73. func NewSeaweedFileSystem(option *Option) *WFS {
  74. wfs := &WFS{
  75. RawFileSystem: fuse.NewDefaultRawFileSystem(),
  76. option: option,
  77. signature: util.RandomInt32(),
  78. inodeToPath: NewInodeToPath(util.FullPath(option.FilerMountRootPath)),
  79. fhmap: NewFileHandleToInode(),
  80. dhmap: NewDirectoryHandleToInode(),
  81. }
  82. wfs.option.filerIndex = int32(rand.Intn(len(option.FilerAddresses)))
  83. wfs.option.setupUniqueCacheDirectory()
  84. if option.CacheSizeMB > 0 {
  85. wfs.chunkCache = chunk_cache.NewTieredChunkCache(256, option.getUniqueCacheDir(), option.CacheSizeMB, 1024*1024)
  86. }
  87. wfs.metaCache = meta_cache.NewMetaCache(path.Join(option.getUniqueCacheDir(), "meta"), option.UidGidMapper,
  88. util.FullPath(option.FilerMountRootPath),
  89. func(path util.FullPath) {
  90. wfs.inodeToPath.MarkChildrenCached(path)
  91. }, func(path util.FullPath) bool {
  92. return wfs.inodeToPath.IsChildrenCached(path)
  93. }, func(filePath util.FullPath, entry *filer_pb.Entry) {
  94. })
  95. grace.OnInterrupt(func() {
  96. wfs.metaCache.Shutdown()
  97. os.RemoveAll(option.getUniqueCacheDir())
  98. os.RemoveAll(option.getTempFilePageDir())
  99. })
  100. if wfs.option.ConcurrentWriters > 0 {
  101. wfs.concurrentWriters = util.NewLimitedConcurrentExecutor(wfs.option.ConcurrentWriters)
  102. }
  103. return wfs
  104. }
  105. func (wfs *WFS) StartBackgroundTasks() {
  106. startTime := time.Now()
  107. go meta_cache.SubscribeMetaEvents(wfs.metaCache, wfs.signature, wfs, wfs.option.FilerMountRootPath, startTime.UnixNano())
  108. go wfs.loopCheckQuota()
  109. }
  110. func (wfs *WFS) String() string {
  111. return "seaweedfs"
  112. }
  113. func (wfs *WFS) Init(server *fuse.Server) {
  114. wfs.fuseServer = server
  115. }
  116. func (wfs *WFS) maybeReadEntry(inode uint64) (path util.FullPath, fh *FileHandle, entry *filer_pb.Entry, status fuse.Status) {
  117. path, status = wfs.inodeToPath.GetPath(inode)
  118. if status != fuse.OK {
  119. return
  120. }
  121. var found bool
  122. if fh, found = wfs.fhmap.FindFileHandle(inode); found {
  123. entry = fh.UpdateEntry(func(entry *filer_pb.Entry) {
  124. if entry != nil && fh.entry.Attributes == nil {
  125. entry.Attributes = &filer_pb.FuseAttributes{}
  126. }
  127. })
  128. } else {
  129. entry, status = wfs.maybeLoadEntry(path)
  130. }
  131. return
  132. }
  133. func (wfs *WFS) maybeLoadEntry(fullpath util.FullPath) (*filer_pb.Entry, fuse.Status) {
  134. // glog.V(3).Infof("read entry cache miss %s", fullpath)
  135. dir, name := fullpath.DirAndName()
  136. // return a valid entry for the mount root
  137. if string(fullpath) == wfs.option.FilerMountRootPath {
  138. return &filer_pb.Entry{
  139. Name: name,
  140. IsDirectory: true,
  141. Attributes: &filer_pb.FuseAttributes{
  142. Mtime: wfs.option.MountMtime.Unix(),
  143. FileMode: uint32(wfs.option.MountMode),
  144. Uid: wfs.option.MountUid,
  145. Gid: wfs.option.MountGid,
  146. Crtime: wfs.option.MountCtime.Unix(),
  147. },
  148. }, fuse.OK
  149. }
  150. // read from async meta cache
  151. meta_cache.EnsureVisited(wfs.metaCache, wfs, util.FullPath(dir))
  152. cachedEntry, cacheErr := wfs.metaCache.FindEntry(context.Background(), fullpath)
  153. if cacheErr == filer_pb.ErrNotFound {
  154. return nil, fuse.ENOENT
  155. }
  156. return cachedEntry.ToProtoEntry(), fuse.OK
  157. }
  158. func (wfs *WFS) LookupFn() wdclient.LookupFileIdFunctionType {
  159. if wfs.option.VolumeServerAccess == "filerProxy" {
  160. return func(fileId string) (targetUrls []string, err error) {
  161. return []string{"http://" + wfs.getCurrentFiler().ToHttpAddress() + "/?proxyChunkId=" + fileId}, nil
  162. }
  163. }
  164. return filer.LookupFn(wfs)
  165. }
  166. func (wfs *WFS) getCurrentFiler() pb.ServerAddress {
  167. i := atomic.LoadInt32(&wfs.option.filerIndex)
  168. return wfs.option.FilerAddresses[i]
  169. }
  170. func (option *Option) setupUniqueCacheDirectory() {
  171. cacheUniqueId := util.Md5String([]byte(option.MountDirectory + string(option.FilerAddresses[0]) + option.FilerMountRootPath + util.Version()))[0:8]
  172. option.uniqueCacheDir = path.Join(option.CacheDir, cacheUniqueId)
  173. os.MkdirAll(option.uniqueCacheDir, os.FileMode(0777)&^option.Umask)
  174. option.uniqueCacheTempPageDir = filepath.Join(path.Join(option.CacheDirWrite, cacheUniqueId), "swap")
  175. os.MkdirAll(option.uniqueCacheTempPageDir, os.FileMode(0777)&^option.Umask)
  176. }
  177. func (option *Option) getTempFilePageDir() string {
  178. return option.uniqueCacheTempPageDir
  179. }
  180. func (option *Option) getUniqueCacheDir() string {
  181. return option.uniqueCacheDir
  182. }