239 lines
7.4 KiB

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