2024-08-28 15:32:30 +00:00
|
|
|
package metabase
|
|
|
|
|
|
|
|
import (
|
2024-10-03 08:06:31 +00:00
|
|
|
"context"
|
2024-08-28 15:32:30 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2024-09-06 08:16:12 +00:00
|
|
|
"sync"
|
2024-08-28 15:32:30 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-adm/internal/commonflags"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-node/config"
|
|
|
|
engineconfig "git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-node/config/engine"
|
|
|
|
shardconfig "git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-node/config/engine/shard"
|
2024-10-03 08:06:31 +00:00
|
|
|
morphconfig "git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-node/config/morph"
|
|
|
|
nodeconfig "git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-node/config/node"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/core/container"
|
2024-08-28 15:32:30 +00:00
|
|
|
meta "git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/metabase"
|
2024-10-03 08:06:31 +00:00
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/morph/client"
|
|
|
|
morphcontainer "git.frostfs.info/TrueCloudLab/frostfs-node/pkg/morph/client/container"
|
2024-08-28 15:32:30 +00:00
|
|
|
"github.com/spf13/cobra"
|
2024-09-06 08:16:12 +00:00
|
|
|
"golang.org/x/sync/errgroup"
|
2024-08-28 15:32:30 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
noCompactFlag = "no-compact"
|
|
|
|
)
|
|
|
|
|
2024-10-03 08:06:31 +00:00
|
|
|
var (
|
|
|
|
errNoPathsFound = errors.New("no metabase paths found")
|
|
|
|
errNoMorphEndpointsFound = errors.New("no morph endpoints found")
|
|
|
|
)
|
2024-08-28 15:32:30 +00:00
|
|
|
|
|
|
|
var UpgradeCmd = &cobra.Command{
|
|
|
|
Use: "upgrade",
|
|
|
|
Short: "Upgrade metabase to latest version",
|
|
|
|
RunE: upgrade,
|
|
|
|
}
|
|
|
|
|
|
|
|
func upgrade(cmd *cobra.Command, _ []string) error {
|
|
|
|
configFile, err := cmd.Flags().GetString(commonflags.ConfigFlag)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
configDir, err := cmd.Flags().GetString(commonflags.ConfigDirFlag)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
appCfg := config.New(configFile, configDir, config.EnvPrefix)
|
2024-10-03 08:06:31 +00:00
|
|
|
paths, err := getMetabasePaths(appCfg)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2024-08-28 15:32:30 +00:00
|
|
|
}
|
|
|
|
if len(paths) == 0 {
|
|
|
|
return errNoPathsFound
|
|
|
|
}
|
|
|
|
cmd.Println("found", len(paths), "metabases:")
|
|
|
|
for i, path := range paths {
|
|
|
|
cmd.Println(i+1, ":", path)
|
|
|
|
}
|
2024-10-03 08:06:31 +00:00
|
|
|
mc, err := createMorphClient(cmd.Context(), appCfg)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer mc.Close()
|
|
|
|
civ, err := createContainerInfoProvider(mc)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
noCompact, _ := cmd.Flags().GetBool(noCompactFlag)
|
2024-08-28 15:32:30 +00:00
|
|
|
result := make(map[string]bool)
|
2024-09-06 08:16:12 +00:00
|
|
|
var resultGuard sync.Mutex
|
|
|
|
eg, ctx := errgroup.WithContext(cmd.Context())
|
2024-08-28 15:32:30 +00:00
|
|
|
for _, path := range paths {
|
2024-09-06 08:16:12 +00:00
|
|
|
eg.Go(func() error {
|
|
|
|
var success bool
|
|
|
|
cmd.Println("upgrading metabase", path, "...")
|
2024-10-03 08:06:31 +00:00
|
|
|
if err := meta.Upgrade(ctx, path, !noCompact, civ, func(a ...any) {
|
2024-09-06 08:16:12 +00:00
|
|
|
cmd.Println(append([]any{time.Now().Format(time.RFC3339), ":", path, ":"}, a...)...)
|
|
|
|
}); err != nil {
|
|
|
|
cmd.Println("error: failed to upgrade metabase", path, ":", err)
|
|
|
|
} else {
|
|
|
|
success = true
|
|
|
|
cmd.Println("metabase", path, "upgraded successfully")
|
|
|
|
}
|
|
|
|
resultGuard.Lock()
|
|
|
|
result[path] = success
|
|
|
|
resultGuard.Unlock()
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if err := eg.Wait(); err != nil {
|
|
|
|
return err
|
2024-08-28 15:32:30 +00:00
|
|
|
}
|
|
|
|
for mb, ok := range result {
|
|
|
|
if ok {
|
|
|
|
cmd.Println(mb, ": success")
|
|
|
|
} else {
|
|
|
|
cmd.Println(mb, ": failed")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-10-03 08:06:31 +00:00
|
|
|
func getMetabasePaths(appCfg *config.Config) ([]string, error) {
|
|
|
|
var paths []string
|
|
|
|
if err := engineconfig.IterateShards(appCfg, false, func(sc *shardconfig.Config) error {
|
|
|
|
paths = append(paths, sc.Metabase().Path())
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return nil, fmt.Errorf("get metabase paths: %w", err)
|
|
|
|
}
|
|
|
|
return paths, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func createMorphClient(ctx context.Context, appCfg *config.Config) (*client.Client, error) {
|
|
|
|
addresses := morphconfig.RPCEndpoint(appCfg)
|
|
|
|
if len(addresses) == 0 {
|
|
|
|
return nil, errNoMorphEndpointsFound
|
|
|
|
}
|
|
|
|
key := nodeconfig.Key(appCfg)
|
|
|
|
cli, err := client.New(ctx,
|
|
|
|
key,
|
|
|
|
client.WithDialTimeout(morphconfig.DialTimeout(appCfg)),
|
|
|
|
client.WithEndpoints(addresses...),
|
|
|
|
client.WithSwitchInterval(morphconfig.SwitchInterval(appCfg)),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("create morph client:%w", err)
|
|
|
|
}
|
|
|
|
return cli, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func createContainerInfoProvider(cli *client.Client) (container.InfoProvider, error) {
|
|
|
|
sh, err := cli.NNSContractAddress(client.NNSContainerContractName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("resolve container contract hash: %w", err)
|
|
|
|
}
|
2024-12-04 12:25:36 +00:00
|
|
|
cc, err := morphcontainer.NewFromMorph(cli, sh, 0)
|
2024-10-03 08:06:31 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("create morph container client: %w", err)
|
|
|
|
}
|
|
|
|
return container.NewInfoProvider(func() (container.Source, error) {
|
|
|
|
return morphcontainer.AsContainerSource(cc), nil
|
|
|
|
}), nil
|
|
|
|
}
|
|
|
|
|
2024-08-28 15:32:30 +00:00
|
|
|
func initUpgradeCommand() {
|
|
|
|
flags := UpgradeCmd.Flags()
|
|
|
|
flags.Bool(noCompactFlag, false, "Do not compact upgraded metabase file")
|
|
|
|
}
|