Browse Source

Bring back NodeInfo NetAddress form the dead (#3545)

A prior change to address accidental DNS lookups introduced the
SocketAddr on peer, which was then used to add it to the addressbook.
Which in turn swallowed the self reported port of the peer, which is
important on a reconnect. This change revives the NetAddress on NodeInfo
which the Peer carries, but now returns an error to avoid nil
dereferencing another issue observed in the past. Additionally we could
potentially address #3532, yet the original problem statemenf of that
issue stands.

As a drive-by optimisation `MarkAsGood` now takes only a `p2p.ID` which
makes it interface a bit stricter and leaner.
pull/3547/head
Alexander Simmerl 5 years ago
committed by GitHub
parent
commit
b5b3b85697
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 31 additions and 28 deletions
  1. +7
    -13
      p2p/node_info.go
  2. +3
    -3
      p2p/pex/addrbook.go
  3. +4
    -4
      p2p/pex/addrbook_test.go
  4. +12
    -3
      p2p/pex/pex_reactor.go
  5. +1
    -1
      p2p/pex/pex_reactor_test.go
  6. +2
    -2
      p2p/switch.go
  7. +1
    -1
      p2p/switch_test.go
  8. +1
    -1
      p2p/test_util.go

+ 7
- 13
p2p/node_info.go View File

@ -24,9 +24,14 @@ func MaxNodeInfoSize() int {
// and determines if we're compatible. // and determines if we're compatible.
type NodeInfo interface { type NodeInfo interface {
ID() ID ID() ID
nodeInfoAddress
nodeInfoTransport nodeInfoTransport
} }
type nodeInfoAddress interface {
NetAddress() (*NetAddress, error)
}
// nodeInfoTransport validates a nodeInfo and checks // nodeInfoTransport validates a nodeInfo and checks
// our compatibility with it. It's for use in the handshake. // our compatibility with it. It's for use in the handshake.
type nodeInfoTransport interface { type nodeInfoTransport interface {
@ -209,20 +214,9 @@ OUTER_LOOP:
// it includes the authenticated peer ID and the self-reported // it includes the authenticated peer ID and the self-reported
// ListenAddr. Note that the ListenAddr is not authenticated and // ListenAddr. Note that the ListenAddr is not authenticated and
// may not match that address actually dialed if its an outbound peer. // may not match that address actually dialed if its an outbound peer.
func (info DefaultNodeInfo) NetAddress() *NetAddress {
func (info DefaultNodeInfo) NetAddress() (*NetAddress, error) {
idAddr := IDAddressString(info.ID(), info.ListenAddr) idAddr := IDAddressString(info.ID(), info.ListenAddr)
netAddr, err := NewNetAddressString(idAddr)
if err != nil {
switch err.(type) {
case ErrNetAddressLookup:
// XXX If the peer provided a host name and the lookup fails here
// we're out of luck.
// TODO: use a NetAddress in DefaultNodeInfo
default:
panic(err) // everything should be well formed by now
}
}
return netAddr
return NewNetAddressString(idAddr)
} }
//----------------------------------------------------------- //-----------------------------------------------------------


+ 3
- 3
p2p/pex/addrbook.go View File

@ -55,7 +55,7 @@ type AddrBook interface {
PickAddress(biasTowardsNewAddrs int) *p2p.NetAddress PickAddress(biasTowardsNewAddrs int) *p2p.NetAddress
// Mark address // Mark address
MarkGood(*p2p.NetAddress)
MarkGood(p2p.ID)
MarkAttempt(*p2p.NetAddress) MarkAttempt(*p2p.NetAddress)
MarkBad(*p2p.NetAddress) MarkBad(*p2p.NetAddress)
@ -296,11 +296,11 @@ func (a *addrBook) PickAddress(biasTowardsNewAddrs int) *p2p.NetAddress {
// MarkGood implements AddrBook - it marks the peer as good and // MarkGood implements AddrBook - it marks the peer as good and
// moves it into an "old" bucket. // moves it into an "old" bucket.
func (a *addrBook) MarkGood(addr *p2p.NetAddress) {
func (a *addrBook) MarkGood(id p2p.ID) {
a.mtx.Lock() a.mtx.Lock()
defer a.mtx.Unlock() defer a.mtx.Unlock()
ka := a.addrLookup[addr.ID]
ka := a.addrLookup[id]
if ka == nil { if ka == nil {
return return
} }


+ 4
- 4
p2p/pex/addrbook_test.go View File

@ -41,7 +41,7 @@ func TestAddrBookPickAddress(t *testing.T) {
assert.NotNil(t, addr, "expected an address") assert.NotNil(t, addr, "expected an address")
// pick an address when we only have old address // pick an address when we only have old address
book.MarkGood(addrSrc.addr)
book.MarkGood(addrSrc.addr.ID)
addr = book.PickAddress(0) addr = book.PickAddress(0)
assert.NotNil(t, addr, "expected an address") assert.NotNil(t, addr, "expected an address")
addr = book.PickAddress(50) addr = book.PickAddress(50)
@ -126,7 +126,7 @@ func TestAddrBookPromoteToOld(t *testing.T) {
// Promote half of them // Promote half of them
for i, addrSrc := range randAddrs { for i, addrSrc := range randAddrs {
if i%2 == 0 { if i%2 == 0 {
book.MarkGood(addrSrc.addr)
book.MarkGood(addrSrc.addr.ID)
} }
} }
@ -330,7 +330,7 @@ func TestAddrBookGetSelectionWithBias(t *testing.T) {
randAddrsLen := len(randAddrs) randAddrsLen := len(randAddrs)
for i, addrSrc := range randAddrs { for i, addrSrc := range randAddrs {
if int((float64(i)/float64(randAddrsLen))*100) >= 20 { if int((float64(i)/float64(randAddrsLen))*100) >= 20 {
book.MarkGood(addrSrc.addr)
book.MarkGood(addrSrc.addr.ID)
} }
} }
@ -569,7 +569,7 @@ func createAddrBookWithMOldAndNNewAddrs(t *testing.T, nOld, nNew int) (book *add
randAddrs := randNetAddressPairs(t, nOld) randAddrs := randNetAddressPairs(t, nOld)
for _, addr := range randAddrs { for _, addr := range randAddrs {
book.AddAddress(addr.addr, addr.src) book.AddAddress(addr.addr, addr.src)
book.MarkGood(addr.addr)
book.MarkGood(addr.addr.ID)
} }
randAddrs = randNetAddressPairs(t, nNew) randAddrs = randNetAddressPairs(t, nNew)


+ 12
- 3
p2p/pex/pex_reactor.go View File

@ -170,12 +170,18 @@ func (r *PEXReactor) AddPeer(p Peer) {
} }
} else { } else {
// inbound peer is its own source // inbound peer is its own source
addr := p.SocketAddr()
addr, err := p.NodeInfo().NetAddress()
if err != nil {
r.Logger.Error("Failed to get peer NetAddress", "err", err, "peer", p)
return
}
// Make it explicit that addr and src are the same for an inbound peer.
src := addr src := addr
// add to book. dont RequestAddrs right away because // add to book. dont RequestAddrs right away because
// we don't trust inbound as much - let ensurePeersRoutine handle it. // we don't trust inbound as much - let ensurePeersRoutine handle it.
err := r.book.AddAddress(addr, src)
err = r.book.AddAddress(addr, src)
r.logErrAddrBook(err) r.logErrAddrBook(err)
} }
} }
@ -312,7 +318,10 @@ func (r *PEXReactor) ReceiveAddrs(addrs []*p2p.NetAddress, src Peer) error {
} }
r.requestsSent.Delete(id) r.requestsSent.Delete(id)
srcAddr := src.SocketAddr()
srcAddr, err := src.NodeInfo().NetAddress()
if err != nil {
return err
}
for _, netAddr := range addrs { for _, netAddr := range addrs {
// Validate netAddr. Disconnect from a peer if it sends us invalid data. // Validate netAddr. Disconnect from a peer if it sends us invalid data.
if netAddr == nil { if netAddr == nil {


+ 1
- 1
p2p/pex/pex_reactor_test.go View File

@ -539,7 +539,7 @@ func testCreateSeed(dir string, id int, knownAddrs, srcAddrs []*p2p.NetAddress)
book.SetLogger(log.TestingLogger()) book.SetLogger(log.TestingLogger())
for j := 0; j < len(knownAddrs); j++ { for j := 0; j < len(knownAddrs); j++ {
book.AddAddress(knownAddrs[j], srcAddrs[j]) book.AddAddress(knownAddrs[j], srcAddrs[j])
book.MarkGood(knownAddrs[j])
book.MarkGood(knownAddrs[j].ID)
} }
sw.SetAddrBook(book) sw.SetAddrBook(book)


+ 2
- 2
p2p/switch.go View File

@ -48,7 +48,7 @@ type AddrBook interface {
AddAddress(addr *NetAddress, src *NetAddress) error AddAddress(addr *NetAddress, src *NetAddress) error
AddOurAddress(*NetAddress) AddOurAddress(*NetAddress)
OurAddress(*NetAddress) bool OurAddress(*NetAddress) bool
MarkGood(*NetAddress)
MarkGood(ID)
RemoveAddress(*NetAddress) RemoveAddress(*NetAddress)
HasAddress(*NetAddress) bool HasAddress(*NetAddress) bool
Save() Save()
@ -385,7 +385,7 @@ func (sw *Switch) SetAddrBook(addrBook AddrBook) {
// like contributed to consensus. // like contributed to consensus.
func (sw *Switch) MarkPeerAsGood(peer Peer) { func (sw *Switch) MarkPeerAsGood(peer Peer) {
if sw.addrBook != nil { if sw.addrBook != nil {
sw.addrBook.MarkGood(peer.SocketAddr())
sw.addrBook.MarkGood(peer.ID())
} }
} }


+ 1
- 1
p2p/switch_test.go View File

@ -626,7 +626,7 @@ func (book *addrBookMock) OurAddress(addr *NetAddress) bool {
_, ok := book.ourAddrs[addr.String()] _, ok := book.ourAddrs[addr.String()]
return ok return ok
} }
func (book *addrBookMock) MarkGood(*NetAddress) {}
func (book *addrBookMock) MarkGood(ID) {}
func (book *addrBookMock) HasAddress(addr *NetAddress) bool { func (book *addrBookMock) HasAddress(addr *NetAddress) bool {
_, ok := book.addrs[addr.String()] _, ok := book.addrs[addr.String()]
return ok return ok


+ 1
- 1
p2p/test_util.go View File

@ -23,7 +23,7 @@ type mockNodeInfo struct {
} }
func (ni mockNodeInfo) ID() ID { return ni.addr.ID } func (ni mockNodeInfo) ID() ID { return ni.addr.ID }
func (ni mockNodeInfo) NetAddress() *NetAddress { return ni.addr }
func (ni mockNodeInfo) NetAddress() (*NetAddress, error) { return ni.addr, nil }
func (ni mockNodeInfo) Validate() error { return nil } func (ni mockNodeInfo) Validate() error { return nil }
func (ni mockNodeInfo) CompatibleWith(other NodeInfo) error { return nil } func (ni mockNodeInfo) CompatibleWith(other NodeInfo) error { return nil }


Loading…
Cancel
Save