2018-05-10 16:33:18 +00:00
|
|
|
package proposer
|
2018-05-11 21:40:23 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math/big"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind"
|
2018-05-25 16:15:13 +00:00
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
2018-05-11 21:40:23 +00:00
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
"github.com/ethereum/go-ethereum/sharding"
|
2018-05-25 16:15:13 +00:00
|
|
|
"github.com/ethereum/go-ethereum/sharding/node"
|
2018-05-11 21:40:23 +00:00
|
|
|
)
|
|
|
|
|
2018-05-25 16:15:13 +00:00
|
|
|
// createCollation creates collation base struct with header
|
|
|
|
// and body. Header consists of shardID, ChunkRoot, period,
|
|
|
|
// proposer addr and signatures. Body contains serialized blob
|
|
|
|
// of a collations transactions.
|
|
|
|
func createCollation(n node.Node, shardId *big.Int, period *big.Int, txs []*types.Transaction) (*sharding.Collation, error) {
|
2018-05-28 15:06:32 +00:00
|
|
|
// shardId has to be within range
|
|
|
|
if shardId.Cmp(big.NewInt(0)) < 0 || shardId.Cmp(big.NewInt(sharding.ShardCount)) > 0 {
|
|
|
|
return nil, fmt.Errorf("can't create collation for shard %v. Must be between 0 and %v", shardId, sharding.ShardCount)
|
|
|
|
}
|
2018-05-25 16:15:13 +00:00
|
|
|
|
2018-05-28 15:06:32 +00:00
|
|
|
// period can't be less than current period.
|
|
|
|
//ctx := context.Background()
|
|
|
|
//b, _ := n.ChainReader().BlockByNumber(ctx, nil)
|
|
|
|
b := big.NewInt(5)
|
|
|
|
currentPeriod := new(big.Int).Div(b, big.NewInt(sharding.PeriodLength))
|
|
|
|
if period.Cmp(currentPeriod) < 0 {
|
|
|
|
return nil, fmt.Errorf("can't create collation with period %v less than current period %v", period, currentPeriod)
|
|
|
|
}
|
|
|
|
|
|
|
|
// serialized tx to blob for collation body.
|
2018-05-25 16:15:13 +00:00
|
|
|
blobs, err := sharding.SerializeTxToBlob(txs)
|
|
|
|
if err != nil {
|
2018-05-28 15:06:32 +00:00
|
|
|
return nil, fmt.Errorf("can't create collation, serialization to blob failed: %v", err)
|
2018-05-25 16:15:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// construct the header, leave chunkRoot and signature fields empty, to be filled later.
|
2018-05-28 15:06:32 +00:00
|
|
|
addr := n.Account().Address
|
2018-05-25 16:15:13 +00:00
|
|
|
header := sharding.NewCollationHeader(shardId, nil, period, &addr, nil)
|
|
|
|
|
|
|
|
// construct the body with header, blobs(serialized txs) and txs.
|
|
|
|
collation := sharding.NewCollation(header, blobs, txs)
|
|
|
|
collation.CalculateChunkRoot()
|
|
|
|
sig, err := n.Sign(collation.Header().Hash())
|
|
|
|
if err != nil {
|
2018-05-28 15:06:32 +00:00
|
|
|
return nil, fmt.Errorf("can't create collation, sign collationHeader failed: %v", err)
|
2018-05-25 16:15:13 +00:00
|
|
|
}
|
|
|
|
|
2018-05-28 15:06:32 +00:00
|
|
|
// add proposer signature to collation header.
|
2018-05-25 16:15:13 +00:00
|
|
|
collation.Header().AddSig(sig)
|
|
|
|
|
|
|
|
return collation, nil
|
|
|
|
}
|
|
|
|
|
2018-05-11 21:40:23 +00:00
|
|
|
// addHeader adds the collation header to the main chain by sending
|
|
|
|
// an addHeader transaction to the sharding manager contract.
|
|
|
|
// There can only exist one header per period per shard, it's proposer's
|
|
|
|
// responsibility to check if a header has been added.
|
2018-05-25 16:15:13 +00:00
|
|
|
func addHeader(n node.Node, collation sharding.Collation) error {
|
2018-05-11 21:40:23 +00:00
|
|
|
log.Info("Adding header to SMC")
|
|
|
|
|
2018-05-25 16:15:13 +00:00
|
|
|
txOps, err := n.CreateTXOpts(big.NewInt(0))
|
2018-05-11 21:40:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to initiate add header transaction: %v", err)
|
|
|
|
}
|
|
|
|
|
2018-05-21 02:28:37 +00:00
|
|
|
var churnkRoot [32]byte
|
|
|
|
copy(churnkRoot[:], collation.Header().ChunkRoot().String())
|
|
|
|
|
2018-05-25 16:15:13 +00:00
|
|
|
_, err = n.SMCTransactor().AddHeader(txOps, collation.Header().ShardID(), collation.Header().Period(), churnkRoot)
|
2018-05-11 21:40:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to add header to SMC: %v", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkHeaderAvailability checks if a collation header has already
|
|
|
|
// added to the main chain. There can only be one header per shard
|
|
|
|
// per period, proposer should check if anyone else has added the header.
|
|
|
|
// checkHeaderAvailability returns true if it's available, false if it's unavailable.
|
2018-05-25 16:15:13 +00:00
|
|
|
func checkHeaderAvailability(n node.Node, shardId big.Int, period big.Int) (bool, error) {
|
2018-05-11 21:40:23 +00:00
|
|
|
log.Info("Checking header in shard: %d, period: %d", shardId, period)
|
|
|
|
|
|
|
|
// Get the period of the last header.
|
2018-05-25 16:15:13 +00:00
|
|
|
lastPeriod, err := n.SMCCaller().LastSubmittedCollation(&bind.CallOpts{}, &shardId)
|
2018-05-11 21:40:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, fmt.Errorf("unable to get the period of last submitted collation: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// True if current period is greater than last added period.
|
|
|
|
return period.Cmp(lastPeriod) > 0, nil
|
|
|
|
}
|