forked from TrueCloudLab/neoneo-go
120 lines
4.4 KiB
Go
120 lines
4.4 KiB
Go
|
/*
|
||
|
Package rolemgmt allows to work with the native RoleManagement contract via RPC.
|
||
|
|
||
|
Safe methods are encapsulated into ContractReader structure while Contract provides
|
||
|
various methods to perform the only RoleManagement state-changing call.
|
||
|
*/
|
||
|
package rolemgmt
|
||
|
|
||
|
import (
|
||
|
"crypto/elliptic"
|
||
|
"fmt"
|
||
|
|
||
|
"github.com/nspcc-dev/neo-go/pkg/core/native/nativenames"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/core/native/noderoles"
|
||
|
"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/crypto/keys"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/neorpc/result"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||
|
)
|
||
|
|
||
|
// Invoker is used by ContractReader to call various methods.
|
||
|
type Invoker interface {
|
||
|
Call(contract util.Uint160, operation string, params ...interface{}) (*result.Invoke, error)
|
||
|
}
|
||
|
|
||
|
// Actor is used by Contract to create and send transactions.
|
||
|
type Actor interface {
|
||
|
Invoker
|
||
|
|
||
|
MakeCall(contract util.Uint160, method string, params ...interface{}) (*transaction.Transaction, error)
|
||
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...interface{}) (*transaction.Transaction, error)
|
||
|
SendCall(contract util.Uint160, method string, params ...interface{}) (util.Uint256, uint32, error)
|
||
|
}
|
||
|
|
||
|
// Hash stores the hash of the native RoleManagement contract.
|
||
|
var Hash = state.CreateNativeContractHash(nativenames.Designation)
|
||
|
|
||
|
const designateMethod = "designateAsRole"
|
||
|
|
||
|
// ContractReader provides an interface to call read-only RoleManagement
|
||
|
// contract's methods.
|
||
|
type ContractReader struct {
|
||
|
invoker Invoker
|
||
|
}
|
||
|
|
||
|
// Contract represents a RoleManagement contract client that can be used to
|
||
|
// invoke all of its methods.
|
||
|
type Contract struct {
|
||
|
ContractReader
|
||
|
|
||
|
actor Actor
|
||
|
}
|
||
|
|
||
|
// DesignationEvent represents an event emitted by RoleManagement contract when
|
||
|
// a new role designation is done.
|
||
|
type DesignationEvent struct {
|
||
|
Role noderoles.Role
|
||
|
BlockIndex uint32
|
||
|
}
|
||
|
|
||
|
// NewReader creates an instance of ContractReader that can be used to read
|
||
|
// data from the contract.
|
||
|
func NewReader(invoker Invoker) *ContractReader {
|
||
|
return &ContractReader{invoker}
|
||
|
}
|
||
|
|
||
|
// New creates an instance of Contract to perform actions using
|
||
|
// the given Actor. Notice that RoleManagement's state can be changed
|
||
|
// only by the network's committee, so the Actor provided must be a committee
|
||
|
// actor for designation methods to work properly.
|
||
|
func New(actor Actor) *Contract {
|
||
|
return &Contract{*NewReader(actor), actor}
|
||
|
}
|
||
|
|
||
|
// GetDesignatedByRole returns the list of the keys designated to serve for the
|
||
|
// given role at the given height. The list can be empty if no keys are
|
||
|
// configured for this role/height.
|
||
|
func (c *ContractReader) GetDesignatedByRole(role noderoles.Role, index uint32) (keys.PublicKeys, error) {
|
||
|
arr, err := unwrap.Array(c.invoker.Call(Hash, "getDesignatedByRole", int64(role), index))
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
pks := make(keys.PublicKeys, len(arr))
|
||
|
for i, item := range arr {
|
||
|
val, err := item.TryBytes()
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("invalid array element #%d: %s", i, item.Type())
|
||
|
}
|
||
|
pks[i], err = keys.NewPublicKeyFromBytes(val, elliptic.P256())
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
return pks, nil
|
||
|
}
|
||
|
|
||
|
// DesignateAsRole creates and sends a transaction that sets the keys used for
|
||
|
// the given node role. The action is successful when transaction ends in HALT
|
||
|
// state. The returned values are transaction hash, its ValidUntilBlock value
|
||
|
// and an error if any.
|
||
|
func (c *Contract) DesignateAsRole(role noderoles.Role, pubs keys.PublicKeys) (util.Uint256, uint32, error) {
|
||
|
return c.actor.SendCall(Hash, designateMethod, int(role), pubs)
|
||
|
}
|
||
|
|
||
|
// DesignateAsRoleTransaction creates a transaction that sets the keys for the
|
||
|
// given node role. This transaction is signed, but not sent to the network,
|
||
|
// instead it's returned to the caller.
|
||
|
func (c *Contract) DesignateAsRoleTransaction(role noderoles.Role, pubs keys.PublicKeys) (*transaction.Transaction, error) {
|
||
|
return c.actor.MakeCall(Hash, designateMethod, int(role), pubs)
|
||
|
}
|
||
|
|
||
|
// DesignateAsRoleUnsigned creates a transaction that sets the keys for the
|
||
|
// given node role. This transaction is not signed and just returned to the
|
||
|
// caller.
|
||
|
func (c *Contract) DesignateAsRoleUnsigned(role noderoles.Role, pubs keys.PublicKeys) (*transaction.Transaction, error) {
|
||
|
return c.actor.MakeUnsignedCall(Hash, designateMethod, nil, int(role), pubs)
|
||
|
}
|