mirror of https://github.com/minio/minio.git
				
				
				
			
		
			
				
	
	
		
			120 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			120 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Go
		
	
	
	
| /*
 | |
|  * MinIO Cloud Storage, (C) 2019 MinIO, Inc.
 | |
|  *
 | |
|  * Licensed under the Apache License, Version 2.0 (the "License");
 | |
|  * you may not use this file except in compliance with the License.
 | |
|  * You may obtain a copy of the License at
 | |
|  *
 | |
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing, software
 | |
|  * distributed under the License is distributed on an "AS IS" BASIS,
 | |
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  */
 | |
| 
 | |
| package cmd
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/minio/minio/cmd/logger"
 | |
| )
 | |
| 
 | |
| const defaultMonitorNewDiskInterval = time.Minute * 10
 | |
| 
 | |
| func initLocalDisksAutoHeal(ctx context.Context, objAPI ObjectLayer) {
 | |
| 	go monitorLocalDisksAndHeal(ctx, objAPI)
 | |
| }
 | |
| 
 | |
| // monitorLocalDisksAndHeal - ensures that detected new disks are healed
 | |
| //  1. Only the concerned erasure set will be listed and healed
 | |
| //  2. Only the node hosting the disk is responsible to perform the heal
 | |
| func monitorLocalDisksAndHeal(ctx context.Context, objAPI ObjectLayer) {
 | |
| 	z, ok := objAPI.(*xlZones)
 | |
| 	if !ok {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	var bgSeq *healSequence
 | |
| 	var found bool
 | |
| 
 | |
| 	for {
 | |
| 		bgSeq, found = globalBackgroundHealState.getHealSequenceByToken(bgHealingUUID)
 | |
| 		if found {
 | |
| 			break
 | |
| 		}
 | |
| 		time.Sleep(time.Second)
 | |
| 	}
 | |
| 
 | |
| 	// Perform automatic disk healing when a disk is replaced locally.
 | |
| 	for {
 | |
| 		select {
 | |
| 		case <-ctx.Done():
 | |
| 			return
 | |
| 		case <-time.After(defaultMonitorNewDiskInterval):
 | |
| 			// Attempt a heal as the server starts-up first.
 | |
| 			localDisksInZoneHeal := make([]Endpoints, len(z.zones))
 | |
| 			for i, ep := range globalEndpoints {
 | |
| 				localDisksToHeal := Endpoints{}
 | |
| 				for _, endpoint := range ep.Endpoints {
 | |
| 					if !endpoint.IsLocal {
 | |
| 						continue
 | |
| 					}
 | |
| 					// Try to connect to the current endpoint
 | |
| 					// and reformat if the current disk is not formatted
 | |
| 					_, _, err := connectEndpoint(endpoint)
 | |
| 					if err == errUnformattedDisk {
 | |
| 						localDisksToHeal = append(localDisksToHeal, endpoint)
 | |
| 					}
 | |
| 				}
 | |
| 				if len(localDisksToHeal) == 0 {
 | |
| 					continue
 | |
| 				}
 | |
| 				localDisksInZoneHeal[i] = localDisksToHeal
 | |
| 			}
 | |
| 
 | |
| 			// Reformat disks
 | |
| 			bgSeq.sourceCh <- healSource{path: SlashSeparator}
 | |
| 
 | |
| 			// Ensure that reformatting disks is finished
 | |
| 			bgSeq.sourceCh <- healSource{path: nopHeal}
 | |
| 
 | |
| 			var erasureSetInZoneToHeal = make([][]int, len(localDisksInZoneHeal))
 | |
| 			// Compute the list of erasure set to heal
 | |
| 			for i, localDisksToHeal := range localDisksInZoneHeal {
 | |
| 				var erasureSetToHeal []int
 | |
| 				for _, endpoint := range localDisksToHeal {
 | |
| 					// Load the new format of this passed endpoint
 | |
| 					_, format, err := connectEndpoint(endpoint)
 | |
| 					if err != nil {
 | |
| 						logger.LogIf(ctx, err)
 | |
| 						continue
 | |
| 					}
 | |
| 					// Calculate the set index where the current endpoint belongs
 | |
| 					setIndex, _, err := findDiskIndex(z.zones[i].format, format)
 | |
| 					if err != nil {
 | |
| 						logger.LogIf(ctx, err)
 | |
| 						continue
 | |
| 					}
 | |
| 
 | |
| 					erasureSetToHeal = append(erasureSetToHeal, setIndex)
 | |
| 				}
 | |
| 				erasureSetInZoneToHeal[i] = erasureSetToHeal
 | |
| 			}
 | |
| 
 | |
| 			// Heal all erasure sets that need
 | |
| 			for i, erasureSetToHeal := range erasureSetInZoneToHeal {
 | |
| 				for _, setIndex := range erasureSetToHeal {
 | |
| 					err := healErasureSet(ctx, setIndex, z.zones[i].sets[setIndex])
 | |
| 					if err != nil {
 | |
| 						logger.LogIf(ctx, err)
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 |