2018-05-28 15:18:14 +00:00
|
|
|
package proposer
|
|
|
|
|
|
|
|
import (
|
2018-06-20 03:59:02 +00:00
|
|
|
"crypto/rand"
|
2018-05-28 15:18:14 +00:00
|
|
|
"math/big"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum"
|
|
|
|
"github.com/ethereum/go-ethereum/accounts"
|
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind"
|
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind/backends"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/core"
|
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
|
|
|
"github.com/ethereum/go-ethereum/sharding/contracts"
|
2018-06-12 23:03:20 +00:00
|
|
|
"github.com/ethereum/go-ethereum/sharding/params"
|
2018-05-28 15:18:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291")
|
|
|
|
addr = crypto.PubkeyToAddress(key.PublicKey)
|
|
|
|
accountBalance, _ = new(big.Int).SetString("1001000000000000000000", 10)
|
|
|
|
)
|
|
|
|
|
|
|
|
// Mock client for testing proposer.
|
|
|
|
type mockNode struct {
|
|
|
|
smc *contracts.SMC
|
|
|
|
t *testing.T
|
2018-06-07 06:03:54 +00:00
|
|
|
depositFlag bool
|
2018-06-01 23:04:42 +00:00
|
|
|
backend *backends.SimulatedBackend
|
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
func (m *mockNode) Account() *accounts.Account {
|
2018-05-28 15:18:14 +00:00
|
|
|
return &accounts.Account{Address: addr}
|
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
func (m *mockNode) SMCCaller() *contracts.SMCCaller {
|
|
|
|
return &m.smc.SMCCaller
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
func (m *mockNode) ChainReader() ethereum.ChainReader {
|
2018-05-28 15:18:14 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
func (m *mockNode) SMCTransactor() *contracts.SMCTransactor {
|
|
|
|
return &m.smc.SMCTransactor
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
func (m *mockNode) CreateTXOpts(value *big.Int) (*bind.TransactOpts, error) {
|
2018-05-28 15:18:14 +00:00
|
|
|
txOpts := transactOpts()
|
|
|
|
txOpts.Value = value
|
|
|
|
return txOpts, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNode) Sign(hash common.Hash) ([]byte, error) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2018-06-12 23:03:20 +00:00
|
|
|
func (m *mockNode) GetShardCount() (int64, error) {
|
|
|
|
return 100, nil
|
|
|
|
}
|
|
|
|
|
2018-05-28 15:18:14 +00:00
|
|
|
func transactOpts() *bind.TransactOpts {
|
|
|
|
return bind.NewKeyedTransactor(key)
|
|
|
|
}
|
|
|
|
|
2018-06-03 06:21:42 +00:00
|
|
|
func setup(t *testing.T) (*backends.SimulatedBackend, *contracts.SMC) {
|
2018-05-28 15:18:14 +00:00
|
|
|
backend := backends.NewSimulatedBackend(core.GenesisAlloc{addr: {Balance: accountBalance}})
|
2018-06-01 23:04:42 +00:00
|
|
|
_, _, smc, err := contracts.DeploySMC(transactOpts(), backend)
|
|
|
|
if err != nil {
|
2018-06-03 06:21:42 +00:00
|
|
|
t.Fatalf("Failed to deploy SMC contract: %v", err)
|
2018-06-01 23:04:42 +00:00
|
|
|
}
|
2018-05-28 15:18:14 +00:00
|
|
|
backend.Commit()
|
2018-06-03 06:21:42 +00:00
|
|
|
return backend, smc
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestCreateCollation(t *testing.T) {
|
2018-06-03 06:21:42 +00:00
|
|
|
backend, smc := setup(t)
|
2018-06-01 23:04:42 +00:00
|
|
|
node := &mockNode{smc: smc, t: t, backend: backend}
|
2018-05-28 15:18:14 +00:00
|
|
|
var txs []*types.Transaction
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
data := make([]byte, 1024)
|
|
|
|
rand.Read(data)
|
|
|
|
txs = append(txs, types.NewTransaction(0, common.HexToAddress("0x0"),
|
|
|
|
nil, 0, nil, data))
|
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err := createCollation(node, node.Account(), node, big.NewInt(0), big.NewInt(1), txs)
|
2018-05-28 15:18:14 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Create collation failed: %v", err)
|
|
|
|
}
|
|
|
|
|
2018-06-01 23:04:42 +00:00
|
|
|
// fast forward to 2nd period.
|
2018-06-13 17:37:23 +00:00
|
|
|
for i := 0; i < 2*int(params.DefaultConfig.PeriodLength); i++ {
|
2018-05-28 15:18:14 +00:00
|
|
|
backend.Commit()
|
|
|
|
}
|
|
|
|
|
2018-06-01 23:04:42 +00:00
|
|
|
// negative test case #1: create collation with shard > shardCount.
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err = createCollation(node, node.Account(), node, big.NewInt(101), big.NewInt(2), txs)
|
2018-05-28 15:18:14 +00:00
|
|
|
if err == nil {
|
2018-06-01 23:04:42 +00:00
|
|
|
t.Errorf("Create collation should have failed with invalid shard number")
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|
2018-06-01 23:04:42 +00:00
|
|
|
// negative test case #2, create collation with blob size > collationBodySizeLimit.
|
2018-05-28 15:18:14 +00:00
|
|
|
var badTxs []*types.Transaction
|
|
|
|
for i := 0; i <= 1024; i++ {
|
|
|
|
data := make([]byte, 1024)
|
|
|
|
rand.Read(data)
|
|
|
|
badTxs = append(badTxs, types.NewTransaction(0, common.HexToAddress("0x0"),
|
|
|
|
nil, 0, nil, data))
|
|
|
|
}
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err = createCollation(node, node.Account(), node, big.NewInt(0), big.NewInt(2), badTxs)
|
2018-05-28 15:18:14 +00:00
|
|
|
if err == nil {
|
2018-06-01 23:04:42 +00:00
|
|
|
t.Errorf("Create collation should have failed with Txs longer than collation body limit")
|
|
|
|
}
|
|
|
|
|
|
|
|
// normal test case #1 create collation with correct parameters.
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err = createCollation(node, node.Account(), node, big.NewInt(5), big.NewInt(5), txs)
|
2018-06-01 23:04:42 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Create collation failed: %v", err)
|
|
|
|
}
|
|
|
|
if collation.Header().Period().Cmp(big.NewInt(5)) != 0 {
|
2018-06-03 06:21:42 +00:00
|
|
|
t.Errorf("Incorrect collation period, want 5, got %v ", collation.Header().Period())
|
2018-06-01 23:04:42 +00:00
|
|
|
}
|
|
|
|
if collation.Header().ShardID().Cmp(big.NewInt(5)) != 0 {
|
|
|
|
t.Errorf("Incorrect shard id, want 5, got %v ", collation.Header().ShardID())
|
|
|
|
}
|
|
|
|
if *collation.ProposerAddress() != node.Account().Address {
|
|
|
|
t.Errorf("Incorrect proposer address, got %v", *collation.ProposerAddress())
|
|
|
|
}
|
|
|
|
if collation.Header().Sig() != nil {
|
|
|
|
t.Errorf("Proposer signaure can not be empty")
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddCollation(t *testing.T) {
|
2018-06-03 06:21:42 +00:00
|
|
|
backend, smc := setup(t)
|
2018-06-01 23:04:42 +00:00
|
|
|
node := &mockNode{smc: smc, t: t, backend: backend}
|
|
|
|
var txs []*types.Transaction
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
data := make([]byte, 1024)
|
|
|
|
rand.Read(data)
|
|
|
|
txs = append(txs, types.NewTransaction(0, common.HexToAddress("0x0"),
|
|
|
|
nil, 0, nil, data))
|
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err := createCollation(node, node.Account(), node, big.NewInt(0), big.NewInt(1), txs)
|
2018-06-01 23:04:42 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Create collation failed: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// fast forward to next period.
|
2018-06-13 17:37:23 +00:00
|
|
|
for i := 0; i < int(params.DefaultConfig.PeriodLength); i++ {
|
2018-06-01 23:04:42 +00:00
|
|
|
backend.Commit()
|
|
|
|
}
|
|
|
|
|
|
|
|
// normal test case #1 create collation with normal parameters.
|
2018-06-07 22:16:34 +00:00
|
|
|
err = addHeader(node, collation)
|
2018-06-01 23:04:42 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%v", err)
|
|
|
|
}
|
|
|
|
backend.Commit()
|
|
|
|
|
|
|
|
// verify collation was correctly added from SMC.
|
|
|
|
collationFromSMC, err := smc.CollationRecords(&bind.CallOpts{}, big.NewInt(0), big.NewInt(1))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Failed to get collation record")
|
|
|
|
}
|
|
|
|
if collationFromSMC.Proposer != node.Account().Address {
|
|
|
|
t.Errorf("Incorrect proposer address, got %v", *collation.ProposerAddress())
|
|
|
|
}
|
|
|
|
if common.BytesToHash(collationFromSMC.ChunkRoot[:]) != *collation.Header().ChunkRoot() {
|
|
|
|
t.Errorf("Incorrect chunk root, got %v", collationFromSMC.ChunkRoot)
|
|
|
|
}
|
|
|
|
|
|
|
|
// negative test case #1 create the same collation that just got added to SMC.
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err = createCollation(node, node.Account(), node, big.NewInt(0), big.NewInt(1), txs)
|
2018-06-01 23:04:42 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("Create collation should fail due to same collation in SMC")
|
|
|
|
}
|
|
|
|
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestCheckCollation(t *testing.T) {
|
2018-06-03 06:21:42 +00:00
|
|
|
backend, smc := setup(t)
|
2018-06-01 23:04:42 +00:00
|
|
|
node := &mockNode{smc: smc, t: t, backend: backend}
|
|
|
|
var txs []*types.Transaction
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
data := make([]byte, 1024)
|
|
|
|
rand.Read(data)
|
|
|
|
txs = append(txs, types.NewTransaction(0, common.HexToAddress("0x0"),
|
|
|
|
nil, 0, nil, data))
|
|
|
|
}
|
|
|
|
|
2018-06-17 17:39:42 +00:00
|
|
|
collation, err := createCollation(node, node.Account(), node, big.NewInt(0), big.NewInt(1), txs)
|
2018-06-01 23:04:42 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Create collation failed: %v", err)
|
|
|
|
}
|
|
|
|
|
2018-06-13 17:37:23 +00:00
|
|
|
for i := 0; i < int(params.DefaultConfig.PeriodLength); i++ {
|
2018-06-01 23:04:42 +00:00
|
|
|
backend.Commit()
|
|
|
|
}
|
|
|
|
|
2018-06-07 22:16:34 +00:00
|
|
|
err = addHeader(node, collation)
|
2018-06-01 23:04:42 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%v", err)
|
|
|
|
}
|
|
|
|
backend.Commit()
|
|
|
|
|
|
|
|
// normal test case 1: check if we can still add header for period 1, should return false.
|
2018-06-07 22:16:34 +00:00
|
|
|
a, err := checkHeaderAdded(node, big.NewInt(0), big.NewInt(1))
|
2018-06-01 23:04:42 +00:00
|
|
|
if err != nil {
|
2018-06-03 06:21:42 +00:00
|
|
|
t.Errorf("Can not check header submitted: %v", err)
|
2018-06-01 23:04:42 +00:00
|
|
|
}
|
|
|
|
if a {
|
2018-06-03 06:21:42 +00:00
|
|
|
t.Errorf("Check header submitted shouldn't return: %v", a)
|
2018-06-01 23:04:42 +00:00
|
|
|
}
|
|
|
|
// normal test case 2: check if we can add header for period 2, should return true.
|
2018-06-07 22:16:34 +00:00
|
|
|
a, err = checkHeaderAdded(node, big.NewInt(0), big.NewInt(2))
|
2018-06-01 23:04:42 +00:00
|
|
|
if !a {
|
2018-06-03 06:21:42 +00:00
|
|
|
t.Errorf("Check header submitted shouldn't return: %v", a)
|
2018-06-01 23:04:42 +00:00
|
|
|
}
|
2018-05-28 15:18:14 +00:00
|
|
|
}
|