|
|
|
package consensus
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
|
|
|
|
"github.com/dedis/kyber"
|
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
consensus_proto "github.com/harmony-one/harmony/api/consensus"
|
|
|
|
"github.com/harmony-one/harmony/api/proto"
|
|
|
|
"github.com/harmony-one/harmony/crypto"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Constructs the announce message
|
|
|
|
func (consensus *Consensus) constructAnnounceMessage() []byte {
|
|
|
|
message := consensus_proto.Message{}
|
|
|
|
message.Type = consensus_proto.MessageType_ANNOUNCE
|
|
|
|
|
|
|
|
// 4 byte consensus id
|
|
|
|
message.ConsensusId = consensus.consensusID
|
|
|
|
|
|
|
|
// 32 byte block hash
|
|
|
|
message.BlockHash = consensus.blockHash[:]
|
|
|
|
|
|
|
|
// 4 byte sender id
|
|
|
|
message.SenderId = uint32(consensus.nodeID)
|
|
|
|
|
|
|
|
// n byte of block header
|
|
|
|
message.Payload = consensus.block
|
|
|
|
|
|
|
|
marshaledMessage, err := message.XXX_Marshal([]byte{}, true)
|
|
|
|
if err != nil {
|
|
|
|
consensus.Log.Debug("Failed to marshal Announce message", "error", err)
|
|
|
|
}
|
|
|
|
// 64 byte of signature on previous data
|
|
|
|
signature := consensus.signMessage(marshaledMessage)
|
|
|
|
message.Signature = signature
|
|
|
|
|
|
|
|
marshaledMessage, err = message.XXX_Marshal([]byte{}, true)
|
|
|
|
if err != nil {
|
|
|
|
consensus.Log.Debug("Failed to marshal Announce message", "error", err)
|
|
|
|
}
|
|
|
|
consensus.Log.Info("New Announce", "NodeID", consensus.nodeID, "bitmap", consensus.bitmap)
|
|
|
|
return proto.ConstructConsensusMessage(marshaledMessage)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the challenge message, returning challenge message in bytes, challenge scalar and aggregated commmitment point.
|
|
|
|
func (consensus *Consensus) constructChallengeMessage(msgType consensus_proto.MessageType) ([]byte, kyber.Scalar, kyber.Point) {
|
|
|
|
message := consensus_proto.Message{}
|
|
|
|
message.Type = msgType
|
|
|
|
|
|
|
|
// 4 byte consensus id
|
|
|
|
message.ConsensusId = consensus.consensusID
|
|
|
|
|
|
|
|
// 32 byte block hash
|
|
|
|
message.BlockHash = consensus.blockHash[:]
|
|
|
|
|
|
|
|
// 4 byte sender id
|
|
|
|
message.SenderId = uint32(consensus.nodeID)
|
|
|
|
|
|
|
|
//// Payload
|
|
|
|
buffer := bytes.NewBuffer([]byte{})
|
|
|
|
|
|
|
|
commitmentsMap := consensus.commitments // msgType == Challenge
|
|
|
|
bitmap := consensus.bitmap
|
|
|
|
if msgType == consensus_proto.MessageType_FINAL_CHALLENGE {
|
|
|
|
commitmentsMap = consensus.finalCommitments
|
|
|
|
bitmap = consensus.finalBitmap
|
|
|
|
}
|
|
|
|
|
|
|
|
// 33 byte aggregated commit
|
|
|
|
commitments := make([]kyber.Point, 0)
|
|
|
|
for _, val := range *commitmentsMap {
|
|
|
|
commitments = append(commitments, val)
|
|
|
|
}
|
|
|
|
aggCommitment, aggCommitmentBytes := getAggregatedCommit(commitments)
|
|
|
|
buffer.Write(aggCommitmentBytes)
|
|
|
|
|
|
|
|
// 33 byte aggregated key
|
|
|
|
buffer.Write(getAggregatedKey(bitmap))
|
|
|
|
|
|
|
|
// 32 byte challenge
|
|
|
|
challengeScalar := getChallenge(aggCommitment, bitmap.AggregatePublic, message.BlockHash)
|
|
|
|
bytes, err := challengeScalar.MarshalBinary()
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Failed to serialize challenge")
|
|
|
|
}
|
|
|
|
buffer.Write(bytes)
|
|
|
|
|
|
|
|
message.Payload = buffer.Bytes()
|
|
|
|
//// END Payload
|
|
|
|
|
|
|
|
marshaledMessage, err := message.XXX_Marshal([]byte{}, true)
|
|
|
|
if err != nil {
|
|
|
|
consensus.Log.Debug("Failed to marshal Challenge message", "error", err)
|
|
|
|
}
|
|
|
|
// 64 byte of signature on previous data
|
|
|
|
signature := consensus.signMessage(marshaledMessage)
|
|
|
|
message.Signature = signature
|
|
|
|
|
|
|
|
marshaledMessage, err = message.XXX_Marshal([]byte{}, true)
|
|
|
|
if err != nil {
|
|
|
|
consensus.Log.Debug("Failed to marshal Challenge message", "error", err)
|
|
|
|
}
|
|
|
|
consensus.Log.Info("New Challenge", "NodeID", consensus.nodeID, "bitmap", consensus.bitmap)
|
|
|
|
return proto.ConstructConsensusMessage(marshaledMessage), challengeScalar, aggCommitment
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the collective signature message
|
|
|
|
func (consensus *Consensus) constructCollectiveSigMessage(collectiveSig [64]byte, bitmap []byte) []byte {
|
|
|
|
message := consensus_proto.Message{}
|
|
|
|
message.Type = consensus_proto.MessageType_COLLECTIVE_SIG
|
|
|
|
|
|
|
|
// 4 byte consensus id
|
|
|
|
message.ConsensusId = consensus.consensusID
|
|
|
|
|
|
|
|
// 32 byte block hash
|
|
|
|
message.BlockHash = consensus.blockHash[:]
|
|
|
|
|
|
|
|
// 4 byte sender id
|
|
|
|
message.SenderId = uint32(consensus.nodeID)
|
|
|
|
|
|
|
|
//// Payload
|
|
|
|
buffer := bytes.NewBuffer([]byte{})
|
|
|
|
|
|
|
|
// 64 byte collective signature
|
|
|
|
buffer.Write(collectiveSig[:])
|
|
|
|
|
|
|
|
// N byte bitmap
|
|
|
|
buffer.Write(bitmap)
|
|
|
|
|
|
|
|
message.Payload = buffer.Bytes()
|
|
|
|
//// END Payload
|
|
|
|
|
|
|
|
marshaledMessage, err := message.XXX_Marshal([]byte{}, true)
|
|
|
|
if err != nil {
|
|
|
|
consensus.Log.Debug("Failed to marshal Challenge message", "error", err)
|
|
|
|
}
|
|
|
|
// 64 byte of signature on previous data
|
|
|
|
signature := consensus.signMessage(marshaledMessage)
|
|
|
|
message.Signature = signature
|
|
|
|
|
|
|
|
marshaledMessage, err = message.XXX_Marshal([]byte{}, true)
|
|
|
|
if err != nil {
|
|
|
|
consensus.Log.Debug("Failed to marshal Challenge message", "error", err)
|
|
|
|
}
|
|
|
|
consensus.Log.Info("New CollectiveSig", "NodeID", consensus.nodeID, "bitmap", consensus.bitmap)
|
|
|
|
return proto.ConstructConsensusMessage(marshaledMessage)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getAggregatedCommit(commitments []kyber.Point) (commitment kyber.Point, bytes []byte) {
|
|
|
|
aggCommitment := crypto.AggregateCommitmentsOnly(crypto.Ed25519Curve, commitments)
|
|
|
|
bytes, err := aggCommitment.MarshalBinary()
|
|
|
|
if err != nil {
|
|
|
|
panic("Failed to deserialize the aggregated commitment")
|
|
|
|
}
|
|
|
|
return aggCommitment, append(bytes[:], byte(0))
|
|
|
|
}
|
|
|
|
|
|
|
|
func getAggregatedKey(bitmap *crypto.Mask) []byte {
|
|
|
|
bytes, err := bitmap.AggregatePublic.MarshalBinary()
|
|
|
|
if err != nil {
|
|
|
|
panic("Failed to deserialize the aggregated key")
|
|
|
|
}
|
|
|
|
return append(bytes[:], byte(0))
|
|
|
|
}
|
|
|
|
|
|
|
|
func getChallenge(aggCommitment, aggKey kyber.Point, message []byte) kyber.Scalar {
|
|
|
|
challenge, err := crypto.Challenge(crypto.Ed25519Curve, aggCommitment, aggKey, message)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Failed to generate challenge")
|
|
|
|
}
|
|
|
|
return challenge
|
|
|
|
}
|