2018-06-21 03:03:02 +00:00
|
|
|
package syncer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math/big"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2018-07-20 21:31:26 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/client/mainchain"
|
|
|
|
"github.com/prysmaticlabs/prysm/client/types"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/sharding/v1"
|
2018-07-24 15:21:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/p2p"
|
2018-07-21 17:51:18 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2018-06-21 03:03:02 +00:00
|
|
|
)
|
|
|
|
|
2018-07-21 17:51:18 +00:00
|
|
|
var log = logrus.WithField("prefix", "syncer")
|
|
|
|
|
2018-06-21 03:03:02 +00:00
|
|
|
// RespondCollationBody is called by a node responding to another node's request
|
|
|
|
// for a collation body given a (shardID, chunkRoot, period, proposerAddress) tuple.
|
|
|
|
// The proposer will fetch the corresponding data from persistent storage (shardDB) by
|
|
|
|
// constructing a collation header from the input and calculating its hash.
|
2018-07-17 18:39:04 +00:00
|
|
|
func RespondCollationBody(req p2p.Message, collationFetcher types.CollationFetcher) (*pb.CollationBodyResponse, error) {
|
2018-06-21 03:03:02 +00:00
|
|
|
// Type assertion helps us catch incorrect data requests.
|
2018-07-17 18:39:04 +00:00
|
|
|
msg, ok := req.Data.(*pb.CollationBodyRequest)
|
2018-06-21 03:03:02 +00:00
|
|
|
if !ok {
|
2018-07-17 18:39:04 +00:00
|
|
|
log.Debugf("Request data type: %T", req.Data)
|
|
|
|
return nil, fmt.Errorf("received incorrect data request type. Data: %+v", msg)
|
2018-06-21 03:03:02 +00:00
|
|
|
}
|
|
|
|
|
2018-07-17 18:39:04 +00:00
|
|
|
shardID := new(big.Int).SetUint64(msg.ShardId)
|
|
|
|
chunkRoot := common.BytesToHash(msg.ChunkRoot)
|
|
|
|
period := new(big.Int).SetUint64(msg.Period)
|
|
|
|
proposer := common.BytesToAddress(msg.ProposerAddress)
|
|
|
|
var sig [32]byte
|
|
|
|
if len(msg.Signature) >= 32 {
|
|
|
|
copy(sig[:], msg.Signature[0:32])
|
|
|
|
}
|
|
|
|
header := types.NewCollationHeader(shardID, &chunkRoot, period, &proposer, sig)
|
2018-06-21 03:03:02 +00:00
|
|
|
|
|
|
|
// Fetch the collation by its header hash from the shardChainDB.
|
|
|
|
headerHash := header.Hash()
|
|
|
|
collation, err := collationFetcher.CollationByHeaderHash(&headerHash)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not fetch collation: %v", err)
|
|
|
|
}
|
2018-07-17 18:39:04 +00:00
|
|
|
if collation == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2018-06-21 03:03:02 +00:00
|
|
|
|
2018-07-17 18:39:04 +00:00
|
|
|
return &pb.CollationBodyResponse{HeaderHash: headerHash.Bytes(), Body: collation.Body()}, nil
|
2018-06-21 03:03:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RequestCollationBody fetches a collation header record submitted to the SMC for
|
|
|
|
// a shardID, period pair and constructs a p2p collationBodyRequest that will
|
|
|
|
// then be relayed to the appropriate proposer that submitted the collation header.
|
2018-07-26 01:57:19 +00:00
|
|
|
// In production, this will be done within a attester service.
|
2018-07-17 18:39:04 +00:00
|
|
|
func RequestCollationBody(fetcher mainchain.RecordFetcher, shardID *big.Int, period *big.Int) (*pb.CollationBodyRequest, error) {
|
2018-06-21 03:03:02 +00:00
|
|
|
|
|
|
|
record, err := fetcher.CollationRecords(&bind.CallOpts{}, shardID, period)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not fetch collation record from SMC: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
sum := 0
|
|
|
|
for _, val := range record.ChunkRoot {
|
|
|
|
sum += int(val)
|
|
|
|
}
|
|
|
|
|
|
|
|
if sum == 0 {
|
2018-07-17 18:39:04 +00:00
|
|
|
log.Debugf("No collation exists for shard %d and period %d", shardID, period)
|
2018-06-21 03:03:02 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Converts from fixed size [32]byte to []byte slice.
|
|
|
|
chunkRoot := common.BytesToHash(record.ChunkRoot[:])
|
|
|
|
|
2018-07-17 18:39:04 +00:00
|
|
|
return &pb.CollationBodyRequest{
|
|
|
|
ChunkRoot: chunkRoot.Bytes(),
|
|
|
|
ShardId: shardID.Uint64(),
|
|
|
|
Period: period.Uint64(),
|
|
|
|
ProposerAddress: record.Proposer.Bytes(),
|
2018-06-21 03:03:02 +00:00
|
|
|
}, nil
|
|
|
|
}
|