|  |  | @ -19,6 +19,7 @@ import ( | 
			
		
	
		
			
				
					|  |  |  | 	"github.com/seaweedfs/seaweedfs/weed/storage/needle_map" | 
			
		
	
		
			
				
					|  |  |  | 	"github.com/seaweedfs/seaweedfs/weed/storage/types" | 
			
		
	
		
			
				
					|  |  |  | 	"github.com/seaweedfs/seaweedfs/weed/util" | 
			
		
	
		
			
				
					|  |  |  | 	"golang.org/x/sync/errgroup" | 
			
		
	
		
			
				
					|  |  |  | 	"io" | 
			
		
	
		
			
				
					|  |  |  | 	"math" | 
			
		
	
		
			
				
					|  |  |  | 	"net/http" | 
			
		
	
	
		
			
				
					|  |  | @ -137,32 +138,46 @@ func (c *commandVolumeFsck) Do(args []string, commandEnv *CommandEnv, writer io. | 
			
		
	
		
			
				
					|  |  |  | 		return fmt.Errorf("read filer buckets path: %v", err) | 
			
		
	
		
			
				
					|  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	collectCutoffFromAtNs := time.Now().Add(-*cutoffTimeAgo).UnixNano() | 
			
		
	
		
			
				
					|  |  |  | 	var collectCutoffFromAtNs int64 = 0 | 
			
		
	
		
			
				
					|  |  |  | 	if cutoffTimeAgo.Seconds() != 0 { | 
			
		
	
		
			
				
					|  |  |  | 		collectCutoffFromAtNs = time.Now().Add(-*cutoffTimeAgo).UnixNano() | 
			
		
	
		
			
				
					|  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  | 	var collectModifyFromAtNs int64 = 0 | 
			
		
	
		
			
				
					|  |  |  | 	if modifyTimeAgo.Seconds() != 0 { | 
			
		
	
		
			
				
					|  |  |  | 		collectModifyFromAtNs = time.Now().Add(-*modifyTimeAgo).UnixNano() | 
			
		
	
		
			
				
					|  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  | 	// collect each volume file ids
 | 
			
		
	
		
			
				
					|  |  |  | 	for dataNodeId, volumeIdToVInfo := range dataNodeVolumeIdToVInfo { | 
			
		
	
		
			
				
					|  |  |  | 		for volumeId, vinfo := range volumeIdToVInfo { | 
			
		
	
		
			
				
					|  |  |  | 			if len(c.volumeIds) > 0 { | 
			
		
	
		
			
				
					|  |  |  | 				if _, ok := c.volumeIds[volumeId]; !ok { | 
			
		
	
		
			
				
					|  |  |  | 	eg, gCtx := errgroup.WithContext(context.Background()) | 
			
		
	
		
			
				
					|  |  |  | 	_ = gCtx | 
			
		
	
		
			
				
					|  |  |  | 	for _dataNodeId, _volumeIdToVInfo := range dataNodeVolumeIdToVInfo { | 
			
		
	
		
			
				
					|  |  |  | 		dataNodeId, volumeIdToVInfo := _dataNodeId, _volumeIdToVInfo | 
			
		
	
		
			
				
					|  |  |  | 		eg.Go(func() error { | 
			
		
	
		
			
				
					|  |  |  | 			for volumeId, vinfo := range volumeIdToVInfo { | 
			
		
	
		
			
				
					|  |  |  | 				if len(c.volumeIds) > 0 { | 
			
		
	
		
			
				
					|  |  |  | 					if _, ok := c.volumeIds[volumeId]; !ok { | 
			
		
	
		
			
				
					|  |  |  | 						delete(volumeIdToVInfo, volumeId) | 
			
		
	
		
			
				
					|  |  |  | 						continue | 
			
		
	
		
			
				
					|  |  |  | 					} | 
			
		
	
		
			
				
					|  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  | 				if *c.collection != "" && vinfo.collection != *c.collection { | 
			
		
	
		
			
				
					|  |  |  | 					delete(volumeIdToVInfo, volumeId) | 
			
		
	
		
			
				
					|  |  |  | 					continue | 
			
		
	
		
			
				
					|  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  | 				err = c.collectOneVolumeFileIds(dataNodeId, volumeId, vinfo, uint64(collectModifyFromAtNs), uint64(collectCutoffFromAtNs)) | 
			
		
	
		
			
				
					|  |  |  | 				if err != nil { | 
			
		
	
		
			
				
					|  |  |  | 					return fmt.Errorf("failed to collect file ids from volume %d on %s: %v", volumeId, vinfo.server, err) | 
			
		
	
		
			
				
					|  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  | 			if *c.collection != "" && vinfo.collection != *c.collection { | 
			
		
	
		
			
				
					|  |  |  | 				delete(volumeIdToVInfo, volumeId) | 
			
		
	
		
			
				
					|  |  |  | 				continue | 
			
		
	
		
			
				
					|  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  | 			err = c.collectOneVolumeFileIds(dataNodeId, volumeId, vinfo, uint64(collectModifyFromAtNs), uint64(collectCutoffFromAtNs)) | 
			
		
	
		
			
				
					|  |  |  | 			if err != nil { | 
			
		
	
		
			
				
					|  |  |  | 				return fmt.Errorf("failed to collect file ids from volume %d on %s: %v", volumeId, vinfo.server, err) | 
			
		
	
		
			
				
					|  |  |  | 			if *c.verbose { | 
			
		
	
		
			
				
					|  |  |  | 				fmt.Fprintf(c.writer, "dn %+v filtred %d volumes and locations.\n", dataNodeId, len(dataNodeVolumeIdToVInfo[dataNodeId])) | 
			
		
	
		
			
				
					|  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  | 		if *c.verbose { | 
			
		
	
		
			
				
					|  |  |  | 			fmt.Fprintf(c.writer, "dn %+v filtred %d volumes and locations.\n", dataNodeId, len(dataNodeVolumeIdToVInfo[dataNodeId])) | 
			
		
	
		
			
				
					|  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  | 			return nil | 
			
		
	
		
			
				
					|  |  |  | 		}) | 
			
		
	
		
			
				
					|  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  | 	err = eg.Wait() | 
			
		
	
		
			
				
					|  |  |  | 	if err != nil { | 
			
		
	
		
			
				
					|  |  |  | 		fmt.Fprintf(c.writer, "got error: %v", err) | 
			
		
	
		
			
				
					|  |  |  | 		return err | 
			
		
	
		
			
				
					|  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  | 	if *c.findMissingChunksInFiler { | 
			
		
	
	
		
			
				
					|  |  | @ -416,7 +431,7 @@ func (c *commandVolumeFsck) collectOneVolumeFileIds(dataNodeId string, volumeId | 
			
		
	
		
			
				
					|  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  | 				buf.Write(resp.FileContent) | 
			
		
	
		
			
				
					|  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  | 			if !vinfo.isReadOnly { | 
			
		
	
		
			
				
					|  |  |  | 			if !vinfo.isReadOnly && (modifyFrom != 0 || cutoffFrom != 0) { | 
			
		
	
		
			
				
					|  |  |  | 				index, err := idx.FirstInvalidIndex(buf.Bytes(), | 
			
		
	
		
			
				
					|  |  |  | 					func(key types.NeedleId, offset types.Offset, size types.Size) (bool, error) { | 
			
		
	
		
			
				
					|  |  |  | 						resp, err := volumeServerClient.ReadNeedleMeta(context.Background(), &volume_server_pb.ReadNeedleMetaRequest{ | 
			
		
	
	
		
			
				
					|  |  | @ -428,7 +443,7 @@ func (c *commandVolumeFsck) collectOneVolumeFileIds(dataNodeId string, volumeId | 
			
		
	
		
			
				
					|  |  |  | 						if err != nil { | 
			
		
	
		
			
				
					|  |  |  | 							return false, fmt.Errorf("read needle meta with id %d from volume %d: %v", key, volumeId, err) | 
			
		
	
		
			
				
					|  |  |  | 						} | 
			
		
	
		
			
				
					|  |  |  | 						if (modifyFrom == 0 || modifyFrom <= resp.AppendAtNs) && (resp.AppendAtNs <= cutoffFrom) { | 
			
		
	
		
			
				
					|  |  |  | 						if (modifyFrom == 0 || modifyFrom <= resp.AppendAtNs) && (cutoffFrom == 0 || resp.AppendAtNs <= cutoffFrom) { | 
			
		
	
		
			
				
					|  |  |  | 							return true, nil | 
			
		
	
		
			
				
					|  |  |  | 						} | 
			
		
	
		
			
				
					|  |  |  | 						return false, nil | 
			
		
	
	
		
			
				
					|  |  | 
 |