mirror of
https://github.com/minio/minio.git
synced 2025-01-25 21:53:16 -05:00
heal: Ignore disks with non quorum modtime and dataDir (#12328)
This commit is contained in:
parent
ecb5525c91
commit
866593fd94
@ -198,6 +198,9 @@ func getLatestFileInfo(ctx context.Context, partsMetadata []FileInfo, errs []err
|
|||||||
// a not-found error or a hash-mismatch error.
|
// a not-found error or a hash-mismatch error.
|
||||||
func disksWithAllParts(ctx context.Context, onlineDisks []StorageAPI, partsMetadata []FileInfo, errs []error, bucket,
|
func disksWithAllParts(ctx context.Context, onlineDisks []StorageAPI, partsMetadata []FileInfo, errs []error, bucket,
|
||||||
object string, scanMode madmin.HealScanMode) ([]StorageAPI, []error) {
|
object string, scanMode madmin.HealScanMode) ([]StorageAPI, []error) {
|
||||||
|
// List of disks having latest version of the object er.meta (by modtime)
|
||||||
|
_, modTime, dataDir := listOnlineDisks(onlineDisks, partsMetadata, errs)
|
||||||
|
|
||||||
availableDisks := make([]StorageAPI, len(onlineDisks))
|
availableDisks := make([]StorageAPI, len(onlineDisks))
|
||||||
dataErrs := make([]error, len(onlineDisks))
|
dataErrs := make([]error, len(onlineDisks))
|
||||||
inconsistent := 0
|
inconsistent := 0
|
||||||
@ -236,6 +239,13 @@ func disksWithAllParts(ctx context.Context, onlineDisks []StorageAPI, partsMetad
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
meta := partsMetadata[i]
|
meta := partsMetadata[i]
|
||||||
|
|
||||||
|
if !meta.ModTime.Equal(modTime) || meta.DataDir != dataDir {
|
||||||
|
dataErrs[i] = errFileCorrupt
|
||||||
|
partsMetadata[i] = FileInfo{}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
if erasureDistributionReliable {
|
if erasureDistributionReliable {
|
||||||
if !meta.IsValid() {
|
if !meta.IsValid() {
|
||||||
continue
|
continue
|
||||||
|
@ -484,13 +484,15 @@ func TestDisksWithAllParts(t *testing.T) {
|
|||||||
t.Fatalf("Failed to read xl meta data %v", reducedErr)
|
t.Fatalf("Failed to read xl meta data %v", reducedErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Test that all disks are returned without any failures with
|
// Test 1: Test that all disks are returned without any failures with
|
||||||
// unmodified meta data
|
// unmodified meta data
|
||||||
partsMetadata, errs := readAllFileInfo(ctx, erasureDisks, bucket, object, "", false)
|
partsMetadata, errs := readAllFileInfo(ctx, erasureDisks, bucket, object, "", false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Failed to read xl meta data %v", err)
|
t.Fatalf("Failed to read xl meta data %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
erasureDisks, _, _ = listOnlineDisks(erasureDisks, partsMetadata, errs)
|
||||||
|
|
||||||
filteredDisks, errs := disksWithAllParts(ctx, erasureDisks, partsMetadata, errs, bucket, object, madmin.HealDeepScan)
|
filteredDisks, errs := disksWithAllParts(ctx, erasureDisks, partsMetadata, errs, bucket, object, madmin.HealDeepScan)
|
||||||
|
|
||||||
if len(filteredDisks) != len(erasureDisks) {
|
if len(filteredDisks) != len(erasureDisks) {
|
||||||
@ -507,8 +509,48 @@ func TestDisksWithAllParts(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Test 2: Not synchronized modtime
|
||||||
|
partsMetadataBackup := partsMetadata[0]
|
||||||
|
partsMetadata[0].ModTime = partsMetadata[0].ModTime.Add(-1 * time.Hour)
|
||||||
|
|
||||||
|
errs = make([]error, len(erasureDisks))
|
||||||
|
filteredDisks, _ = disksWithAllParts(ctx, erasureDisks, partsMetadata, errs, bucket, object, madmin.HealDeepScan)
|
||||||
|
|
||||||
|
if len(filteredDisks) != len(erasureDisks) {
|
||||||
|
t.Errorf("Unexpected number of disks: %d", len(filteredDisks))
|
||||||
|
}
|
||||||
|
for diskIndex, disk := range filteredDisks {
|
||||||
|
if diskIndex == 0 && disk != nil {
|
||||||
|
t.Errorf("Disk not filtered as expected, disk: %d", diskIndex)
|
||||||
|
}
|
||||||
|
if diskIndex != 0 && disk == nil {
|
||||||
|
t.Errorf("Disk erroneously filtered, diskIndex: %d", diskIndex)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
partsMetadata[0] = partsMetadataBackup // Revert before going to the next test
|
||||||
|
|
||||||
|
// Test 3: Not synchronized DataDir
|
||||||
|
partsMetadataBackup = partsMetadata[1]
|
||||||
|
partsMetadata[1].DataDir = "foo-random"
|
||||||
|
|
||||||
|
errs = make([]error, len(erasureDisks))
|
||||||
|
filteredDisks, _ = disksWithAllParts(ctx, erasureDisks, partsMetadata, errs, bucket, object, madmin.HealDeepScan)
|
||||||
|
|
||||||
|
if len(filteredDisks) != len(erasureDisks) {
|
||||||
|
t.Errorf("Unexpected number of disks: %d", len(filteredDisks))
|
||||||
|
}
|
||||||
|
for diskIndex, disk := range filteredDisks {
|
||||||
|
if diskIndex == 1 && disk != nil {
|
||||||
|
t.Errorf("Disk not filtered as expected, disk: %d", diskIndex)
|
||||||
|
}
|
||||||
|
if diskIndex != 1 && disk == nil {
|
||||||
|
t.Errorf("Disk erroneously filtered, diskIndex: %d", diskIndex)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
partsMetadata[1] = partsMetadataBackup // Revert before going to the next test
|
||||||
|
|
||||||
|
// Test 4: key = disk index, value = part name with hash mismatch
|
||||||
diskFailures := make(map[int]string)
|
diskFailures := make(map[int]string)
|
||||||
// key = disk index, value = part name with hash mismatch
|
|
||||||
diskFailures[0] = "part.1"
|
diskFailures[0] = "part.1"
|
||||||
diskFailures[3] = "part.1"
|
diskFailures[3] = "part.1"
|
||||||
diskFailures[15] = "part.1"
|
diskFailures[15] = "part.1"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user