2024-02-02 12:36:14 +00:00
|
|
|
package helper
|
2024-02-01 14:28:10 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
|
2024-02-02 12:04:48 +00:00
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-adm/internal/commonflags"
|
2024-02-02 12:26:57 +00:00
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/cmd/frostfs-adm/internal/modules/morph/constants"
|
2024-02-01 14:28:10 +00:00
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/morph/client/netmap"
|
2024-02-01 14:53:00 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
2024-02-02 06:49:03 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/management"
|
2024-02-01 14:53:00 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/callflag"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/emit"
|
2024-02-01 14:28:10 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
|
|
"github.com/spf13/viper"
|
|
|
|
)
|
|
|
|
|
2024-02-02 06:49:03 +00:00
|
|
|
var NetmapConfigKeys = []string{
|
|
|
|
netmap.EpochDurationConfig,
|
|
|
|
netmap.MaxObjectSizeConfig,
|
|
|
|
netmap.ContainerFeeConfig,
|
|
|
|
netmap.ContainerAliasFeeConfig,
|
|
|
|
netmap.IrCandidateFeeConfig,
|
|
|
|
netmap.WithdrawFeeConfig,
|
|
|
|
netmap.HomomorphicHashingDisabledKey,
|
|
|
|
netmap.MaintenanceModeAllowedConfig,
|
|
|
|
}
|
|
|
|
|
2024-03-11 14:55:50 +00:00
|
|
|
var errFailedToFetchListOfNetworkKeys = errors.New("can't fetch list of network config keys from the netmap contract")
|
|
|
|
|
2024-02-01 14:28:10 +00:00
|
|
|
func GetDefaultNetmapContractConfigMap() map[string]any {
|
|
|
|
m := make(map[string]any)
|
2024-02-02 12:04:48 +00:00
|
|
|
m[netmap.EpochDurationConfig] = viper.GetInt64(commonflags.EpochDurationInitFlag)
|
|
|
|
m[netmap.MaxObjectSizeConfig] = viper.GetInt64(commonflags.MaxObjectSizeInitFlag)
|
2024-04-08 09:27:30 +00:00
|
|
|
m[netmap.MaxECDataCountConfig] = viper.GetInt64(commonflags.MaxECDataCountFlag)
|
|
|
|
m[netmap.MaxECParityCountConfig] = viper.GetInt64(commonflags.MaxECParityCounFlag)
|
2024-02-02 12:04:48 +00:00
|
|
|
m[netmap.ContainerFeeConfig] = viper.GetInt64(commonflags.ContainerFeeInitFlag)
|
|
|
|
m[netmap.ContainerAliasFeeConfig] = viper.GetInt64(commonflags.ContainerAliasFeeInitFlag)
|
|
|
|
m[netmap.IrCandidateFeeConfig] = viper.GetInt64(commonflags.CandidateFeeInitFlag)
|
|
|
|
m[netmap.WithdrawFeeConfig] = viper.GetInt64(commonflags.WithdrawFeeInitFlag)
|
|
|
|
m[netmap.HomomorphicHashingDisabledKey] = viper.GetBool(commonflags.HomomorphicHashDisabledInitFlag)
|
|
|
|
m[netmap.MaintenanceModeAllowedConfig] = viper.GetBool(commonflags.MaintenanceModeAllowedInitFlag)
|
2024-02-01 14:28:10 +00:00
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
func ParseConfigFromNetmapContract(arr []stackitem.Item) (map[string][]byte, error) {
|
|
|
|
m := make(map[string][]byte, len(arr))
|
|
|
|
for _, param := range arr {
|
|
|
|
tuple, ok := param.Value().([]stackitem.Item)
|
|
|
|
if !ok || len(tuple) != 2 {
|
|
|
|
return nil, errors.New("invalid ListConfig response from netmap contract")
|
|
|
|
}
|
|
|
|
|
|
|
|
k, err := tuple[0].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("invalid config key from netmap contract")
|
|
|
|
}
|
|
|
|
|
|
|
|
v, err := tuple[1].TryBytes()
|
|
|
|
if err != nil {
|
|
|
|
return nil, InvalidConfigValueErr(string(k))
|
|
|
|
}
|
|
|
|
m[string(k)] = v
|
|
|
|
}
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func InvalidConfigValueErr(key string) error {
|
|
|
|
return fmt.Errorf("invalid %s config value from netmap contract", key)
|
|
|
|
}
|
2024-02-01 14:53:00 +00:00
|
|
|
|
2024-09-18 15:14:54 +00:00
|
|
|
func EmitNewEpochCall(bw *io.BufBinWriter, wCtx *InitializeContext, nmHash util.Uint160, countEpoch int64) error {
|
|
|
|
if countEpoch <= 0 {
|
|
|
|
return errors.New("number of epochs cannot be less than 1")
|
|
|
|
}
|
|
|
|
|
2024-02-01 14:53:00 +00:00
|
|
|
curr, err := unwrap.Int64(wCtx.ReadOnlyInvoker.Call(nmHash, "epoch"))
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("can't fetch current epoch from the netmap contract")
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:14:54 +00:00
|
|
|
newEpoch := curr + countEpoch
|
2024-02-01 14:53:00 +00:00
|
|
|
wCtx.Command.Printf("Current epoch: %d, increase to %d.\n", curr, newEpoch)
|
|
|
|
|
|
|
|
// In NeoFS this is done via Notary contract. Here, however, we can form the
|
|
|
|
// transaction locally.
|
|
|
|
emit.AppCall(bw.BinWriter, nmHash, "newEpoch", callflag.All, newEpoch)
|
|
|
|
return bw.Err
|
|
|
|
}
|
2024-02-02 06:49:03 +00:00
|
|
|
|
|
|
|
func GetNetConfigFromNetmapContract(roInvoker *invoker.Invoker) ([]stackitem.Item, error) {
|
|
|
|
r := management.NewReader(roInvoker)
|
2024-06-28 13:35:18 +00:00
|
|
|
cs, err := GetContractByID(r, 1)
|
2024-02-02 06:49:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("get nns contract: %w", err)
|
|
|
|
}
|
2024-02-02 12:26:57 +00:00
|
|
|
nmHash, err := NNSResolveHash(roInvoker, cs.Hash, DomainOf(constants.NetmapContract))
|
2024-02-02 06:49:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("can't get netmap contract hash: %w", err)
|
|
|
|
}
|
|
|
|
arr, err := unwrap.Array(roInvoker.Call(nmHash, "listConfig"))
|
|
|
|
if err != nil {
|
2024-03-11 14:55:50 +00:00
|
|
|
return nil, errFailedToFetchListOfNetworkKeys
|
2024-02-02 06:49:03 +00:00
|
|
|
}
|
|
|
|
return arr, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func MergeNetmapConfig(roInvoker *invoker.Invoker, md map[string]any) error {
|
|
|
|
arr, err := GetNetConfigFromNetmapContract(roInvoker)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
m, err := ParseConfigFromNetmapContract(arr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for k, v := range m {
|
|
|
|
for _, key := range NetmapConfigKeys {
|
|
|
|
if k == key {
|
|
|
|
md[k] = v
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|