aboutsummaryrefslogtreecommitdiff
path: root/weed/shell/command_volume_check_disk_test.go
diff options
context:
space:
mode:
authorLisandro Pin <lisandro.pin@proton.ch>2025-11-11 01:03:38 +0100
committerGitHub <noreply@github.com>2025-11-10 16:03:38 -0800
commit9744382a183f9b2d0ca0db44ffd30192631000aa (patch)
treef96353c600dd6740f50dd4a78f444859d74cf93c /weed/shell/command_volume_check_disk_test.go
parentd8dd3e4c5348518964d965bdfc15e74d1c4ca81a (diff)
downloadseaweedfs-9744382a183f9b2d0ca0db44ffd30192631000aa.tar.xz
seaweedfs-9744382a183f9b2d0ca0db44ffd30192631000aa.zip
Rework parameters passing for functions within `volume.check.disk`. (#7448)
* Rework parameters passing for functions within `volume.check.disk`. We'll need to rework this logic to account for read-only volumes, and there're already way too many parameters shuffled around. Grouping these into a single struct simplifies the overall codebase. * similar fix * Improved Error Handling in Tests * propagate the errors * edge cases * edge case on modified time * clean up --------- Co-authored-by: chrislu <chris.lu@gmail.com>
Diffstat (limited to 'weed/shell/command_volume_check_disk_test.go')
-rw-r--r--weed/shell/command_volume_check_disk_test.go265
1 files changed, 245 insertions, 20 deletions
diff --git a/weed/shell/command_volume_check_disk_test.go b/weed/shell/command_volume_check_disk_test.go
index d86b40f1f..eee9103a8 100644
--- a/weed/shell/command_volume_check_disk_test.go
+++ b/weed/shell/command_volume_check_disk_test.go
@@ -1,7 +1,7 @@
package shell
import (
- "os"
+ "bytes"
"testing"
"time"
@@ -13,63 +13,288 @@ type testCommandVolumeCheckDisk struct {
}
type shouldSkipVolume struct {
+ name string
a VolumeReplica
b VolumeReplica
pulseTimeAtSecond int64
+ syncDeletions bool
shouldSkipVolume bool
}
func TestShouldSkipVolume(t *testing.T) {
- cmdVolumeCheckDisk := testCommandVolumeCheckDisk{}
- cmdVolumeCheckDisk.writer = os.Stdout
var tests = []shouldSkipVolume{
{
- VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ name: "identical volumes should be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
FileCount: 1000,
DeleteCount: 100,
ModifiedAtSecond: 1696583300},
},
- VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
FileCount: 1000,
DeleteCount: 100,
ModifiedAtSecond: 1696583300},
},
- 1696583400,
- true,
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: true,
},
{
- VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ name: "different file counts should not be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
FileCount: 1001,
DeleteCount: 100,
ModifiedAtSecond: 1696583300},
},
- VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
FileCount: 1000,
DeleteCount: 100,
ModifiedAtSecond: 1696583300},
},
- 1696583400,
- false,
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: false,
},
{
- VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ name: "different delete counts with syncDeletions enabled should not be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
FileCount: 1000,
DeleteCount: 100,
ModifiedAtSecond: 1696583300},
},
- VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
FileCount: 1000,
DeleteCount: 101,
ModifiedAtSecond: 1696583300},
},
- 1696583400,
- false,
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: false,
},
+ {
+ name: "different delete counts with syncDeletions disabled should be skipped if file counts match",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583300},
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 101,
+ ModifiedAtSecond: 1696583300},
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: false,
+ shouldSkipVolume: true,
+ },
+ // Edge case: Zero file and delete counts
+ {
+ name: "volumes with zero file counts should be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 0,
+ DeleteCount: 0,
+ ModifiedAtSecond: 1696583300},
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 0,
+ DeleteCount: 0,
+ ModifiedAtSecond: 1696583300},
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: true,
+ },
+ {
+ name: "volumes with zero and non-zero file counts should not be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1,
+ DeleteCount: 0,
+ ModifiedAtSecond: 1696583300},
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 0,
+ DeleteCount: 0,
+ ModifiedAtSecond: 1696583300},
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: false,
+ },
+ // Edge case: Recently modified volumes (after pulse time)
+ // Note: VolumePulsePeriod is 10 seconds, so pulse cutoff is now - 20 seconds
+ // When both volumes are recently modified, skip check to avoid false positives
+ {
+ name: "recently modified volumes with same file counts should be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583395}, // Modified 5 seconds ago
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583390}, // Modified 10 seconds ago
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: true, // Same counts = skip
+ },
+ {
+ name: "one volume modified before pulse cutoff with different file counts should not be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583370}, // Modified 30 seconds ago (before cutoff at -20s)
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 999,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583370}, // Same modification time
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: false, // Different counts + old enough = needs sync
+ },
+ // Edge case: Different ModifiedAtSecond values, same file counts
+ {
+ name: "different modification times with same file counts should be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583300}, // 100 seconds before pulse time
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583350}, // 50 seconds before pulse time
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: true, // Same counts, both before cutoff
+ },
+ // Edge case: Very close to pulse time boundary
+ {
+ name: "volumes modified exactly at pulse cutoff boundary with different counts should not be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1001,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583379}, // Just before cutoff (pulseTime - 21s)
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583379}, // Just before cutoff
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: false, // At boundary with different counts - needs sync
+ },
+ {
+ name: "volumes modified just after pulse cutoff boundary with same counts should be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583381}, // Just after cutoff (pulseTime - 19s)
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583381}, // Just after cutoff
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: true, // Same counts + recent = skip to avoid false positive
+ },
+ // Edge case: Large file count differences
+ {
+ name: "large file count difference with old modification time should not be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 10000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583300},
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 5000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583300},
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: false, // Large difference requires sync
+ },
+ // Edge case: Both volumes modified AFTER pulse cutoff time
+ // When ModifiedAtSecond >= pulseTimeAtSecond for both volumes with same counts,
+ // the condition (a.info.FileCount != b.info.FileCount) is false, so we skip
+ // without calling eqVolumeFileCount
+ {
+ name: "both volumes modified after pulse cutoff with same file counts should be skipped",
+ a: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583405}, // After pulse cutoff (1696583380)
+ },
+ b: VolumeReplica{nil, &master_pb.VolumeInformationMessage{
+ FileCount: 1000,
+ DeleteCount: 100,
+ ModifiedAtSecond: 1696583410}, // After pulse cutoff
+ },
+ pulseTimeAtSecond: 1696583400,
+ syncDeletions: true,
+ shouldSkipVolume: true, // Same counts = skip without calling eqVolumeFileCount
+ },
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ var buf bytes.Buffer
+ vcd := &volumeCheckDisk{
+ writer: &buf,
+ now: time.Unix(tt.pulseTimeAtSecond, 0),
+ verbose: false, // reduce noise in tests
+ syncDeletions: tt.syncDeletions,
+ }
+ result, err := vcd.shouldSkipVolume(&tt.a, &tt.b)
+ if err != nil {
+ // In unit tests, we expect no errors from shouldSkipVolume
+ // since we're using test data without actual network calls
+ t.Errorf("shouldSkipVolume() returned unexpected error: %v", err)
+ return
+ }
+ if result != tt.shouldSkipVolume {
+ t.Errorf("shouldSkipVolume() = %v, want %v\nFileCount A=%d B=%d, DeleteCount A=%d B=%d",
+ result, tt.shouldSkipVolume,
+ tt.a.info.FileCount, tt.b.info.FileCount,
+ tt.a.info.DeleteCount, tt.b.info.DeleteCount)
+ }
+ })
+ }
+}
+
+// TestVolumeCheckDiskHelperMethods tests the helper methods on volumeCheckDisk
+func TestVolumeCheckDiskHelperMethods(t *testing.T) {
+ var buf bytes.Buffer
+ vcd := &volumeCheckDisk{
+ writer: &buf,
+ verbose: true,
}
- for num, tt := range tests {
- pulseTime := time.Unix(tt.pulseTimeAtSecond, 0)
- if isShould := cmdVolumeCheckDisk.shouldSkipVolume(&tt.a, &tt.b, pulseTime, true, true); isShould != tt.shouldSkipVolume {
- t.Fatalf("result of should skip volume is unexpected for %d test", num)
- }
+
+ // Test write method
+ vcd.write("test %s\n", "message")
+ if buf.String() != "test message\n" {
+ t.Errorf("write() output = %q, want %q", buf.String(), "test message\n")
+ }
+
+ // Test writeVerbose with verbose=true
+ buf.Reset()
+ vcd.writeVerbose("verbose %d\n", 123)
+ if buf.String() != "verbose 123\n" {
+ t.Errorf("writeVerbose() with verbose=true output = %q, want %q", buf.String(), "verbose 123\n")
+ }
+
+ // Test writeVerbose with verbose=false
+ buf.Reset()
+ vcd.verbose = false
+ vcd.writeVerbose("should not appear\n")
+ if buf.String() != "" {
+ t.Errorf("writeVerbose() with verbose=false output = %q, want empty", buf.String())
}
}