2023-09-03 23:47:09 +00:00
|
|
|
|
package master
|
|
|
|
|
|
|
|
|
|
import (
|
2024-03-01 07:12:31 +00:00
|
|
|
|
"bytes"
|
2024-03-21 07:14:45 +00:00
|
|
|
|
"context"
|
2024-03-01 07:12:31 +00:00
|
|
|
|
"encoding/binary"
|
2023-09-03 23:47:09 +00:00
|
|
|
|
"strings"
|
2024-03-21 07:14:45 +00:00
|
|
|
|
"time"
|
2023-09-03 23:47:09 +00:00
|
|
|
|
|
|
|
|
|
"github.com/iden3/go-iden3-crypto/poseidon"
|
2024-05-25 05:22:50 +00:00
|
|
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
2024-03-01 07:12:31 +00:00
|
|
|
|
"github.com/mr-tron/base58"
|
2023-09-03 23:47:09 +00:00
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
"go.uber.org/zap"
|
|
|
|
|
"google.golang.org/protobuf/proto"
|
|
|
|
|
"google.golang.org/protobuf/types/known/anypb"
|
|
|
|
|
"source.quilibrium.com/quilibrium/monorepo/go-libp2p-blossomsub/pb"
|
|
|
|
|
"source.quilibrium.com/quilibrium/monorepo/node/consensus"
|
|
|
|
|
"source.quilibrium.com/quilibrium/monorepo/node/protobufs"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func (e *MasterClockConsensusEngine) handleMessage(message *pb.Message) error {
|
|
|
|
|
e.logger.Debug(
|
|
|
|
|
"received message",
|
|
|
|
|
zap.Binary("data", message.Data),
|
|
|
|
|
zap.Binary("from", message.From),
|
|
|
|
|
zap.Binary("signature", message.Signature),
|
|
|
|
|
)
|
|
|
|
|
msg := &protobufs.Message{}
|
|
|
|
|
if err := proto.Unmarshal(message.Data, msg); err != nil {
|
|
|
|
|
return errors.Wrap(err, "handle message")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
any := &anypb.Any{}
|
|
|
|
|
if err := proto.Unmarshal(msg.Payload, any); err != nil {
|
|
|
|
|
return errors.Wrap(err, "handle message")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch any.TypeUrl {
|
|
|
|
|
case protobufs.ClockFrameType:
|
|
|
|
|
if err := e.handleClockFrameData(
|
|
|
|
|
message.From,
|
|
|
|
|
any,
|
|
|
|
|
); err != nil {
|
|
|
|
|
return errors.Wrap(err, "handle message")
|
|
|
|
|
}
|
2024-03-13 01:28:48 +00:00
|
|
|
|
return nil
|
2024-03-01 07:12:31 +00:00
|
|
|
|
case protobufs.SelfTestReportType:
|
|
|
|
|
if err := e.handleSelfTestReport(
|
|
|
|
|
message.From,
|
|
|
|
|
any,
|
|
|
|
|
); err != nil {
|
|
|
|
|
return errors.Wrap(err, "handle message")
|
|
|
|
|
}
|
2024-03-13 01:28:48 +00:00
|
|
|
|
return nil
|
2023-09-03 23:47:09 +00:00
|
|
|
|
}
|
2023-09-25 02:43:35 +00:00
|
|
|
|
|
2024-03-13 01:28:48 +00:00
|
|
|
|
return errors.Wrap(errors.New("invalid message"), "handle message")
|
2023-09-03 23:47:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (e *MasterClockConsensusEngine) handleClockFrameData(
|
|
|
|
|
peerID []byte,
|
|
|
|
|
any *anypb.Any,
|
|
|
|
|
) error {
|
|
|
|
|
frame := &protobufs.ClockFrame{}
|
|
|
|
|
if err := any.UnmarshalTo(frame); err != nil {
|
|
|
|
|
return errors.Wrap(err, "handle clock frame data")
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-05 05:11:01 +00:00
|
|
|
|
head, err := e.masterTimeReel.Head()
|
|
|
|
|
if err != nil {
|
|
|
|
|
panic(err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if frame.FrameNumber < head.FrameNumber {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-03 23:47:09 +00:00
|
|
|
|
if e.difficulty != frame.Difficulty {
|
2023-09-28 07:59:27 +00:00
|
|
|
|
e.logger.Debug(
|
2023-09-03 23:47:09 +00:00
|
|
|
|
"frame difficulty mismatched",
|
|
|
|
|
zap.Uint32("difficulty", frame.Difficulty),
|
|
|
|
|
)
|
2023-09-25 02:43:35 +00:00
|
|
|
|
return errors.Wrap(
|
|
|
|
|
errors.New("frame difficulty"),
|
|
|
|
|
"handle clock frame data",
|
|
|
|
|
)
|
2023-09-03 23:47:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 07:59:27 +00:00
|
|
|
|
e.logger.Debug(
|
2023-09-03 23:47:09 +00:00
|
|
|
|
"got clock frame",
|
|
|
|
|
zap.Binary("sender", peerID),
|
|
|
|
|
zap.Binary("filter", frame.Filter),
|
|
|
|
|
zap.Uint64("frame_number", frame.FrameNumber),
|
|
|
|
|
zap.Int("proof_count", len(frame.AggregateProofs)),
|
|
|
|
|
)
|
|
|
|
|
|
2024-03-13 01:28:48 +00:00
|
|
|
|
go func() {
|
|
|
|
|
select {
|
|
|
|
|
case e.frameValidationCh <- frame:
|
|
|
|
|
default:
|
|
|
|
|
e.logger.Debug(
|
|
|
|
|
"dropped frame due to overwhelmed queue",
|
|
|
|
|
zap.Binary("sender", peerID),
|
|
|
|
|
zap.Binary("filter", frame.Filter),
|
|
|
|
|
zap.Uint64("frame_number", frame.FrameNumber),
|
|
|
|
|
zap.Int("proof_count", len(frame.AggregateProofs)),
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}()
|
2023-09-03 23:47:09 +00:00
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-01 07:12:31 +00:00
|
|
|
|
func (e *MasterClockConsensusEngine) handleSelfTestReport(
|
|
|
|
|
peerID []byte,
|
|
|
|
|
any *anypb.Any,
|
|
|
|
|
) error {
|
|
|
|
|
report := &protobufs.SelfTestReport{}
|
|
|
|
|
if err := any.UnmarshalTo(report); err != nil {
|
|
|
|
|
return errors.Wrap(err, "handle self test report")
|
|
|
|
|
}
|
|
|
|
|
|
v1.4.18-patch-2 (#230)
* feat: IPC for wesolowski
* update self peer info
* remove digests and signatures
* add new binaries and digests
* Signatory #13 added
* Signatory #4 added (#231)
* added sig.6 files (#232)
* Signatory #9 added (#233)
* Added signatories #1, #2, #3, #5, #8, #12, #14, #15, #16, #17
* remove binaries, release ready
---------
Co-authored-by: 0xOzgur <29779769+0xOzgur@users.noreply.github.com>
Co-authored-by: Demipoet <161999657+demipoet@users.noreply.github.com>
Co-authored-by: Freekers <1370857+Freekers@users.noreply.github.com>
2024-05-29 17:51:47 +00:00
|
|
|
|
if bytes.Equal(peerID, e.pubSub.GetPeerID()) {
|
|
|
|
|
info := e.peerInfoManager.GetPeerInfo(peerID)
|
|
|
|
|
info.LastSeen = time.Now().UnixMilli()
|
|
|
|
|
info.DifficultyMetric = report.DifficultyMetric
|
|
|
|
|
info.MasterHeadFrame = report.MasterHeadFrame
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-25 05:22:50 +00:00
|
|
|
|
// minimum proof size is one timestamp, one vdf proof, must match one fewer
|
|
|
|
|
// than core count
|
|
|
|
|
if len(report.Proof) < 516+8 ||
|
|
|
|
|
((len(report.Proof)-8)/516) != int(report.Cores-1) {
|
|
|
|
|
e.logger.Warn(
|
|
|
|
|
"received invalid proof from peer",
|
|
|
|
|
zap.String("peer_id", peer.ID(peerID).String()),
|
2024-05-27 05:10:15 +00:00
|
|
|
|
zap.Int("proof_size", len(report.Proof)),
|
|
|
|
|
zap.Uint32("cores", report.Cores),
|
2024-05-25 05:22:50 +00:00
|
|
|
|
)
|
|
|
|
|
e.pubSub.SetPeerScore(peerID, -1000)
|
|
|
|
|
return errors.Wrap(errors.New("invalid report"), "handle self test report")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
e.logger.Debug(
|
|
|
|
|
"received proof from peer",
|
|
|
|
|
zap.String("peer_id", peer.ID(peerID).String()),
|
|
|
|
|
)
|
|
|
|
|
|
2024-03-21 07:14:45 +00:00
|
|
|
|
info := e.peerInfoManager.GetPeerInfo(peerID)
|
|
|
|
|
if info != nil {
|
2024-05-25 05:22:50 +00:00
|
|
|
|
if (time.Now().UnixMilli() - info.LastSeen) < (270 * 1000) {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-27 05:10:15 +00:00
|
|
|
|
info.DifficultyMetric = report.DifficultyMetric
|
2024-03-21 07:14:45 +00:00
|
|
|
|
info.MasterHeadFrame = report.MasterHeadFrame
|
2024-05-25 05:22:50 +00:00
|
|
|
|
|
2024-03-23 20:26:57 +00:00
|
|
|
|
if info.Bandwidth <= 1048576 {
|
|
|
|
|
go func() {
|
|
|
|
|
ctx, cancel := context.WithTimeout(context.TODO(), 5*time.Minute)
|
|
|
|
|
defer cancel()
|
|
|
|
|
ch := e.pubSub.GetMultiaddrOfPeerStream(ctx, peerID)
|
|
|
|
|
select {
|
|
|
|
|
case <-ch:
|
|
|
|
|
go func() {
|
|
|
|
|
e.bandwidthTestCh <- peerID
|
|
|
|
|
}()
|
|
|
|
|
case <-ctx.Done():
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
}
|
2024-05-25 05:22:50 +00:00
|
|
|
|
|
|
|
|
|
proof := report.Proof
|
|
|
|
|
timestamp := binary.BigEndian.Uint64(proof[:8])
|
|
|
|
|
proof = proof[8:]
|
|
|
|
|
|
|
|
|
|
challenge := binary.BigEndian.AppendUint64([]byte{}, report.MasterHeadFrame)
|
|
|
|
|
challenge = append(challenge, peerID...)
|
|
|
|
|
|
|
|
|
|
proofs := make([][]byte, (len(report.Proof)-8)/516)
|
|
|
|
|
for i := 0; i < len(proofs); i++ {
|
|
|
|
|
proofs[i] = proof[i*516 : (i+1)*516]
|
|
|
|
|
}
|
2024-05-27 05:10:15 +00:00
|
|
|
|
go func() {
|
|
|
|
|
e.verifyTestCh <- verifyChallenge{
|
|
|
|
|
peerID: peerID,
|
|
|
|
|
challenge: challenge,
|
|
|
|
|
timestamp: int64(timestamp),
|
|
|
|
|
difficultyMetric: report.DifficultyMetric,
|
|
|
|
|
proofs: proofs,
|
|
|
|
|
}
|
|
|
|
|
}()
|
2024-05-25 05:22:50 +00:00
|
|
|
|
|
2024-03-01 07:12:31 +00:00
|
|
|
|
return nil
|
|
|
|
|
}
|
2024-03-21 07:14:45 +00:00
|
|
|
|
|
|
|
|
|
e.addPeerManifestReport(peerID, report)
|
2024-03-01 07:12:31 +00:00
|
|
|
|
|
|
|
|
|
memory := binary.BigEndian.Uint64(report.Memory)
|
2024-03-08 05:05:04 +00:00
|
|
|
|
e.logger.Debug(
|
2024-03-01 07:12:31 +00:00
|
|
|
|
"received self test report",
|
|
|
|
|
zap.String("peer_id", base58.Encode(peerID)),
|
|
|
|
|
zap.Uint32("difficulty", report.Difficulty),
|
|
|
|
|
zap.Int64("difficulty_metric", report.DifficultyMetric),
|
|
|
|
|
zap.Int64("commit_16_metric", report.Commit_16Metric),
|
|
|
|
|
zap.Int64("commit_128_metric", report.Commit_128Metric),
|
|
|
|
|
zap.Int64("commit_1024_metric", report.Commit_1024Metric),
|
|
|
|
|
zap.Int64("commit_65536_metric", report.Commit_65536Metric),
|
|
|
|
|
zap.Int64("proof_16_metric", report.Proof_16Metric),
|
|
|
|
|
zap.Int64("proof_128_metric", report.Proof_128Metric),
|
|
|
|
|
zap.Int64("proof_1024_metric", report.Proof_1024Metric),
|
|
|
|
|
zap.Int64("proof_65536_metric", report.Proof_65536Metric),
|
|
|
|
|
zap.Uint32("cores", report.Cores),
|
|
|
|
|
zap.Uint64("memory", memory),
|
|
|
|
|
zap.Uint64("storage", binary.BigEndian.Uint64(report.Storage)),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if report.Cores < 3 || memory < 16000000000 {
|
2024-03-12 07:45:20 +00:00
|
|
|
|
e.logger.Debug(
|
2024-03-01 07:12:31 +00:00
|
|
|
|
"peer reported invalid configuration",
|
|
|
|
|
zap.String("peer_id", base58.Encode(peerID)),
|
|
|
|
|
zap.Uint32("difficulty", report.Difficulty),
|
|
|
|
|
zap.Int64("difficulty_metric", report.DifficultyMetric),
|
|
|
|
|
zap.Int64("commit_16_metric", report.Commit_16Metric),
|
|
|
|
|
zap.Int64("commit_128_metric", report.Commit_128Metric),
|
|
|
|
|
zap.Int64("commit_1024_metric", report.Commit_1024Metric),
|
|
|
|
|
zap.Int64("commit_65536_metric", report.Commit_65536Metric),
|
|
|
|
|
zap.Int64("proof_16_metric", report.Proof_16Metric),
|
|
|
|
|
zap.Int64("proof_128_metric", report.Proof_128Metric),
|
|
|
|
|
zap.Int64("proof_1024_metric", report.Proof_1024Metric),
|
|
|
|
|
zap.Int64("proof_65536_metric", report.Proof_65536Metric),
|
|
|
|
|
zap.Uint32("cores", report.Cores),
|
|
|
|
|
zap.Uint64("memory", memory),
|
|
|
|
|
zap.Uint64("storage", binary.BigEndian.Uint64(report.Storage)),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// tag: dusk – nuke this peer for now
|
|
|
|
|
e.pubSub.SetPeerScore(peerID, -1000)
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-13 01:28:48 +00:00
|
|
|
|
go func() {
|
2024-03-21 07:14:45 +00:00
|
|
|
|
ctx, cancel := context.WithTimeout(context.TODO(), 5*time.Minute)
|
|
|
|
|
defer cancel()
|
|
|
|
|
ch := e.pubSub.GetMultiaddrOfPeerStream(ctx, peerID)
|
|
|
|
|
select {
|
|
|
|
|
case <-ch:
|
|
|
|
|
go func() {
|
|
|
|
|
e.bandwidthTestCh <- peerID
|
|
|
|
|
}()
|
|
|
|
|
case <-ctx.Done():
|
|
|
|
|
}
|
2024-03-13 01:28:48 +00:00
|
|
|
|
}()
|
2024-03-01 07:12:31 +00:00
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-27 05:10:15 +00:00
|
|
|
|
// This does not publish any longer, frames strictly are picked up from sync
|
2023-09-03 23:47:09 +00:00
|
|
|
|
func (e *MasterClockConsensusEngine) publishProof(
|
|
|
|
|
frame *protobufs.ClockFrame,
|
|
|
|
|
) error {
|
2024-02-13 07:04:56 +00:00
|
|
|
|
e.logger.Debug(
|
|
|
|
|
"publishing frame",
|
|
|
|
|
zap.Uint64("frame_number", frame.FrameNumber),
|
|
|
|
|
)
|
|
|
|
|
|
2024-03-17 21:14:37 +00:00
|
|
|
|
e.masterTimeReel.Insert(frame, false)
|
2023-09-03 23:47:09 +00:00
|
|
|
|
|
2024-02-13 07:04:56 +00:00
|
|
|
|
e.state = consensus.EngineStateCollecting
|
|
|
|
|
|
2023-09-03 23:47:09 +00:00
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (e *MasterClockConsensusEngine) publishMessage(
|
|
|
|
|
filter []byte,
|
|
|
|
|
message proto.Message,
|
|
|
|
|
) error {
|
|
|
|
|
any := &anypb.Any{}
|
|
|
|
|
if err := any.MarshalFrom(message); err != nil {
|
|
|
|
|
return errors.Wrap(err, "publish message")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// annoying protobuf any hack
|
|
|
|
|
any.TypeUrl = strings.Replace(
|
|
|
|
|
any.TypeUrl,
|
|
|
|
|
"type.googleapis.com",
|
|
|
|
|
"types.quilibrium.com",
|
|
|
|
|
1,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
payload, err := proto.Marshal(any)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return errors.Wrap(err, "publish message")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
h, err := poseidon.HashBytes(payload)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return errors.Wrap(err, "publish message")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
msg := &protobufs.Message{
|
|
|
|
|
Hash: h.Bytes(),
|
|
|
|
|
Address: e.filter,
|
|
|
|
|
Payload: payload,
|
|
|
|
|
}
|
|
|
|
|
data, err := proto.Marshal(msg)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return errors.Wrap(err, "publish message")
|
|
|
|
|
}
|
|
|
|
|
return e.pubSub.PublishToBitmask(filter, data)
|
|
|
|
|
}
|