forked from TrueCloudLab/restic
Merge pull request #2540 from dp-github/issue-2531
Count hard-linked files correctly in stats command
This commit is contained in:
commit
f2bf06a419
3 changed files with 28 additions and 5 deletions
5
changelog/unreleased/issue-2531
Normal file
5
changelog/unreleased/issue-2531
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
Bugfix: Fix incorrect size calculation in `stats --mode restore-size`
|
||||||
|
|
||||||
|
The restore-size mode of stats was counting hard-linked files as if they were independent.
|
||||||
|
|
||||||
|
https://github.com/restic/restic/issues/2531
|
5
changelog/unreleased/issue-2537
Normal file
5
changelog/unreleased/issue-2537
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
Bugfix: Fix incorrect file counts in `stats --mode restore-size`
|
||||||
|
|
||||||
|
The restore-size mode of stats was failing to count empty directories and some files with hard links.
|
||||||
|
|
||||||
|
https://github.com/restic/restic/issues/2537
|
|
@ -90,6 +90,7 @@ func runStats(gopts GlobalOptions, args []string) error {
|
||||||
// create a container for the stats (and other needed state)
|
// create a container for the stats (and other needed state)
|
||||||
stats := &statsContainer{
|
stats := &statsContainer{
|
||||||
uniqueFiles: make(map[fileID]struct{}),
|
uniqueFiles: make(map[fileID]struct{}),
|
||||||
|
uniqueInodes: make(map[uint64]struct{}),
|
||||||
fileBlobs: make(map[string]restic.IDSet),
|
fileBlobs: make(map[string]restic.IDSet),
|
||||||
blobs: restic.NewBlobSet(),
|
blobs: restic.NewBlobSet(),
|
||||||
blobsSeen: restic.NewBlobSet(),
|
blobsSeen: restic.NewBlobSet(),
|
||||||
|
@ -247,8 +248,16 @@ func statsWalkTree(repo restic.Repository, stats *statsContainer) walker.WalkFun
|
||||||
// as this is a file in the snapshot, we can simply count its
|
// as this is a file in the snapshot, we can simply count its
|
||||||
// size without worrying about uniqueness, since duplicate files
|
// size without worrying about uniqueness, since duplicate files
|
||||||
// will still be restored
|
// will still be restored
|
||||||
stats.TotalSize += node.Size
|
|
||||||
stats.TotalFileCount++
|
stats.TotalFileCount++
|
||||||
|
|
||||||
|
// if inodes are present, only count each inode once
|
||||||
|
// (hard links do not increase restore size)
|
||||||
|
if _, ok := stats.uniqueInodes[node.Inode]; !ok || node.Inode == 0 {
|
||||||
|
stats.uniqueInodes[node.Inode] = struct{}{}
|
||||||
|
stats.TotalSize += node.Size
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return true, nil
|
return true, nil
|
||||||
|
@ -301,6 +310,10 @@ type statsContainer struct {
|
||||||
// contents (hashed sequence of content blob IDs)
|
// contents (hashed sequence of content blob IDs)
|
||||||
uniqueFiles map[fileID]struct{}
|
uniqueFiles map[fileID]struct{}
|
||||||
|
|
||||||
|
// uniqueInodes marks visited files according to their
|
||||||
|
// inode # (hashed sequence of inode numbers)
|
||||||
|
uniqueInodes map[uint64]struct{}
|
||||||
|
|
||||||
// fileBlobs maps a file name (path) to the set of
|
// fileBlobs maps a file name (path) to the set of
|
||||||
// blobs that have been seen as a part of the file
|
// blobs that have been seen as a part of the file
|
||||||
fileBlobs map[string]restic.IDSet
|
fileBlobs map[string]restic.IDSet
|
||||||
|
|
Loading…
Reference in a new issue