mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2024-01-19 02:48:24 +00:00
add fs.meta.changeVolumeId
This commit is contained in:
parent
de081c0d64
commit
6e8dbf8868
121
weed/shell/command_fs_meta_change_volume_id.go
Normal file
121
weed/shell/command_fs_meta_change_volume_id.go
Normal file
|
@ -0,0 +1,121 @@
|
|||
package shell
|
||||
|
||||
import (
|
||||
"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) 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 {
|
||||
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(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
|
||||
}
|
Loading…
Reference in a new issue