|
|
package mount
import ( "context" "fmt" "github.com/hanwen/go-fuse/v2/fuse" "github.com/seaweedfs/seaweedfs/weed/glog" "github.com/seaweedfs/seaweedfs/weed/pb/filer_pb" "math" "time" )
const blockSize = 512
type statsCache struct { filer_pb.StatisticsResponse lastChecked int64 // unix time in seconds
}
func (wfs *WFS) StatFs(cancel <-chan struct{}, in *fuse.InHeader, out *fuse.StatfsOut) (code fuse.Status) {
// glog.V(4).Infof("reading fs stats")
if wfs.stats.lastChecked < time.Now().Unix()-20 {
err := wfs.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
request := &filer_pb.StatisticsRequest{ Collection: wfs.option.Collection, Replication: wfs.option.Replication, Ttl: fmt.Sprintf("%ds", wfs.option.TtlSec), DiskType: string(wfs.option.DiskType), }
glog.V(4).Infof("reading filer stats: %+v", request) resp, err := client.Statistics(context.Background(), request) if err != nil { glog.V(0).Infof("reading filer stats %v: %v", request, err) return err } glog.V(4).Infof("read filer stats: %+v", resp)
wfs.stats.TotalSize = resp.TotalSize wfs.stats.UsedSize = resp.UsedSize wfs.stats.FileCount = resp.FileCount wfs.stats.lastChecked = time.Now().Unix()
return nil }) if err != nil { glog.V(0).Infof("filer Statistics: %v", err) return fuse.OK } }
totalDiskSize := wfs.stats.TotalSize usedDiskSize := wfs.stats.UsedSize actualFileCount := wfs.stats.FileCount
if wfs.option.Quota > 0 && totalDiskSize > uint64(wfs.option.Quota) { totalDiskSize = uint64(wfs.option.Quota) if usedDiskSize > totalDiskSize { totalDiskSize = usedDiskSize } }
// http://man.he.net/man2/statfs
/* struct statfs { __fsword_t f_type; // Type of filesystem (see below)
__fsword_t f_bsize; // Optimal transfer block size
fsblkcnt_t f_blocks; // Total data blocks in filesystem
fsblkcnt_t f_bfree; // Free blocks in filesystem
fsblkcnt_t f_bavail; // Free blocks available to
unprivileged user fsfilcnt_t f_files; // Total file nodes in filesystem
fsfilcnt_t f_ffree; // Free file nodes in filesystem
fsid_t f_fsid; // Filesystem ID
__fsword_t f_namelen; // Maximum length of filenames
__fsword_t f_frsize; // Fragment size (since Linux 2.6)
__fsword_t f_flags; // Mount flags of filesystem
(since Linux 2.6.36) __fsword_t f_spare[xxx]; // Padding bytes reserved for future use
}; */
// Compute the total number of available blocks
out.Blocks = totalDiskSize / blockSize if out.Blocks <= 0 { out.Blocks = 1 } // Compute the number of used blocks
numBlocks := usedDiskSize / blockSize
remainingBlocks := int64(out.Blocks) - int64(numBlocks) if remainingBlocks < 0 { remainingBlocks = 0 }
// Report the number of free and available blocks for the block size
out.Bfree = uint64(remainingBlocks) out.Bavail = uint64(remainingBlocks) out.Bsize = uint32(blockSize)
// Report the total number of possible files in the file system (and those free)
out.Files = math.MaxInt64 out.Ffree = math.MaxInt64 - actualFileCount
// Report the maximum length of a name and the minimum fragment size
out.NameLen = 1024 out.Frsize = uint32(blockSize)
return fuse.OK }
|