2021-08-10 08:45:01 +00:00
|
|
|
package morph
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2022-03-18 11:12:58 +00:00
|
|
|
"os"
|
2021-08-10 09:34:27 +00:00
|
|
|
"sort"
|
2021-08-10 08:45:01 +00:00
|
|
|
|
2023-03-07 13:38:26 +00:00
|
|
|
cid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/id"
|
2021-08-10 08:46:06 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/hash"
|
2021-08-10 08:45:01 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
2022-09-05 12:45:38 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"
|
2021-08-10 08:45:01 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/callflag"
|
2021-08-10 08:51:11 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
2021-08-10 08:45:01 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/emit"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/spf13/viper"
|
|
|
|
)
|
|
|
|
|
|
|
|
var errInvalidContainerResponse = errors.New("invalid response from container contract")
|
|
|
|
|
2022-10-26 10:58:11 +00:00
|
|
|
func getContainerContractHash(cmd *cobra.Command, inv *invoker.Invoker, c Client) (util.Uint160, error) {
|
|
|
|
s, err := cmd.Flags().GetString(containerContractFlag)
|
|
|
|
var ch util.Uint160
|
|
|
|
if err == nil {
|
|
|
|
ch, err = util.Uint160DecodeStringLE(s)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
nnsCs, err := c.GetContractStateByID(1)
|
|
|
|
if err != nil {
|
|
|
|
return util.Uint160{}, fmt.Errorf("can't get NNS contract state: %w", err)
|
|
|
|
}
|
2022-12-23 17:35:35 +00:00
|
|
|
ch, err = nnsResolveHash(inv, nnsCs.Hash, containerContract+".frostfs")
|
2022-10-26 10:58:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return util.Uint160{}, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ch, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getContainersList(inv *invoker.Invoker, ch util.Uint160) ([][]byte, error) {
|
|
|
|
res, err := inv.Call(ch, "list", "")
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("%w: %v", errInvalidContainerResponse, err)
|
|
|
|
}
|
|
|
|
itm, err := unwrap.Item(res, err)
|
|
|
|
if _, ok := itm.(stackitem.Null); !ok {
|
|
|
|
return unwrap.ArrayOfBytes(res, err)
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2021-08-10 08:45:01 +00:00
|
|
|
func dumpContainers(cmd *cobra.Command, _ []string) error {
|
|
|
|
filename, err := cmd.Flags().GetString(containerDumpFlag)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("invalid filename: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := getN3Client(viper.GetViper())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("can't create N3 client: %w", err)
|
|
|
|
}
|
|
|
|
|
2022-09-05 12:45:38 +00:00
|
|
|
inv := invoker.New(c, nil)
|
|
|
|
|
2022-10-26 10:58:11 +00:00
|
|
|
ch, err := getContainerContractHash(cmd, inv, c)
|
2021-08-10 08:45:01 +00:00
|
|
|
if err != nil {
|
2022-10-26 10:58:11 +00:00
|
|
|
return fmt.Errorf("unable to get contaract hash: %w", err)
|
2021-08-10 08:45:01 +00:00
|
|
|
}
|
|
|
|
|
2022-10-26 10:58:11 +00:00
|
|
|
cids, err := getContainersList(inv, ch)
|
2021-08-10 08:45:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%w: %v", errInvalidContainerResponse, err)
|
|
|
|
}
|
|
|
|
|
2021-08-10 09:34:27 +00:00
|
|
|
isOK, err := getCIDFilterFunc(cmd)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-08-10 08:45:01 +00:00
|
|
|
var containers []*Container
|
|
|
|
bw := io.NewBufBinWriter()
|
|
|
|
for _, id := range cids {
|
2021-08-10 09:34:27 +00:00
|
|
|
if !isOK(id) {
|
|
|
|
continue
|
|
|
|
}
|
2021-08-10 08:45:01 +00:00
|
|
|
bw.Reset()
|
|
|
|
emit.AppCall(bw.BinWriter, ch, "get", callflag.All, id)
|
|
|
|
emit.AppCall(bw.BinWriter, ch, "eACL", callflag.All, id)
|
2022-09-05 12:45:38 +00:00
|
|
|
res, err := inv.Run(bw.Bytes())
|
2021-08-10 08:45:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("can't get container info: %w", err)
|
|
|
|
}
|
|
|
|
if len(res.Stack) != 2 {
|
|
|
|
return fmt.Errorf("%w: expected 2 items on stack", errInvalidContainerResponse)
|
|
|
|
}
|
|
|
|
|
|
|
|
cnt := new(Container)
|
|
|
|
err = cnt.FromStackItem(res.Stack[0])
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%w: %v", errInvalidContainerResponse, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ea := new(EACL)
|
|
|
|
err = ea.FromStackItem(res.Stack[1])
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%w: %v", errInvalidContainerResponse, err)
|
|
|
|
}
|
|
|
|
if len(ea.Value) != 0 {
|
|
|
|
cnt.EACL = ea
|
|
|
|
}
|
|
|
|
|
|
|
|
containers = append(containers, cnt)
|
|
|
|
}
|
|
|
|
|
|
|
|
out, err := json.Marshal(containers)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-03-18 11:12:58 +00:00
|
|
|
return os.WriteFile(filename, out, 0o660)
|
2021-08-10 08:45:01 +00:00
|
|
|
}
|
|
|
|
|
2022-10-26 10:58:11 +00:00
|
|
|
func listContainers(cmd *cobra.Command, _ []string) error {
|
|
|
|
c, err := getN3Client(viper.GetViper())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("can't create N3 client: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
inv := invoker.New(c, nil)
|
|
|
|
|
|
|
|
ch, err := getContainerContractHash(cmd, inv, c)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to get contaract hash: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cids, err := getContainersList(inv, ch)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%w: %v", errInvalidContainerResponse, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, id := range cids {
|
|
|
|
var idCnr cid.ID
|
|
|
|
err = idCnr.Decode(id)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to decode container id: %w", err)
|
|
|
|
}
|
|
|
|
cmd.Println(idCnr)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-08-10 08:46:06 +00:00
|
|
|
func restoreContainers(cmd *cobra.Command, _ []string) error {
|
|
|
|
filename, err := cmd.Flags().GetString(containerDumpFlag)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("invalid filename: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
wCtx, err := newInitializeContext(cmd, viper.GetViper())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-10-24 12:49:13 +00:00
|
|
|
defer wCtx.close()
|
2021-08-10 08:46:06 +00:00
|
|
|
|
2023-03-22 14:25:35 +00:00
|
|
|
containers, err := parseContainers(filename)
|
2021-08-10 08:46:06 +00:00
|
|
|
if err != nil {
|
2023-03-22 14:25:35 +00:00
|
|
|
return err
|
2021-08-10 08:46:06 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 14:25:35 +00:00
|
|
|
ch, err := fetchContainerContractHash(wCtx)
|
2021-08-10 08:46:06 +00:00
|
|
|
if err != nil {
|
2023-03-22 14:25:35 +00:00
|
|
|
return err
|
2021-08-10 08:46:06 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 14:25:35 +00:00
|
|
|
isOK, err := getCIDFilterFunc(cmd)
|
2021-08-10 08:46:06 +00:00
|
|
|
if err != nil {
|
2023-03-22 14:25:35 +00:00
|
|
|
return err
|
2021-08-10 08:46:06 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 14:25:35 +00:00
|
|
|
err = restoreOrPutContainers(containers, isOK, cmd, wCtx, ch)
|
2021-08-10 09:34:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-03-22 14:25:35 +00:00
|
|
|
return wCtx.awaitTx()
|
|
|
|
}
|
|
|
|
|
|
|
|
func restoreOrPutContainers(containers []Container, isOK func([]byte) bool, cmd *cobra.Command, wCtx *initializeContext, ch util.Uint160) error {
|
2021-08-10 08:46:06 +00:00
|
|
|
bw := io.NewBufBinWriter()
|
|
|
|
for _, cnt := range containers {
|
|
|
|
hv := hash.Sha256(cnt.Value)
|
2021-08-10 09:34:27 +00:00
|
|
|
if !isOK(hv[:]) {
|
|
|
|
continue
|
|
|
|
}
|
2021-08-10 08:46:06 +00:00
|
|
|
bw.Reset()
|
2023-03-22 14:25:35 +00:00
|
|
|
restored, err := isContainerRestored(cmd, wCtx, ch, bw, hv)
|
2021-08-10 08:46:06 +00:00
|
|
|
if err != nil {
|
2023-03-22 14:25:35 +00:00
|
|
|
return err
|
2021-08-10 08:46:06 +00:00
|
|
|
}
|
2023-03-22 14:25:35 +00:00
|
|
|
if restored {
|
2021-08-10 08:46:06 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
bw.Reset()
|
2023-03-22 14:25:35 +00:00
|
|
|
|
|
|
|
putContainer(bw, ch, cnt)
|
|
|
|
|
2021-08-10 08:46:06 +00:00
|
|
|
if bw.Err != nil {
|
|
|
|
panic(bw.Err)
|
|
|
|
}
|
|
|
|
|
2022-10-24 12:49:13 +00:00
|
|
|
if err := wCtx.sendConsensusTx(bw.Bytes()); err != nil {
|
2021-08-10 08:46:06 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2023-03-22 14:25:35 +00:00
|
|
|
return nil
|
|
|
|
}
|
2021-08-10 08:46:06 +00:00
|
|
|
|
2023-03-22 14:25:35 +00:00
|
|
|
func putContainer(bw *io.BufBinWriter, ch util.Uint160, cnt Container) {
|
|
|
|
emit.AppCall(bw.BinWriter, ch, "put", callflag.All,
|
|
|
|
cnt.Value, cnt.Signature, cnt.PublicKey, cnt.Token)
|
|
|
|
if ea := cnt.EACL; ea != nil {
|
|
|
|
emit.AppCall(bw.BinWriter, ch, "setEACL", callflag.All,
|
|
|
|
ea.Value, ea.Signature, ea.PublicKey, ea.Token)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func isContainerRestored(cmd *cobra.Command, wCtx *initializeContext, containerHash util.Uint160, bw *io.BufBinWriter, hashValue util.Uint256) (bool, error) {
|
|
|
|
emit.AppCall(bw.BinWriter, containerHash, "get", callflag.All, hashValue.BytesBE())
|
|
|
|
res, err := wCtx.Client.InvokeScript(bw.Bytes(), nil)
|
|
|
|
if err != nil {
|
|
|
|
return false, fmt.Errorf("can't check if container is already restored: %w", err)
|
|
|
|
}
|
|
|
|
if len(res.Stack) == 0 {
|
|
|
|
return false, errors.New("empty stack")
|
|
|
|
}
|
|
|
|
|
|
|
|
old := new(Container)
|
|
|
|
if err := old.FromStackItem(res.Stack[0]); err != nil {
|
|
|
|
return false, fmt.Errorf("%w: %v", errInvalidContainerResponse, err)
|
|
|
|
}
|
|
|
|
if len(old.Value) != 0 {
|
|
|
|
var id cid.ID
|
|
|
|
id.SetSHA256(hashValue)
|
|
|
|
cmd.Printf("Container %s is already deployed.\n", id)
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseContainers(filename string) ([]Container, error) {
|
|
|
|
data, err := os.ReadFile(filename)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("can't read dump file: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var containers []Container
|
|
|
|
err = json.Unmarshal(data, &containers)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("can't parse dump file: %w", err)
|
|
|
|
}
|
|
|
|
return containers, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func fetchContainerContractHash(wCtx *initializeContext) (util.Uint160, error) {
|
|
|
|
nnsCs, err := wCtx.Client.GetContractStateByID(1)
|
|
|
|
if err != nil {
|
|
|
|
return util.Uint160{}, fmt.Errorf("can't get NNS contract state: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ch, err := nnsResolveHash(wCtx.ReadOnlyInvoker, nnsCs.Hash, containerContract+".frostfs")
|
|
|
|
if err != nil {
|
|
|
|
return util.Uint160{}, fmt.Errorf("can't fetch container contract hash: %w", err)
|
|
|
|
}
|
|
|
|
return ch, nil
|
2021-08-10 08:46:06 +00:00
|
|
|
}
|
|
|
|
|
2021-08-10 08:45:01 +00:00
|
|
|
// Container represents container struct in contract storage.
|
|
|
|
type Container struct {
|
|
|
|
Value []byte `json:"value"`
|
|
|
|
Signature []byte `json:"signature"`
|
|
|
|
PublicKey []byte `json:"public_key"`
|
|
|
|
Token []byte `json:"token"`
|
|
|
|
EACL *EACL `json:"eacl"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// EACL represents extended ACL struct in contract storage.
|
|
|
|
type EACL struct {
|
|
|
|
Value []byte `json:"value"`
|
|
|
|
Signature []byte `json:"signature"`
|
|
|
|
PublicKey []byte `json:"public_key"`
|
|
|
|
Token []byte `json:"token"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToStackItem implements stackitem.Convertible.
|
|
|
|
func (c *Container) ToStackItem() (stackitem.Item, error) {
|
|
|
|
return stackitem.NewStruct([]stackitem.Item{
|
|
|
|
stackitem.NewByteArray(c.Value),
|
|
|
|
stackitem.NewByteArray(c.Signature),
|
|
|
|
stackitem.NewByteArray(c.PublicKey),
|
|
|
|
stackitem.NewByteArray(c.Token),
|
|
|
|
}), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromStackItem implements stackitem.Convertible.
|
|
|
|
func (c *Container) FromStackItem(item stackitem.Item) error {
|
|
|
|
arr, ok := item.Value().([]stackitem.Item)
|
|
|
|
if !ok || len(arr) != 4 {
|
|
|
|
return errors.New("invalid stack item type")
|
|
|
|
}
|
|
|
|
|
|
|
|
value, err := arr[0].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid container value")
|
|
|
|
}
|
|
|
|
|
|
|
|
sig, err := arr[1].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid container signature")
|
|
|
|
}
|
|
|
|
|
|
|
|
pub, err := arr[2].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid container public key")
|
|
|
|
}
|
|
|
|
|
|
|
|
tok, err := arr[3].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid container token")
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Value = value
|
|
|
|
c.Signature = sig
|
|
|
|
c.PublicKey = pub
|
|
|
|
c.Token = tok
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToStackItem implements stackitem.Convertible.
|
|
|
|
func (c *EACL) ToStackItem() (stackitem.Item, error) {
|
|
|
|
return stackitem.NewStruct([]stackitem.Item{
|
|
|
|
stackitem.NewByteArray(c.Value),
|
|
|
|
stackitem.NewByteArray(c.Signature),
|
|
|
|
stackitem.NewByteArray(c.PublicKey),
|
|
|
|
stackitem.NewByteArray(c.Token),
|
|
|
|
}), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromStackItem implements stackitem.Convertible.
|
|
|
|
func (c *EACL) FromStackItem(item stackitem.Item) error {
|
|
|
|
arr, ok := item.Value().([]stackitem.Item)
|
|
|
|
if !ok || len(arr) != 4 {
|
|
|
|
return errors.New("invalid stack item type")
|
|
|
|
}
|
|
|
|
|
|
|
|
value, err := arr[0].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid eACL value")
|
|
|
|
}
|
|
|
|
|
|
|
|
sig, err := arr[1].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid eACL signature")
|
|
|
|
}
|
|
|
|
|
|
|
|
pub, err := arr[2].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid eACL public key")
|
|
|
|
}
|
|
|
|
|
|
|
|
tok, err := arr[3].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("invalid eACL token")
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Value = value
|
|
|
|
c.Signature = sig
|
|
|
|
c.PublicKey = pub
|
|
|
|
c.Token = tok
|
|
|
|
return nil
|
|
|
|
}
|
2021-08-10 09:34:27 +00:00
|
|
|
|
|
|
|
// getCIDFilterFunc returns filtering function for container IDs.
|
|
|
|
// Raw byte slices are used because it works with structures returned
|
|
|
|
// from contract.
|
|
|
|
func getCIDFilterFunc(cmd *cobra.Command) (func([]byte) bool, error) {
|
|
|
|
rawIDs, err := cmd.Flags().GetStringSlice(containerIDsFlag)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(rawIDs) == 0 {
|
|
|
|
return func([]byte) bool { return true }, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range rawIDs {
|
2022-05-12 16:37:46 +00:00
|
|
|
err := new(cid.ID).DecodeString(rawIDs[i])
|
2021-08-10 09:34:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("can't parse CID %s: %w", rawIDs[i], err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sort.Strings(rawIDs)
|
|
|
|
return func(rawID []byte) bool {
|
|
|
|
var v [32]byte
|
|
|
|
copy(v[:], rawID)
|
|
|
|
|
2022-05-12 16:37:46 +00:00
|
|
|
var id cid.ID
|
2021-08-10 09:34:27 +00:00
|
|
|
id.SetSHA256(v)
|
2022-05-12 16:37:46 +00:00
|
|
|
idStr := id.EncodeToString()
|
2021-08-10 09:34:27 +00:00
|
|
|
n := sort.Search(len(rawIDs), func(i int) bool { return rawIDs[i] >= idStr })
|
|
|
|
return n < len(rawIDs) && rawIDs[n] == idStr
|
|
|
|
}, nil
|
|
|
|
}
|