forked from TrueCloudLab/neoneo-go
neotest: allow to extract simple signers from multi-signer
There is a quirk related to ordering: we store accounts in such an order that is expected by multi-signature verification script. This was done to speed up transaction/block signing which is done quite frequently in tests. This commit allows to provide accounts in any order and to extract a single signer from multi-signer based on this order. Signed-off-by: Evgeniy Stratonikov <evgeniy@nspcc.ru>
This commit is contained in:
parent
d4689db47e
commit
37ecf51d13
2 changed files with 66 additions and 8 deletions
|
@ -3,6 +3,7 @@ package neotest
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"sort"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
||||||
|
@ -35,11 +36,21 @@ type SingleSigner interface {
|
||||||
Account() *wallet.Account
|
Account() *wallet.Account
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MultiSigner is the interface for multisignature signing account.
|
||||||
|
type MultiSigner interface {
|
||||||
|
Signer
|
||||||
|
// Single returns simple-signature signer for n-th account in list.
|
||||||
|
Single(n int) SingleSigner
|
||||||
|
}
|
||||||
|
|
||||||
// signer represents simple-signature signer.
|
// signer represents simple-signature signer.
|
||||||
type signer wallet.Account
|
type signer wallet.Account
|
||||||
|
|
||||||
// multiSigner represents single multi-signature signer consisting of provided accounts.
|
// multiSigner represents single multi-signature signer consisting of provided accounts.
|
||||||
type multiSigner []*wallet.Account
|
type multiSigner struct {
|
||||||
|
accounts []*wallet.Account
|
||||||
|
m int
|
||||||
|
}
|
||||||
|
|
||||||
// NewSingleSigner returns multi-signature signer for the provided account.
|
// NewSingleSigner returns multi-signature signer for the provided account.
|
||||||
// It must contain exactly as many accounts as needed to sign the script.
|
// It must contain exactly as many accounts as needed to sign the script.
|
||||||
|
@ -78,7 +89,7 @@ func (s *signer) Account() *wallet.Account {
|
||||||
|
|
||||||
// NewMultiSigner returns multi-signature signer for the provided account.
|
// NewMultiSigner returns multi-signature signer for the provided account.
|
||||||
// It must contain at least as many accounts as needed to sign the script.
|
// It must contain at least as many accounts as needed to sign the script.
|
||||||
func NewMultiSigner(accs ...*wallet.Account) Signer {
|
func NewMultiSigner(accs ...*wallet.Account) MultiSigner {
|
||||||
if len(accs) == 0 {
|
if len(accs) == 0 {
|
||||||
panic("empty account list")
|
panic("empty account list")
|
||||||
}
|
}
|
||||||
|
@ -91,30 +102,35 @@ func NewMultiSigner(accs ...*wallet.Account) Signer {
|
||||||
panic(fmt.Sprintf("verification script requires %d signatures, "+
|
panic(fmt.Sprintf("verification script requires %d signatures, "+
|
||||||
"but only %d accounts were provided", m, len(accs)))
|
"but only %d accounts were provided", m, len(accs)))
|
||||||
}
|
}
|
||||||
|
sort.Slice(accs, func(i, j int) bool {
|
||||||
|
p1 := accs[i].PrivateKey().PublicKey()
|
||||||
|
p2 := accs[j].PrivateKey().PublicKey()
|
||||||
|
return p1.Cmp(p2) == -1
|
||||||
|
})
|
||||||
for _, acc := range accs {
|
for _, acc := range accs {
|
||||||
if !bytes.Equal(script, acc.Contract.Script) {
|
if !bytes.Equal(script, acc.Contract.Script) {
|
||||||
panic("all accounts must have equal verification script")
|
panic("all accounts must have equal verification script")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return multiSigner(accs[:m])
|
return multiSigner{accounts: accs, m: m}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ScriptHash implements Signer interface.
|
// ScriptHash implements Signer interface.
|
||||||
func (m multiSigner) ScriptHash() util.Uint160 {
|
func (m multiSigner) ScriptHash() util.Uint160 {
|
||||||
return m[0].Contract.ScriptHash()
|
return m.accounts[0].Contract.ScriptHash()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Script implements Signer interface.
|
// Script implements Signer interface.
|
||||||
func (m multiSigner) Script() []byte {
|
func (m multiSigner) Script() []byte {
|
||||||
return m[0].Contract.Script
|
return m.accounts[0].Contract.Script
|
||||||
}
|
}
|
||||||
|
|
||||||
// SignHashable implements Signer interface.
|
// SignHashable implements Signer interface.
|
||||||
func (m multiSigner) SignHashable(magic uint32, item hash.Hashable) []byte {
|
func (m multiSigner) SignHashable(magic uint32, item hash.Hashable) []byte {
|
||||||
var script []byte
|
var script []byte
|
||||||
for _, acc := range m {
|
for i := 0; i < m.m; i++ {
|
||||||
sign := acc.PrivateKey().SignHashable(magic, item)
|
sign := m.accounts[i].PrivateKey().SignHashable(magic, item)
|
||||||
script = append(script, byte(opcode.PUSHDATA1), 64)
|
script = append(script, byte(opcode.PUSHDATA1), 64)
|
||||||
script = append(script, sign...)
|
script = append(script, sign...)
|
||||||
}
|
}
|
||||||
|
@ -138,9 +154,19 @@ func (m multiSigner) SignTx(magic netmode.Magic, tx *transaction.Transaction) er
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Single implements MultiSigner interface.
|
||||||
|
func (m multiSigner) Single(n int) SingleSigner {
|
||||||
|
if len(m.accounts) <= n {
|
||||||
|
panic("invalid index")
|
||||||
|
}
|
||||||
|
return NewSingleSigner(wallet.NewAccountFromPrivateKey(m.accounts[n].PrivateKey()))
|
||||||
|
}
|
||||||
|
|
||||||
func checkMultiSigner(t *testing.T, s Signer) {
|
func checkMultiSigner(t *testing.T, s Signer) {
|
||||||
accs, ok := s.(multiSigner)
|
ms, ok := s.(multiSigner)
|
||||||
require.True(t, ok, "expected to be a multi-signer")
|
require.True(t, ok, "expected to be a multi-signer")
|
||||||
|
|
||||||
|
accs := ms.accounts
|
||||||
require.True(t, len(accs) > 0, "empty multi-signer")
|
require.True(t, len(accs) > 0, "empty multi-signer")
|
||||||
|
|
||||||
m := len(accs[0].Contract.Parameters)
|
m := len(accs[0].Contract.Parameters)
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
package neotest
|
package neotest
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"sort"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
@ -14,3 +17,32 @@ func TestSingleSigner(t *testing.T) {
|
||||||
s := NewSingleSigner(a)
|
s := NewSingleSigner(a)
|
||||||
require.Equal(t, s.ScriptHash(), s.Account().Contract.ScriptHash())
|
require.Equal(t, s.ScriptHash(), s.Account().Contract.ScriptHash())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestMultiSigner(t *testing.T) {
|
||||||
|
const size = 4
|
||||||
|
|
||||||
|
pubs := make(keys.PublicKeys, size)
|
||||||
|
accs := make([]*wallet.Account, size)
|
||||||
|
for i := range accs {
|
||||||
|
a, err := wallet.NewAccount()
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
accs[i] = a
|
||||||
|
pubs[i] = a.PrivateKey().PublicKey()
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Sort(pubs)
|
||||||
|
m := smartcontract.GetDefaultHonestNodeCount(size)
|
||||||
|
for i := range accs {
|
||||||
|
require.NoError(t, accs[i].ConvertMultisig(m, pubs))
|
||||||
|
}
|
||||||
|
|
||||||
|
s := NewMultiSigner(accs...)
|
||||||
|
for i := range pubs {
|
||||||
|
for j := range accs {
|
||||||
|
if pub := accs[j].PrivateKey().PublicKey(); pub.Equal(pubs[i]) {
|
||||||
|
require.Equal(t, pub, s.Single(i).Account().PrivateKey().PublicKey())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue