From c8cc91eeeee4732f2c41f105e326f86938ace010 Mon Sep 17 00:00:00 2001 From: Roman Khimov Date: Fri, 14 Aug 2020 16:22:15 +0300 Subject: [PATCH 1/2] network: request blocks when there is a ping with bigger than ours height Turns out, C# node no longer broadcasts an Inv when it's creating a block, instead it sends a ping and if we're not paying attention to the height specified there we're technically missing a new block. Of course we'll get it later after ping timer expiration and regular ping/pong sequence, but that's delaying it for no good reason. --- pkg/network/helper_test.go | 5 +++++ pkg/network/peer.go | 3 +++ pkg/network/server.go | 10 ++++++++++ pkg/network/tcp_peer.go | 8 ++++++++ 4 files changed, 26 insertions(+) diff --git a/pkg/network/helper_test.go b/pkg/network/helper_test.go index e7cc7a5f0..eb6649b60 100644 --- a/pkg/network/helper_test.go +++ b/pkg/network/helper_test.go @@ -286,6 +286,11 @@ func (p *localPeer) SendPing(m *Message) error { _ = p.EnqueueMessage(m) return nil } +func (p *localPeer) HandlePing(ping *payload.Ping) error { + p.lastBlockIndex = ping.LastBlockIndex + return nil +} + func (p *localPeer) HandlePong(pong *payload.Ping) error { p.lastBlockIndex = pong.LastBlockIndex p.pingSent-- diff --git a/pkg/network/peer.go b/pkg/network/peer.go index 873347b4a..da7bd850f 100644 --- a/pkg/network/peer.go +++ b/pkg/network/peer.go @@ -67,6 +67,9 @@ type Peer interface { HandleVersion(*payload.Version) error HandleVersionAck() error + // HandlePing checks ping contents against Peer's state and updates it. + HandlePing(ping *payload.Ping) error + // HandlePong checks pong contents against Peer's state and updates it. HandlePong(pong *payload.Ping) error } diff --git a/pkg/network/server.go b/pkg/network/server.go index 3a926f19c..d3f22c9fe 100644 --- a/pkg/network/server.go +++ b/pkg/network/server.go @@ -453,6 +453,16 @@ func (s *Server) handleBlockCmd(p Peer, block *block.Block) error { // handlePing processes ping request. func (s *Server) handlePing(p Peer, ping *payload.Ping) error { + err := p.HandlePing(ping) + if err != nil { + return err + } + if s.chain.BlockHeight() < ping.LastBlockIndex { + err = s.requestBlocks(p) + if err != nil { + return err + } + } return p.EnqueueP2PMessage(NewMessage(CMDPong, payload.NewPing(s.chain.BlockHeight(), s.id))) } diff --git a/pkg/network/tcp_peer.go b/pkg/network/tcp_peer.go index aec25796e..64d884feb 100644 --- a/pkg/network/tcp_peer.go +++ b/pkg/network/tcp_peer.go @@ -431,6 +431,14 @@ func (p *TCPPeer) SendPing(msg *Message) error { return p.EnqueueMessage(msg) } +// HandlePing handles a ping message received from the peer. +func (p *TCPPeer) HandlePing(ping *payload.Ping) error { + p.lock.Lock() + defer p.lock.Unlock() + p.lastBlockIndex = ping.LastBlockIndex + return nil +} + // HandlePong handles a pong message received from the peer and does appropriate // accounting of outstanding pings and timeouts. func (p *TCPPeer) HandlePong(pong *payload.Ping) error { From 8d19f0e6f56836bfcdb84857c895be5d1b9205fc Mon Sep 17 00:00:00 2001 From: Roman Khimov Date: Fri, 14 Aug 2020 16:25:13 +0300 Subject: [PATCH 2/2] network: don't request block we already have GetBlockByIndex handler starts sending blocks right from the start index and if that index is s.chain.BlockHeight() then we're requesting and receiving a block we already have. --- pkg/network/server.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/network/server.go b/pkg/network/server.go index d3f22c9fe..2920ef1c5 100644 --- a/pkg/network/server.go +++ b/pkg/network/server.go @@ -704,7 +704,7 @@ func (s *Server) handleGetAddrCmd(p Peer) error { // to sync up in blocks. A maximum of maxBlockBatch will // send at once. func (s *Server) requestBlocks(p Peer) error { - payload := payload.NewGetBlockByIndex(s.chain.BlockHeight(), -1) + payload := payload.NewGetBlockByIndex(s.chain.BlockHeight()+1, -1) return p.EnqueueP2PMessage(NewMessage(CMDGetBlockByIndex, payload)) }