The core protocol of WoopChain
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
woop/node/node_cross_shard.go

135 lines
4.8 KiB

package node
import (
"github.com/ethereum/go-ethereum/rlp"
proto_node "github.com/woop-chain/woop/api/proto/node"
"github.com/woop-chain/woop/block"
"github.com/woop-chain/woop/consensus"
"github.com/woop-chain/woop/core"
"github.com/woop-chain/woop/core/types"
nodeconfig "github.com/woop-chain/woop/internal/configs/node"
"github.com/woop-chain/woop/internal/utils"
"github.com/woop-chain/woop/p2p"
"github.com/woop-chain/woop/shard"
)
// BroadcastCXReceipts broadcasts cross shard receipts to correspoding
// destination shards
func BroadcastCXReceipts(newBlock *types.Block, consensus *consensus.Consensus) {
commitSigAndBitmap := newBlock.GetCurrentCommitSig()
//#### Read payload data from committed msg
if len(commitSigAndBitmap) <= 96 {
utils.Logger().Debug().Int("commitSigAndBitmapLen", len(commitSigAndBitmap)).Msg("[BroadcastCXReceipts] commitSigAndBitmap Not Enough Length")
return
}
commitSig := make([]byte, 96)
commitBitmap := make([]byte, len(commitSigAndBitmap)-96)
offset := 0
copy(commitSig[:], commitSigAndBitmap[offset:offset+96])
offset += 96
copy(commitBitmap[:], commitSigAndBitmap[offset:])
//#### END Read payload data from committed msg
epoch := newBlock.Header().Epoch()
shardingConfig := shard.Schedule.InstanceForEpoch(epoch)
shardNum := int(shardingConfig.NumShards())
myShardID := consensus.ShardID
utils.Logger().Info().Int("shardNum", shardNum).Uint32("myShardID", myShardID).Uint64("blockNum", newBlock.NumberU64()).Msg("[BroadcastCXReceipts]")
for i := 0; i < shardNum; i++ {
if i == int(myShardID) {
continue
}
BroadcastCXReceiptsWithShardID(newBlock.Header(), commitSig, commitBitmap, uint32(i), consensus)
}
}
// BroadcastCXReceiptsWithShardID broadcasts cross shard receipts to given ToShardID
func BroadcastCXReceiptsWithShardID(block *block.Header, commitSig []byte, commitBitmap []byte, toShardID uint32, consensus *consensus.Consensus) {
myShardID := consensus.ShardID
utils.Logger().Debug().
Uint32("toShardID", toShardID).
Uint32("myShardID", myShardID).
Uint64("blockNum", block.NumberU64()).
Msg("[BroadcastCXReceiptsWithShardID]")
cxReceipts, err := consensus.Blockchain().ReadCXReceipts(toShardID, block.NumberU64(), block.Hash())
if err != nil || len(cxReceipts) == 0 {
utils.Logger().Debug().Uint32("ToShardID", toShardID).
Int("numCXReceipts", len(cxReceipts)).
Msg("[CXMerkleProof] No receipts found for the destination shard")
return
}
merkleProof, err := consensus.Blockchain().CXMerkleProof(toShardID, block)
if err != nil {
utils.Logger().Warn().
Uint32("ToShardID", toShardID).
Msg("[BroadcastCXReceiptsWithShardID] Unable to get merkleProof")
return
}
cxReceiptsProof := &types.CXReceiptsProof{
Receipts: cxReceipts,
MerkleProof: merkleProof,
Header: block,
CommitSig: commitSig,
CommitBitmap: commitBitmap,
}
groupID := nodeconfig.NewGroupIDByShardID(nodeconfig.ShardID(toShardID))
utils.Logger().Info().Uint32("ToShardID", toShardID).
Str("GroupID", string(groupID)).
Interface("cxp", cxReceiptsProof).
Msg("[BroadcastCXReceiptsWithShardID] ReadCXReceipts and MerkleProof ready. Sending CX receipts...")
// TODO ek – limit concurrency
go consensus.GetHost().SendMessageToGroups([]nodeconfig.GroupID{groupID},
p2p.ConstructMessage(proto_node.ConstructCXReceiptsProof(cxReceiptsProof)),
)
}
// BroadcastMissingCXReceipts broadcasts missing cross shard receipts per request
func BroadcastMissingCXReceipts(c *consensus.Consensus) {
var (
sendNextTime = make([]core.CxEntry, 0)
cxPool = c.Registry().GetCxPool()
blockchain = c.Blockchain()
)
it := cxPool.Pool().Iterator()
for entry := range it.C {
cxEntry := entry.(core.CxEntry)
toShardID := cxEntry.ToShardID
blk := blockchain.GetBlockByHash(cxEntry.BlockHash)
if blk == nil {
continue
}
blockNum := blk.NumberU64()
nextHeader := blockchain.GetHeaderByNumber(blockNum + 1)
if nextHeader == nil {
sendNextTime = append(sendNextTime, cxEntry)
continue
}
sig := nextHeader.LastCommitSignature()
bitmap := nextHeader.LastCommitBitmap()
BroadcastCXReceiptsWithShardID(blk.Header(), sig[:], bitmap, toShardID, c)
}
cxPool.Clear()
// this should not happen or maybe happen for impatient user
for _, entry := range sendNextTime {
cxPool.Add(entry)
}
}
// ProcessReceiptMessage store the receipts and merkle proof in local data store
func (node *Node) ProcessReceiptMessage(msgPayload []byte) {
cxp := types.CXReceiptsProof{}
if err := rlp.DecodeBytes(msgPayload, &cxp); err != nil {
utils.Logger().Error().Err(err).
Msg("[ProcessReceiptMessage] Unable to Decode message Payload")
return
}
utils.Logger().Debug().Interface("cxp", cxp).
Msg("[ProcessReceiptMessage] Add CXReceiptsProof to pending Receipts")
// TODO: integrate with txpool
node.AddPendingReceipts(&cxp)
}