package control

import (
	"git.frostfs.info/TrueCloudLab/frostfs-api-go/v2/rpc/client"
	"git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-cli/internal/key"
	commonCmd "git.frostfs.info/TrueCloudLab/frostfs-node/cmd/internal/common"
	"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/services/control"
	"github.com/spf13/cobra"
)

const ignoreErrorsFlag = "no-errors"

var evacuateShardCmd = &cobra.Command{
	Use:   "evacuate",
	Short: "Evacuate objects from shard",
	Long:  "Evacuate objects from shard to other shards",
	Run:   evacuateShard,
}

func evacuateShard(cmd *cobra.Command, _ []string) {
	pk := key.Get(cmd)

	req := &control.EvacuateShardRequest{Body: new(control.EvacuateShardRequest_Body)}
	req.Body.Shard_ID = getShardIDList(cmd)
	req.Body.IgnoreErrors, _ = cmd.Flags().GetBool(ignoreErrorsFlag)

	signRequest(cmd, pk, req)

	cli := getClient(cmd, pk)

	var resp *control.EvacuateShardResponse
	var err error
	err = cli.ExecRaw(func(client *client.Client) error {
		resp, err = control.EvacuateShard(client, req)
		return err
	})
	commonCmd.ExitOnErr(cmd, "rpc error: %w", err)

	cmd.Printf("Objects moved: %d\n", resp.GetBody().GetCount())

	verifyResponse(cmd, resp.GetSignature(), resp.GetBody())

	cmd.Println("Shard has successfully been evacuated.")
}

func initControlEvacuateShardCmd() {
	initControlFlags(evacuateShardCmd)

	flags := evacuateShardCmd.Flags()
	flags.StringSlice(shardIDFlag, nil, "List of shard IDs in base58 encoding")
	flags.Bool(shardAllFlag, false, "Process all shards")
	flags.Bool(ignoreErrorsFlag, false, "Skip invalid/unreadable objects")

	evacuateShardCmd.MarkFlagsMutuallyExclusive(shardIDFlag, shardAllFlag)
}