dnsseeder/zcash/client.go

213 lines
5.3 KiB
Go
Raw Normal View History

2019-10-09 15:32:15 -07:00
package zcash
import (
"log"
"net"
"os"
"sync"
"time"
"github.com/btcsuite/btcd/peer"
"github.com/btcsuite/btcd/wire"
"github.com/gtank/coredns-zcash/zcash/network"
"github.com/pkg/errors"
)
var (
ErrRepeatConnection = errors.New("attempted repeat connection to existing peer")
)
2019-10-09 15:32:15 -07:00
var defaultPeerConfig = &peer.Config{
UserAgentName: "MagicBean",
UserAgentVersion: "2.0.7",
ChainParams: nil,
Services: 0,
TrickleInterval: time.Second * 10,
ProtocolVersion: 170009, // Blossom
}
type Seeder struct {
peer *peer.Peer
config *peer.Config
logger *log.Logger
2019-10-09 15:32:15 -07:00
handshakeSignals *sync.Map
pendingPeers *sync.Map
livePeers *sync.Map
2019-10-09 15:32:15 -07:00
// For mutating the above
peerState sync.RWMutex
}
func NewSeeder(network network.Network) (*Seeder, error) {
config, err := newSeederPeerConfig(network, defaultPeerConfig)
if err != nil {
return nil, errors.Wrap(err, "could not construct seeder")
}
logger := log.New(os.Stdout, "zcash_seeder: ", log.Ldate|log.Ltime|log.Lshortfile|log.LUTC)
newSeeder := Seeder{
config: config,
logger: logger,
handshakeSignals: new(sync.Map),
pendingPeers: new(sync.Map),
livePeers: new(sync.Map),
}
newSeeder.config.Listeners.OnVerAck = newSeeder.onVerAck
return &newSeeder, nil
}
func newTestSeeder(network network.Network) (*Seeder, error) {
config, err := newSeederPeerConfig(network, defaultPeerConfig)
if err != nil {
return nil, errors.Wrap(err, "could not construct seeder")
}
sink, _ := os.OpenFile(os.DevNull, os.O_WRONLY, 0666)
logger := log.New(sink, "zcash_seeder: ", log.Ldate|log.Ltime|log.Lshortfile|log.LUTC)
// Allows connections to self for easy mocking
config.AllowSelfConns = true
newSeeder := Seeder{
config: config,
logger: logger,
handshakeSignals: new(sync.Map),
pendingPeers: new(sync.Map),
livePeers: new(sync.Map),
}
newSeeder.config.Listeners.OnVerAck = newSeeder.onVerAck
return &newSeeder, nil
}
2019-10-09 15:32:15 -07:00
func newSeederPeerConfig(magic network.Network, template *peer.Config) (*peer.Config, error) {
var newPeerConfig peer.Config
// Load the default values
if template != nil {
newPeerConfig = *template
}
params, err := network.GetNetworkParams(magic)
if err != nil {
return nil, errors.Wrap(err, "couldn't construct peer config")
}
newPeerConfig.ChainParams = params
return &newPeerConfig, nil
}
func (s *Seeder) onVerAck(p *peer.Peer, msg *wire.MsgVerAck) {
// Check if we're expecting to hear from this peer
_, ok := s.pendingPeers.Load(p.Addr())
2019-10-09 15:32:15 -07:00
if !ok {
s.logger.Printf("Got verack from unexpected peer %s", p.Addr())
return
}
2019-10-09 15:32:15 -07:00
// Add to set of live peers
s.livePeers.Store(p.Addr(), p)
2019-10-09 15:32:15 -07:00
// Remove from set of pending peers
s.pendingPeers.Delete(p.Addr())
2019-10-09 15:32:15 -07:00
// Signal successful connection
if signal, ok := s.handshakeSignals.Load(p.Addr()); ok {
signal.(chan struct{}) <- struct{}{}
return
2019-10-09 15:32:15 -07:00
}
s.logger.Printf("Got verack from peer without a callback channel: %s", p.Addr())
2019-10-09 15:32:15 -07:00
}
// ConnectToPeer attempts to connect to a peer on the default port at the
// specified address. It returns either a live peer connection or an error.
func (s *Seeder) ConnectToPeer(addr string) error {
2019-10-09 15:32:15 -07:00
connectionString := net.JoinHostPort(addr, s.config.ChainParams.DefaultPort)
p, err := peer.NewOutboundPeer(s.config, connectionString)
if err != nil {
return errors.Wrap(err, "constructing outbound peer")
2019-10-09 15:32:15 -07:00
}
_, alreadyPending := s.pendingPeers.LoadOrStore(p.Addr(), p)
_, alreadyHandshaking := s.handshakeSignals.LoadOrStore(p.Addr(), make(chan struct{}, 1))
_, alreadyLive := s.livePeers.Load(p.Addr())
if alreadyPending || alreadyHandshaking || alreadyLive {
s.logger.Printf("Attempted repeat connection to peer %s", p.Addr())
return ErrRepeatConnection
}
2019-10-09 15:32:15 -07:00
conn, err := net.Dial("tcp", p.Addr())
if err != nil {
return errors.Wrap(err, "dialing new peer address")
2019-10-09 15:32:15 -07:00
}
// Begin connection negotiation.
s.logger.Printf("Handshake initated with new peer %s", p.Addr())
2019-10-09 15:32:15 -07:00
p.AssociateConnection(conn)
handshakeChan, _ := s.handshakeSignals.Load(p.Addr())
select {
case <-handshakeChan.(chan struct{}):
s.logger.Printf("Handshake completed with new peer %s", p.Addr())
s.handshakeSignals.Delete(p.Addr())
return nil
case <-time.After(time.Second * 1):
return errors.New("peer handshake timed out")
2019-10-09 15:32:15 -07:00
}
panic("This should be unreachable")
}
func (s *Seeder) GetPeer(addr string) (*peer.Peer, error) {
lookupKey := net.JoinHostPort(addr, s.config.ChainParams.DefaultPort)
p, ok := s.livePeers.Load(lookupKey)
2019-10-09 15:32:15 -07:00
if ok {
return p.(*peer.Peer), nil
2019-10-09 15:32:15 -07:00
}
return nil, errors.New("no such active peer")
}
func (s *Seeder) DisconnectAllPeers() {
s.pendingPeers.Range(func(key, value interface{}) bool {
p, ok := value.(*peer.Peer)
if !ok {
s.logger.Printf("Invalid peer in pendingPeers")
return false
}
s.logger.Printf("Disconnecting from pending peer %s", p.Addr())
p.Disconnect()
p.WaitForDisconnect()
s.pendingPeers.Delete(key)
return true
})
s.livePeers.Range(func(key, value interface{}) bool {
p, ok := value.(*peer.Peer)
if !ok {
s.logger.Printf("Invalid peer in livePeers")
return false
}
s.logger.Printf("Disconnecting from live peer %s", p.Addr())
p.Disconnect()
p.WaitForDisconnect()
s.livePeers.Delete(key)
return true
})
2019-10-09 15:32:15 -07:00
}
func (s *Seeder) RequestAddresses() {}