diff options
| author | Lisandro Pin <lisandro.pin@proton.ch> | 2025-11-11 01:03:38 +0100 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2025-11-10 16:03:38 -0800 |
| commit | 9744382a183f9b2d0ca0db44ffd30192631000aa (patch) | |
| tree | f96353c600dd6740f50dd4a78f444859d74cf93c /weed/shell/command_volume_check_disk_test.go | |
| parent | d8dd3e4c5348518964d965bdfc15e74d1c4ca81a (diff) | |
| download | seaweedfs-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.go | 265 |
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()) } } |
