You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							130 lines
						
					
					
						
							4.0 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							130 lines
						
					
					
						
							4.0 KiB
						
					
					
				| package shell | |
| 
 | |
| import ( | |
| 	"context" | |
| 	"flag" | |
| 	"fmt" | |
| 	"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb" | |
| 	"github.com/seaweedfs/seaweedfs/weed/storage/needle" | |
| 	"github.com/seaweedfs/seaweedfs/weed/util" | |
| 	"io" | |
| 	"os" | |
| 	"strconv" | |
| 	"strings" | |
| ) | |
| 
 | |
| func init() { | |
| 	Commands = append(Commands, &commandFsMetaChangeVolumeId{}) | |
| } | |
| 
 | |
| type commandFsMetaChangeVolumeId struct { | |
| } | |
| 
 | |
| func (c *commandFsMetaChangeVolumeId) Name() string { | |
| 	return "fs.meta.changeVolumeId" | |
| } | |
| 
 | |
| func (c *commandFsMetaChangeVolumeId) Help() string { | |
| 	return `change volume id in existing metadata. | |
|  | |
| 	fs.meta.changeVolumeId -dir=/path/to/a/dir -fromVolumeId=x -toVolumeId=y -force | |
| 	fs.meta.changeVolumeId -dir=/path/to/a/dir -mapping=/path/to/mapping/file -force | |
|  | |
| 	The mapping file should have these lines, each line is: [fromVolumeId]=>[toVolumeId] | |
| 	e.g. | |
| 		1 => 2 | |
| 		3 => 4 | |
|  | |
| ` | |
| } | |
| 
 | |
| func (c *commandFsMetaChangeVolumeId) HasTag(CommandTag) bool { | |
| 	return false | |
| } | |
| 
 | |
| func (c *commandFsMetaChangeVolumeId) Do(args []string, commandEnv *CommandEnv, writer io.Writer) (err error) { | |
| 
 | |
| 	fsMetaChangeVolumeIdCommand := flag.NewFlagSet(c.Name(), flag.ContinueOnError) | |
| 	dir := fsMetaChangeVolumeIdCommand.String("dir", "/", "fix all metadata under this folder") | |
| 	mappingFileName := fsMetaChangeVolumeIdCommand.String("mapping", "", "a file with multiple volume id changes, with each line as x=>y") | |
| 	fromVolumeId := fsMetaChangeVolumeIdCommand.Uint("fromVolumeId", 0, "change metadata with this volume id") | |
| 	toVolumeId := fsMetaChangeVolumeIdCommand.Uint("toVolumeId", 0, "change metadata to this volume id") | |
| 	isForce := fsMetaChangeVolumeIdCommand.Bool("force", false, "applying the metadata changes") | |
| 	if err = fsMetaChangeVolumeIdCommand.Parse(args); err != nil { | |
| 		return err | |
| 	} | |
| 
 | |
| 	// load the mapping | |
| 	mapping := make(map[needle.VolumeId]needle.VolumeId) | |
| 	if *mappingFileName != "" { | |
| 		readMappingFromFile(*mappingFileName, mapping) | |
| 	} else { | |
| 		if *fromVolumeId == *toVolumeId { | |
| 			return fmt.Errorf("no volume id changes") | |
| 		} | |
| 		if *fromVolumeId == 0 || *toVolumeId == 0 { | |
| 			return fmt.Errorf("volume id can not be zero") | |
| 		} | |
| 		mapping[needle.VolumeId(*fromVolumeId)] = needle.VolumeId(*toVolumeId) | |
| 	} | |
| 
 | |
| 	return commandEnv.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error { | |
| 		return filer_pb.TraverseBfs(commandEnv, util.FullPath(*dir), func(parentPath util.FullPath, entry *filer_pb.Entry) { | |
| 			if !entry.IsDirectory { | |
| 				var hasChanges bool | |
| 				for _, chunk := range entry.Chunks { | |
| 					if chunk.IsChunkManifest { | |
| 						fmt.Printf("Change volume id for large file is not implemented yet: %s/%s\n", parentPath, entry.Name) | |
| 						return | |
| 					} | |
| 					chunkVolumeId := chunk.Fid.VolumeId | |
| 					if toVolumeId, found := mapping[needle.VolumeId(chunkVolumeId)]; found { | |
| 						hasChanges = true | |
| 						chunk.Fid.VolumeId = uint32(toVolumeId) | |
| 						chunk.FileId = "" | |
| 					} | |
| 				} | |
| 				if hasChanges { | |
| 					println("Updating", parentPath, entry.Name) | |
| 					if *isForce { | |
| 						if updateErr := filer_pb.UpdateEntry(context.Background(), client, &filer_pb.UpdateEntryRequest{ | |
| 							Directory: string(parentPath), | |
| 							Entry:     entry, | |
| 						}); updateErr != nil { | |
| 							fmt.Printf("failed to update %s/%s: %v\n", parentPath, entry.Name, updateErr) | |
| 						} | |
| 					} | |
| 				} | |
| 			} | |
| 		}) | |
| 	}) | |
| } | |
| 
 | |
| func readMappingFromFile(filename string, mapping map[needle.VolumeId]needle.VolumeId) error { | |
| 	mappingFile, openErr := os.Open(filename) | |
| 	if openErr != nil { | |
| 		return fmt.Errorf("failed to open file %s: %v", filename, openErr) | |
| 	} | |
| 	defer mappingFile.Close() | |
| 	mappingContent, readErr := io.ReadAll(mappingFile) | |
| 	if readErr != nil { | |
| 		return fmt.Errorf("failed to read file %s: %v", filename, readErr) | |
| 	} | |
| 	for _, line := range strings.Split(string(mappingContent), "\n") { | |
| 		parts := strings.Split(line, "=>") | |
| 		if len(parts) != 2 { | |
| 			println("unrecognized line:", line) | |
| 			continue | |
| 		} | |
| 		x, errX := strconv.ParseUint(strings.TrimSpace(parts[0]), 10, 64) | |
| 		if errX != nil { | |
| 			return errX | |
| 		} | |
| 		y, errY := strconv.ParseUint(strings.TrimSpace(parts[1]), 10, 64) | |
| 		if errY != nil { | |
| 			return errY | |
| 		} | |
| 		mapping[needle.VolumeId(x)] = needle.VolumeId(y) | |
| 	} | |
| 	return nil | |
| }
 |