forked from TrueCloudLab/frostfs-node
[#1035] neofs-adm: update contracts in a single transaction
Signed-off-by: Evgenii Stratonikov <evgeniy@nspcc.ru>
This commit is contained in:
parent
a8ba573ec8
commit
4673c81451
4 changed files with 126 additions and 65 deletions
|
@ -61,7 +61,7 @@ func initializeSideChainCmd(cmd *cobra.Command, args []string) error {
|
||||||
|
|
||||||
// 4. Deploy NeoFS contracts.
|
// 4. Deploy NeoFS contracts.
|
||||||
cmd.Println("Stage 4: deploy NeoFS contracts.")
|
cmd.Println("Stage 4: deploy NeoFS contracts.")
|
||||||
if err := initCtx.deployContracts(deployMethodName); err != nil {
|
if err := initCtx.deployContracts(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,9 +12,11 @@ import (
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/native"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/native/nativenames"
|
"github.com/nspcc-dev/neo-go/pkg/core/native/nativenames"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||||||
|
io2 "github.com/nspcc-dev/neo-go/pkg/io"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/rpc/client"
|
"github.com/nspcc-dev/neo-go/pkg/rpc/client"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
||||||
|
@ -137,7 +139,7 @@ func (c *initializeContext) deployNNS(method string) error {
|
||||||
return c.awaitTx()
|
return c.awaitTx()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *initializeContext) deployContracts(method string) error {
|
func (c *initializeContext) updateContracts() error {
|
||||||
mgmtHash := c.nativeHash(nativenames.Management)
|
mgmtHash := c.nativeHash(nativenames.Management)
|
||||||
alphaCs := c.getContract(alphabetContract)
|
alphaCs := c.getContract(alphabetContract)
|
||||||
|
|
||||||
|
@ -147,27 +149,118 @@ func (c *initializeContext) deployContracts(method string) error {
|
||||||
}
|
}
|
||||||
nnsHash := nnsCs.Hash
|
nnsHash := nnsCs.Hash
|
||||||
|
|
||||||
|
totalGasCost := int64(0)
|
||||||
|
w := io2.NewBufBinWriter()
|
||||||
|
|
||||||
var keysParam []smartcontract.Parameter
|
var keysParam []smartcontract.Parameter
|
||||||
|
|
||||||
// alphabet contracts should be deployed by individual nodes to get different hashes.
|
// alphabet contracts should be deployed by individual nodes to get different hashes.
|
||||||
for i, acc := range c.Accounts {
|
for i, acc := range c.Accounts {
|
||||||
ctrHash := state.CreateContractHash(acc.Contract.ScriptHash(), alphaCs.NEF.Checksum, alphaCs.Manifest.Name)
|
ctrHash, err := nnsResolveHash(c.Client, nnsHash, getAlphabetNNSDomain(i))
|
||||||
if method == updateMethodName {
|
if err != nil {
|
||||||
ctrHash, err = nnsResolveHash(c.Client, nnsHash, getAlphabetNNSDomain(i))
|
return fmt.Errorf("can't resolve hash for contract update: %w", err)
|
||||||
if err != nil {
|
}
|
||||||
|
|
||||||
|
keysParam = append(keysParam, smartcontract.Parameter{
|
||||||
|
Type: smartcontract.PublicKeyType,
|
||||||
|
Value: acc.PrivateKey().PublicKey().Bytes(),
|
||||||
|
})
|
||||||
|
|
||||||
|
params := getContractDeployParameters(alphaCs.RawNEF, alphaCs.RawManifest,
|
||||||
|
c.getAlphabetDeployParameters(i, len(c.Wallets)))
|
||||||
|
signer := transaction.Signer{
|
||||||
|
Account: c.CommitteeAcc.Contract.ScriptHash(),
|
||||||
|
Scopes: transaction.CalledByEntry,
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := c.Client.InvokeFunction(ctrHash, updateMethodName, params, []transaction.Signer{signer})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("can't deploy alphabet #%d contract: %w", i, err)
|
||||||
|
}
|
||||||
|
if res.State != vm.HaltState.String() {
|
||||||
|
return fmt.Errorf("can't deploy alpabet #%d contract: %s", i, res.FaultException)
|
||||||
|
}
|
||||||
|
|
||||||
|
totalGasCost += res.GasConsumed
|
||||||
|
w.WriteBytes(res.Script)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, ctrName := range contractList {
|
||||||
|
cs := c.getContract(ctrName)
|
||||||
|
|
||||||
|
method := updateMethodName
|
||||||
|
ctrHash, err := nnsResolveHash(c.Client, nnsHash, ctrName+".neofs")
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, errMissingNNSRecord) {
|
||||||
|
// if contract not found we deploy it instead of update
|
||||||
|
method = deployMethodName
|
||||||
|
} else {
|
||||||
return fmt.Errorf("can't resolve hash for contract update: %w", err)
|
return fmt.Errorf("can't resolve hash for contract update: %w", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if c.isUpdated(ctrHash, alphaCs) {
|
|
||||||
c.Command.Printf("Alphabet contract #%d is already deployed.\n", i)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
invokeHash := mgmtHash
|
invokeHash := mgmtHash
|
||||||
if method == updateMethodName {
|
if method == updateMethodName {
|
||||||
invokeHash = ctrHash
|
invokeHash = ctrHash
|
||||||
}
|
}
|
||||||
|
|
||||||
|
params := getContractDeployParameters(cs.RawNEF, cs.RawManifest,
|
||||||
|
c.getContractDeployData(ctrName, keysParam))
|
||||||
|
signer := transaction.Signer{
|
||||||
|
Account: c.CommitteeAcc.Contract.ScriptHash(),
|
||||||
|
Scopes: transaction.Global,
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := c.Client.InvokeFunction(invokeHash, method, params, []transaction.Signer{signer})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("can't deploy %s contract: %w", ctrName, err)
|
||||||
|
}
|
||||||
|
if res.State != vm.HaltState.String() {
|
||||||
|
return fmt.Errorf("can't deploy %s contract: %s", ctrName, res.FaultException)
|
||||||
|
}
|
||||||
|
|
||||||
|
totalGasCost += res.GasConsumed
|
||||||
|
w.WriteBytes(res.Script)
|
||||||
|
|
||||||
|
if method == deployMethodName {
|
||||||
|
// same actions are done in initializeContext.setNNS, can be unified
|
||||||
|
domain := ctrName + ".neofs"
|
||||||
|
script, err := c.nnsRegisterDomainScript(nnsHash, cs.Hash, domain)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if script != nil {
|
||||||
|
totalGasCost += defaultRegisterSysfee + native.GASFactor
|
||||||
|
w.WriteBytes(script)
|
||||||
|
}
|
||||||
|
c.Command.Printf("NNS: Set %s -> %s\n", domain, cs.Hash.StringLE())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.Err != nil {
|
||||||
|
return w.Err
|
||||||
|
}
|
||||||
|
if err := c.sendCommitteeTx(w.Bytes(), totalGasCost); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return c.awaitTx()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *initializeContext) deployContracts() error {
|
||||||
|
mgmtHash := c.nativeHash(nativenames.Management)
|
||||||
|
alphaCs := c.getContract(alphabetContract)
|
||||||
|
|
||||||
|
var keysParam []smartcontract.Parameter
|
||||||
|
|
||||||
|
// alphabet contracts should be deployed by individual nodes to get different hashes.
|
||||||
|
for i, acc := range c.Accounts {
|
||||||
|
ctrHash := state.CreateContractHash(acc.Contract.ScriptHash(), alphaCs.NEF.Checksum, alphaCs.Manifest.Name)
|
||||||
|
if c.isUpdated(ctrHash, alphaCs) {
|
||||||
|
c.Command.Printf("Alphabet contract #%d is already deployed.\n", i)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
invokeHash := mgmtHash
|
||||||
keysParam = append(keysParam, smartcontract.Parameter{
|
keysParam = append(keysParam, smartcontract.Parameter{
|
||||||
Type: smartcontract.PublicKeyType,
|
Type: smartcontract.PublicKeyType,
|
||||||
Value: acc.PrivateKey().PublicKey().Bytes(),
|
Value: acc.PrivateKey().PublicKey().Bytes(),
|
||||||
|
@ -179,14 +272,8 @@ func (c *initializeContext) deployContracts(method string) error {
|
||||||
Account: acc.Contract.ScriptHash(),
|
Account: acc.Contract.ScriptHash(),
|
||||||
Scopes: transaction.CalledByEntry,
|
Scopes: transaction.CalledByEntry,
|
||||||
}
|
}
|
||||||
if method == updateMethodName {
|
|
||||||
signer = transaction.Signer{
|
|
||||||
Account: c.CommitteeAcc.Contract.ScriptHash(),
|
|
||||||
Scopes: transaction.CalledByEntry,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
res, err := c.Client.InvokeFunction(invokeHash, method, params, []transaction.Signer{signer})
|
res, err := c.Client.InvokeFunction(invokeHash, deployMethodName, params, []transaction.Signer{signer})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("can't deploy alphabet #%d contract: %w", i, err)
|
return fmt.Errorf("can't deploy alphabet #%d contract: %w", i, err)
|
||||||
}
|
}
|
||||||
|
@ -194,51 +281,27 @@ func (c *initializeContext) deployContracts(method string) error {
|
||||||
return fmt.Errorf("can't deploy alpabet #%d contract: %s", i, res.FaultException)
|
return fmt.Errorf("can't deploy alpabet #%d contract: %s", i, res.FaultException)
|
||||||
}
|
}
|
||||||
|
|
||||||
if method == updateMethodName {
|
h, err := c.Client.SignAndPushInvocationTx(res.Script, acc, -1, 0, []client.SignerAccount{{
|
||||||
if err := c.sendCommitteeTx(res.Script, res.GasConsumed); err != nil {
|
Signer: signer,
|
||||||
return err
|
Account: acc,
|
||||||
}
|
}})
|
||||||
} else {
|
if err != nil {
|
||||||
h, err := c.Client.SignAndPushInvocationTx(res.Script, acc, -1, 0, []client.SignerAccount{{
|
return fmt.Errorf("can't push deploy transaction: %w", err)
|
||||||
Signer: signer,
|
|
||||||
Account: acc,
|
|
||||||
}})
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("can't push deploy transaction: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
c.Hashes = append(c.Hashes, h)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
c.Hashes = append(c.Hashes, h)
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, ctrName := range contractList {
|
for _, ctrName := range contractList {
|
||||||
cs := c.getContract(ctrName)
|
cs := c.getContract(ctrName)
|
||||||
|
|
||||||
ctrHash := cs.Hash
|
ctrHash := cs.Hash
|
||||||
|
|
||||||
methodCur := method // prevent overriding in if-block
|
|
||||||
|
|
||||||
if method == updateMethodName {
|
|
||||||
ctrHash, err = nnsResolveHash(c.Client, nnsHash, ctrName+".neofs")
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, errMissingNNSRecord) {
|
|
||||||
// if contract not found we deploy it instead of update
|
|
||||||
methodCur = deployMethodName
|
|
||||||
} else {
|
|
||||||
return fmt.Errorf("can't resolve hash for contract update: %w", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if c.isUpdated(ctrHash, cs) {
|
if c.isUpdated(ctrHash, cs) {
|
||||||
c.Command.Printf("%s contract is already deployed.\n", ctrName)
|
c.Command.Printf("%s contract is already deployed.\n", ctrName)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
invokeHash := mgmtHash
|
invokeHash := mgmtHash
|
||||||
if methodCur == updateMethodName {
|
|
||||||
invokeHash = ctrHash
|
|
||||||
}
|
|
||||||
|
|
||||||
params := getContractDeployParameters(cs.RawNEF, cs.RawManifest,
|
params := getContractDeployParameters(cs.RawNEF, cs.RawManifest,
|
||||||
c.getContractDeployData(ctrName, keysParam))
|
c.getContractDeployData(ctrName, keysParam))
|
||||||
signer := transaction.Signer{
|
signer := transaction.Signer{
|
||||||
|
@ -246,7 +309,7 @@ func (c *initializeContext) deployContracts(method string) error {
|
||||||
Scopes: transaction.Global,
|
Scopes: transaction.Global,
|
||||||
}
|
}
|
||||||
|
|
||||||
res, err := c.Client.InvokeFunction(invokeHash, methodCur, params, []transaction.Signer{signer})
|
res, err := c.Client.InvokeFunction(invokeHash, deployMethodName, params, []transaction.Signer{signer})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("can't deploy %s contract: %w", ctrName, err)
|
return fmt.Errorf("can't deploy %s contract: %w", ctrName, err)
|
||||||
}
|
}
|
||||||
|
@ -257,15 +320,6 @@ func (c *initializeContext) deployContracts(method string) error {
|
||||||
if err := c.sendCommitteeTx(res.Script, res.GasConsumed); err != nil {
|
if err := c.sendCommitteeTx(res.Script, res.GasConsumed); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if method == updateMethodName && methodCur == deployMethodName {
|
|
||||||
// same actions are done in initializeContext.setNNS, can be unified
|
|
||||||
domain := ctrName + ".neofs"
|
|
||||||
if err := c.nnsRegisterDomain(nnsCs.Hash, cs.Hash, domain); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
c.Command.Printf("NNS: Set %s -> %s\n", domain, cs.Hash.StringLE())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return c.awaitTx()
|
return c.awaitTx()
|
||||||
|
|
|
@ -75,10 +75,10 @@ func getAlphabetNNSDomain(i int) string {
|
||||||
return alphabetContract + strconv.FormatUint(uint64(i), 10) + ".neofs"
|
return alphabetContract + strconv.FormatUint(uint64(i), 10) + ".neofs"
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *initializeContext) nnsRegisterDomain(nnsHash, expectedHash util.Uint160, domain string) error {
|
func (c *initializeContext) nnsRegisterDomainScript(nnsHash, expectedHash util.Uint160, domain string) ([]byte, error) {
|
||||||
ok, err := c.Client.NNSIsAvailable(nnsHash, domain)
|
ok, err := c.Client.NNSIsAvailable(nnsHash, domain)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
bw := io.NewBufBinWriter()
|
bw := io.NewBufBinWriter()
|
||||||
|
@ -90,10 +90,10 @@ func (c *initializeContext) nnsRegisterDomain(nnsHash, expectedHash util.Uint160
|
||||||
} else {
|
} else {
|
||||||
s, err := nnsResolveHash(c.Client, nnsHash, domain)
|
s, err := nnsResolveHash(c.Client, nnsHash, domain)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return nil, err
|
||||||
}
|
}
|
||||||
if s == expectedHash {
|
if s == expectedHash {
|
||||||
return nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,9 +103,16 @@ func (c *initializeContext) nnsRegisterDomain(nnsHash, expectedHash util.Uint160
|
||||||
if bw.Err != nil {
|
if bw.Err != nil {
|
||||||
panic(bw.Err)
|
panic(bw.Err)
|
||||||
}
|
}
|
||||||
|
return bw.Bytes(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *initializeContext) nnsRegisterDomain(nnsHash, expectedHash util.Uint160, domain string) error {
|
||||||
|
script, err := c.nnsRegisterDomainScript(nnsHash, expectedHash, domain)
|
||||||
|
if script == nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
sysFee := int64(defaultRegisterSysfee + native.GASFactor)
|
sysFee := int64(defaultRegisterSysfee + native.GASFactor)
|
||||||
return c.sendCommitteeTx(bw.Bytes(), sysFee)
|
return c.sendCommitteeTx(script, sysFee)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *initializeContext) nnsRootRegistered(nnsHash util.Uint160) (bool, error) {
|
func (c *initializeContext) nnsRootRegistered(nnsHash util.Uint160) (bool, error) {
|
||||||
|
|
|
@ -17,5 +17,5 @@ func updateContracts(cmd *cobra.Command, _ []string) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
return wCtx.deployContracts(updateMethodName)
|
return wCtx.updateContracts()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue