|
|
|
package node
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math/big"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/harmony-one/bls/ffi/go/bls"
|
|
|
|
"github.com/harmony-one/harmony/drand"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
|
|
|
"github.com/ethereum/go-ethereum/params"
|
|
|
|
proto_discovery "github.com/harmony-one/harmony/api/proto/discovery"
|
|
|
|
"github.com/harmony-one/harmony/consensus"
|
|
|
|
"github.com/harmony-one/harmony/core/types"
|
|
|
|
"github.com/harmony-one/harmony/crypto/pki"
|
|
|
|
"github.com/harmony-one/harmony/internal/utils"
|
|
|
|
"github.com/harmony-one/harmony/p2p"
|
|
|
|
"github.com/harmony-one/harmony/p2p/p2pimpl"
|
|
|
|
"golang.org/x/crypto/sha3"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestNewNode(t *testing.T) {
|
|
|
|
_, pubKey := utils.GenKey("1", "2")
|
|
|
|
leader := p2p.Peer{IP: "127.0.0.1", Port: "8882", PubKey: pubKey}
|
|
|
|
validator := p2p.Peer{IP: "127.0.0.1", Port: "8885"}
|
|
|
|
priKey, _, _ := utils.GenKeyP2P("127.0.0.1", "9902")
|
|
|
|
host, err := p2pimpl.NewHost(&leader, priKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("newhost failure: %v", err)
|
|
|
|
}
|
|
|
|
consensus := consensus.New(host, "0", []p2p.Peer{leader, validator}, leader)
|
|
|
|
node := New(host, consensus, nil)
|
|
|
|
if node.Consensus == nil {
|
|
|
|
t.Error("Consensus is not initialized for the node")
|
|
|
|
}
|
|
|
|
|
|
|
|
if node.blockchain == nil {
|
|
|
|
t.Error("Blockchain is not initialized for the node")
|
|
|
|
}
|
|
|
|
|
|
|
|
if node.blockchain.CurrentBlock() == nil {
|
|
|
|
t.Error("Genesis block is not initialized for the node")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetSyncingPeers(t *testing.T) {
|
|
|
|
_, pubKey := utils.GenKey("1", "2")
|
|
|
|
leader := p2p.Peer{IP: "127.0.0.1", Port: "8882", PubKey: pubKey}
|
|
|
|
validator := p2p.Peer{IP: "127.0.0.1", Port: "8885"}
|
|
|
|
priKey, _, _ := utils.GenKeyP2P("127.0.0.1", "9902")
|
|
|
|
host, err := p2pimpl.NewHost(&leader, priKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("newhost failure: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
consensus := consensus.New(host, "0", []p2p.Peer{leader, validator}, leader)
|
|
|
|
|
|
|
|
node := New(host, consensus, nil)
|
|
|
|
peer := p2p.Peer{IP: "127.0.0.1", Port: "8000"}
|
|
|
|
peer2 := p2p.Peer{IP: "127.0.0.1", Port: "8001"}
|
|
|
|
node.Neighbors.Store("minh", peer)
|
|
|
|
node.Neighbors.Store("mark", peer2)
|
|
|
|
res := node.GetSyncingPeers()
|
|
|
|
if len(res) == 0 || !(res[0].IP == peer.IP || res[0].IP == peer2.IP) {
|
|
|
|
t.Error("GetSyncingPeers should return list of {peer, peer2}")
|
|
|
|
}
|
|
|
|
if len(res) == 0 || (res[0].Port != "5000" && res[0].Port != "5001") {
|
|
|
|
t.Errorf("Syncing ports should be 5000, got %v", res[0].Port)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddPeers(t *testing.T) {
|
|
|
|
pubKey1 := pki.GetBLSPrivateKeyFromInt(333).GetPublicKey()
|
|
|
|
pubKey2 := pki.GetBLSPrivateKeyFromInt(444).GetPublicKey()
|
|
|
|
|
|
|
|
peers1 := []*p2p.Peer{
|
|
|
|
&p2p.Peer{
|
|
|
|
IP: "127.0.0.1",
|
|
|
|
Port: "8888",
|
|
|
|
PubKey: pubKey1,
|
|
|
|
ValidatorID: 1,
|
|
|
|
},
|
|
|
|
&p2p.Peer{
|
|
|
|
IP: "127.0.0.1",
|
|
|
|
Port: "9999",
|
|
|
|
PubKey: pubKey2,
|
|
|
|
ValidatorID: 2,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
_, pubKey := utils.GenKey("1", "2")
|
|
|
|
leader := p2p.Peer{IP: "127.0.0.1", Port: "8982", PubKey: pubKey}
|
|
|
|
validator := p2p.Peer{IP: "127.0.0.1", Port: "8985"}
|
|
|
|
priKey, _, _ := utils.GenKeyP2P("127.0.0.1", "9902")
|
|
|
|
host, err := p2pimpl.NewHost(&leader, priKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("newhost failure: %v", err)
|
|
|
|
}
|
|
|
|
consensus := consensus.New(host, "0", []p2p.Peer{leader, validator}, leader)
|
|
|
|
dRand := drand.New(host, "0", []p2p.Peer{leader, validator}, leader, nil)
|
|
|
|
|
|
|
|
node := New(host, consensus, nil)
|
|
|
|
node.DRand = dRand
|
|
|
|
r1 := node.AddPeers(peers1)
|
|
|
|
e1 := 2
|
|
|
|
if r1 != e1 {
|
|
|
|
t.Errorf("Add %v peers, expectd %v", r1, e1)
|
|
|
|
}
|
|
|
|
r2 := node.AddPeers(peers1)
|
|
|
|
e2 := 0
|
|
|
|
if r2 != e2 {
|
|
|
|
t.Errorf("Add %v peers, expectd %v", r2, e2)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func sendPingMessage(node *Node, leader p2p.Peer) {
|
|
|
|
pubKey1 := pki.GetBLSPrivateKeyFromInt(333).GetPublicKey()
|
|
|
|
|
|
|
|
p1 := p2p.Peer{
|
|
|
|
IP: "127.0.0.1",
|
|
|
|
Port: "9999",
|
|
|
|
PubKey: pubKey1,
|
|
|
|
}
|
|
|
|
|
|
|
|
ping1 := proto_discovery.NewPingMessage(p1)
|
|
|
|
buf1 := ping1.ConstructPingMessage()
|
|
|
|
|
|
|
|
fmt.Println("waiting for 5 seconds ...")
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
|
|
|
|
node.SendMessage(leader, buf1)
|
|
|
|
fmt.Println("sent ping message ...")
|
|
|
|
}
|
|
|
|
|
|
|
|
func sendPongMessage(node *Node, leader p2p.Peer) {
|
|
|
|
pubKey1 := pki.GetBLSPrivateKeyFromInt(333).GetPublicKey()
|
|
|
|
pubKey2 := pki.GetBLSPrivateKeyFromInt(444).GetPublicKey()
|
|
|
|
p1 := p2p.Peer{
|
|
|
|
IP: "127.0.0.1",
|
|
|
|
Port: "9998",
|
|
|
|
PubKey: pubKey1,
|
|
|
|
}
|
|
|
|
p2 := p2p.Peer{
|
|
|
|
IP: "127.0.0.1",
|
|
|
|
Port: "9999",
|
|
|
|
PubKey: pubKey2,
|
|
|
|
}
|
|
|
|
|
|
|
|
pubKeys := []*bls.PublicKey{pubKey1, pubKey2}
|
|
|
|
leaderPubKey := pki.GetBLSPrivateKeyFromInt(888).GetPublicKey()
|
|
|
|
|
|
|
|
pong1 := proto_discovery.NewPongMessage([]p2p.Peer{p1, p2}, pubKeys, leaderPubKey)
|
|
|
|
buf1 := pong1.ConstructPongMessage()
|
|
|
|
|
|
|
|
fmt.Println("waiting for 10 seconds ...")
|
|
|
|
time.Sleep(10 * time.Second)
|
|
|
|
|
|
|
|
node.SendMessage(leader, buf1)
|
|
|
|
fmt.Println("sent pong message ...")
|
|
|
|
}
|
|
|
|
|
|
|
|
func exitServer() {
|
|
|
|
fmt.Println("wait 5 seconds to terminate the process ...")
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
|
|
|
|
os.Exit(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPingPongHandler(t *testing.T) {
|
|
|
|
_, pubKey := utils.GenKey("127.0.0.1", "8881")
|
|
|
|
leader := p2p.Peer{IP: "127.0.0.1", Port: "8881", PubKey: pubKey}
|
|
|
|
// validator := p2p.Peer{IP: "127.0.0.1", Port: "9991"}
|
|
|
|
priKey, _, _ := utils.GenKeyP2P("127.0.0.1", "9902")
|
|
|
|
host, err := p2pimpl.NewHost(&leader, priKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("newhost failure: %v", err)
|
|
|
|
}
|
|
|
|
consensus := consensus.New(host, "0", []p2p.Peer{leader}, leader)
|
|
|
|
node := New(host, consensus, nil)
|
|
|
|
//go sendPingMessage(leader)
|
|
|
|
go sendPongMessage(node, leader)
|
|
|
|
go exitServer()
|
|
|
|
node.StartServer()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUpdateStakingDeposit(t *testing.T) {
|
|
|
|
_, pubKey := utils.GenKey("1", "2")
|
|
|
|
leader := p2p.Peer{IP: "127.0.0.1", Port: "8882", PubKey: pubKey}
|
|
|
|
validator := p2p.Peer{IP: "127.0.0.1", Port: "8885"}
|
|
|
|
priKey, _, _ := utils.GenKeyP2P("127.0.0.1", "9902")
|
|
|
|
host, err := p2pimpl.NewHost(&leader, priKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("newhost failure: %v", err)
|
|
|
|
}
|
|
|
|
consensus := consensus.New(host, "0", []p2p.Peer{leader, validator}, leader)
|
|
|
|
|
|
|
|
node := New(host, consensus, nil)
|
|
|
|
node.CurrentStakes = make(map[common.Address]int64)
|
|
|
|
|
|
|
|
DepositContractPriKey, _ := crypto.GenerateKey() //DepositContractPriKey is pk for contract
|
|
|
|
DepositContractAddress := crypto.PubkeyToAddress(DepositContractPriKey.PublicKey) //DepositContractAddress is the address for the contract
|
|
|
|
node.StakingContractAddress = DepositContractAddress
|
|
|
|
node.AccountKey, _ = crypto.GenerateKey()
|
|
|
|
Address := crypto.PubkeyToAddress(node.AccountKey.PublicKey)
|
|
|
|
callingFunction := "0xd0e30db0"
|
|
|
|
amount := new(big.Int)
|
|
|
|
amount.SetString("10", 10)
|
|
|
|
dataEnc := common.FromHex(callingFunction) //Deposit Does not take a argument, stake is transferred via amount.
|
|
|
|
tx1, err := types.SignTx(types.NewTransaction(0, DepositContractAddress, node.Consensus.ShardID, amount, params.TxGasContractCreation*10, nil, dataEnc), types.HomesteadSigner{}, node.AccountKey)
|
|
|
|
|
|
|
|
var txs []*types.Transaction
|
|
|
|
txs = append(txs, tx1)
|
|
|
|
header := &types.Header{Extra: []byte("hello")}
|
|
|
|
block := types.NewBlock(header, txs, nil)
|
|
|
|
node.UpdateStakingList(block)
|
|
|
|
if len(node.CurrentStakes) == 0 {
|
|
|
|
t.Error("New node's stake was not added")
|
|
|
|
}
|
|
|
|
value, ok := node.CurrentStakes[Address]
|
|
|
|
if !ok {
|
|
|
|
t.Error("The correct address was not added")
|
|
|
|
}
|
|
|
|
if value != 10 {
|
|
|
|
t.Error("The correct stake value was not added")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUpdateStakingWithdrawal(t *testing.T) {
|
|
|
|
_, pubKey := utils.GenKey("1", "2")
|
|
|
|
leader := p2p.Peer{IP: "127.0.0.1", Port: "8882", PubKey: pubKey}
|
|
|
|
validator := p2p.Peer{IP: "127.0.0.1", Port: "8885"}
|
|
|
|
priKey, _, _ := utils.GenKeyP2P("127.0.0.1", "9902")
|
|
|
|
host, err := p2pimpl.NewHost(&leader, priKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("newhost failure: %v", err)
|
|
|
|
}
|
|
|
|
consensus := consensus.New(host, "0", []p2p.Peer{leader, validator}, leader)
|
|
|
|
|
|
|
|
node := New(host, consensus, nil)
|
|
|
|
node.CurrentStakes = make(map[common.Address]int64)
|
|
|
|
|
|
|
|
DepositContractPriKey, _ := crypto.GenerateKey() //DepositContractPriKey is pk for contract
|
|
|
|
DepositContractAddress := crypto.PubkeyToAddress(DepositContractPriKey.PublicKey) //DepositContractAddress is the address for the contract
|
|
|
|
node.StakingContractAddress = DepositContractAddress
|
|
|
|
node.AccountKey, _ = crypto.GenerateKey()
|
|
|
|
Address := crypto.PubkeyToAddress(node.AccountKey.PublicKey)
|
|
|
|
initialStake := int64(1010)
|
|
|
|
node.CurrentStakes[Address] = initialStake //initial stake
|
|
|
|
|
|
|
|
withdrawFnSignature := []byte("withdraw(uint256)")
|
|
|
|
hash := sha3.NewLegacyKeccak256()
|
|
|
|
hash.Write(withdrawFnSignature)
|
|
|
|
methodID := hash.Sum(nil)[:4]
|
|
|
|
|
|
|
|
amount := "10"
|
|
|
|
stakeToWithdraw := new(big.Int)
|
|
|
|
stakeToWithdraw.SetString(amount, 10)
|
|
|
|
paddedAmount := common.LeftPadBytes(stakeToWithdraw.Bytes(), 32)
|
|
|
|
|
|
|
|
remainingStakeShouldBe := initialStake - stakeToWithdraw.Int64()
|
|
|
|
|
|
|
|
var dataEnc []byte
|
|
|
|
dataEnc = append(dataEnc, methodID...)
|
|
|
|
dataEnc = append(dataEnc, paddedAmount...)
|
|
|
|
tx, err := types.SignTx(types.NewTransaction(0, DepositContractAddress, node.Consensus.ShardID, big.NewInt(0), params.TxGasContractCreation*10, nil, dataEnc), types.HomesteadSigner{}, node.AccountKey)
|
|
|
|
|
|
|
|
var txs []*types.Transaction
|
|
|
|
txs = append(txs, tx)
|
|
|
|
header := &types.Header{Extra: []byte("hello")}
|
|
|
|
block := types.NewBlock(header, txs, nil)
|
|
|
|
node.UpdateStakingList(block)
|
|
|
|
currentStake, ok := node.CurrentStakes[Address]
|
|
|
|
if !ok {
|
|
|
|
t.Error("The correct address was not present")
|
|
|
|
}
|
|
|
|
if currentStake != remainingStakeShouldBe {
|
|
|
|
t.Error("The correct stake value was not subtracted")
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|