2018-02-21 04:57:52 +00:00
|
|
|
package sharding
|
|
|
|
|
|
|
|
import (
|
|
|
|
"math/big"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
2018-05-03 15:48:25 +00:00
|
|
|
"github.com/ethereum/go-ethereum/crypto/sha3"
|
|
|
|
"github.com/ethereum/go-ethereum/rlp"
|
2018-02-21 04:57:52 +00:00
|
|
|
)
|
|
|
|
|
2018-05-03 01:16:07 +00:00
|
|
|
// Collation base struct.
|
2018-02-21 04:57:52 +00:00
|
|
|
type Collation struct {
|
2018-05-11 02:00:54 +00:00
|
|
|
header *CollationHeader
|
|
|
|
// body represents the serialized blob of a collation's transactions.
|
2018-05-16 02:55:02 +00:00
|
|
|
// this is a read-only property.
|
2018-05-11 02:00:54 +00:00
|
|
|
body []byte
|
|
|
|
// transactions serves as a useful slice to store deserialized chunks from the
|
|
|
|
// collation's body. Every time this transactions slice is updated, the serialized
|
|
|
|
// body would need to be recalculated. This will be a useful property for proposers
|
|
|
|
// in our system.
|
2018-02-21 04:57:52 +00:00
|
|
|
transactions []*types.Transaction
|
|
|
|
}
|
|
|
|
|
2018-05-03 01:16:07 +00:00
|
|
|
// CollationHeader base struct.
|
2018-02-21 04:57:52 +00:00
|
|
|
type CollationHeader struct {
|
2018-05-11 02:00:54 +00:00
|
|
|
// RLP decoding only works on exported properties of structs. In this case, we want
|
|
|
|
// to keep collation properties as read-only and only accessible through getters.
|
|
|
|
// We can accomplish this through this nested data property.
|
2018-05-09 21:25:27 +00:00
|
|
|
data collationHeaderData
|
|
|
|
}
|
|
|
|
|
|
|
|
type collationHeaderData struct {
|
|
|
|
ShardID *big.Int // the shard ID of the shard.
|
|
|
|
ChunkRoot *common.Hash // the root of the chunk tree which identifies collation body.
|
2018-05-16 05:13:06 +00:00
|
|
|
Period *big.Int // the period number in which collation to be included.
|
2018-05-09 21:25:27 +00:00
|
|
|
ProposerAddress *common.Address // address of the collation proposer.
|
|
|
|
ProposerSignature []byte // the proposer's signature for calculating collation hash.
|
|
|
|
}
|
|
|
|
|
2018-05-14 13:53:20 +00:00
|
|
|
// NewCollation initializes a collation and leaves it up to clients to serialize, deserialize
|
|
|
|
// and provide the body and transactions upon creation.
|
|
|
|
func NewCollation(header *CollationHeader, body []byte, transactions []*types.Transaction) *Collation {
|
|
|
|
return &Collation{header, body, transactions}
|
|
|
|
}
|
|
|
|
|
2018-05-09 21:25:27 +00:00
|
|
|
// NewCollationHeader initializes a collation header struct.
|
|
|
|
func NewCollationHeader(shardID *big.Int, chunkRoot *common.Hash, period *big.Int, proposerAddress *common.Address, proposerSignature []byte) *CollationHeader {
|
|
|
|
data := collationHeaderData{
|
|
|
|
ShardID: shardID,
|
|
|
|
ChunkRoot: chunkRoot,
|
|
|
|
Period: period,
|
|
|
|
ProposerAddress: proposerAddress,
|
|
|
|
ProposerSignature: proposerSignature,
|
|
|
|
}
|
2018-05-09 21:31:46 +00:00
|
|
|
return &CollationHeader{data}
|
2018-02-21 04:57:52 +00:00
|
|
|
}
|
|
|
|
|
2018-05-16 16:59:00 +00:00
|
|
|
// Hash takes the keccak256 of the collation header's data contents.
|
2018-05-03 15:48:25 +00:00
|
|
|
func (h *CollationHeader) Hash() (hash common.Hash) {
|
|
|
|
hw := sha3.NewKeccak256()
|
2018-05-16 16:59:00 +00:00
|
|
|
rlp.Encode(hw, h.data)
|
2018-05-03 15:48:25 +00:00
|
|
|
hw.Sum(hash[:0])
|
|
|
|
return hash
|
|
|
|
}
|
|
|
|
|
2018-05-11 01:29:42 +00:00
|
|
|
// ShardID the collation corresponds to.
|
2018-05-09 21:25:27 +00:00
|
|
|
func (h *CollationHeader) ShardID() *big.Int { return h.data.ShardID }
|
2018-05-07 20:30:06 +00:00
|
|
|
|
|
|
|
// Period the collation corresponds to.
|
2018-05-09 21:25:27 +00:00
|
|
|
func (h *CollationHeader) Period() *big.Int { return h.data.Period }
|
2018-05-07 20:30:06 +00:00
|
|
|
|
|
|
|
// ChunkRoot of the serialized collation body.
|
2018-05-09 21:25:27 +00:00
|
|
|
func (h *CollationHeader) ChunkRoot() *common.Hash { return h.data.ChunkRoot }
|
|
|
|
|
|
|
|
// EncodeRLP gives an encoded representation of the collation header.
|
|
|
|
func (h *CollationHeader) EncodeRLP() ([]byte, error) {
|
2018-05-15 18:48:13 +00:00
|
|
|
return rlp.EncodeToBytes(&h.data)
|
2018-05-09 21:25:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeRLP uses an RLP Stream to populate the data field of a collation header.
|
|
|
|
func (h *CollationHeader) DecodeRLP(s *rlp.Stream) error {
|
2018-05-11 01:29:42 +00:00
|
|
|
return s.Decode(&h.data)
|
2018-05-09 21:25:27 +00:00
|
|
|
}
|
2018-05-07 20:30:06 +00:00
|
|
|
|
2018-05-03 01:16:07 +00:00
|
|
|
// Header returns the collation's header.
|
2018-05-02 15:37:44 +00:00
|
|
|
func (c *Collation) Header() *CollationHeader { return c.header }
|
|
|
|
|
2018-05-07 20:30:06 +00:00
|
|
|
// Body returns the collation's byte body.
|
|
|
|
func (c *Collation) Body() []byte { return c.body }
|
|
|
|
|
2018-05-03 01:16:07 +00:00
|
|
|
// Transactions returns an array of tx's in the collation.
|
2018-02-22 03:39:51 +00:00
|
|
|
func (c *Collation) Transactions() []*types.Transaction { return c.transactions }
|
|
|
|
|
2018-05-03 01:16:07 +00:00
|
|
|
// ProposerAddress is the coinbase addr of the creator for the collation.
|
2018-05-09 21:25:27 +00:00
|
|
|
func (c *Collation) ProposerAddress() *common.Address {
|
|
|
|
return c.header.data.ProposerAddress
|
|
|
|
}
|
2018-03-21 04:07:40 +00:00
|
|
|
|
2018-05-09 21:31:46 +00:00
|
|
|
// CalculateChunkRoot updates the collation header's chunk root based on the body.
|
|
|
|
func (c *Collation) CalculateChunkRoot() {
|
|
|
|
// TODO: this needs to be based on blob serialization.
|
2018-05-11 01:29:42 +00:00
|
|
|
// For proof of custody we need to split chunks (body) into chunk + salt and
|
|
|
|
// take the merkle root of that.
|
2018-05-09 21:25:27 +00:00
|
|
|
chunkRoot := common.BytesToHash(c.body)
|
|
|
|
c.header.data.ChunkRoot = &chunkRoot
|
|
|
|
}
|