mirror of
https://github.com/kaspanet/kaspad.git
synced 2025-06-06 14:16:43 +00:00

* [NOD-1130] Delete rpcadapters.go. * [NOD-1130] Delete p2p. Move rpc to top level. * [NOD-1130] Remove DAGParams from rpcserverConfig. * [NOD-1130] Remove rpcserverPeer, rpcserverConnManager, rpcserverSyncManager, and rpcserverConfig. * [NOD-1130] Remove wallet RPC commands. * [NOD-1130] Remove wallet RPC commands. * [NOD-1130] Remove connmgr and peer. * [NOD-1130] Move rpcmodel into rpc. * [NOD-1130] Implement ConnectionCount. * [NOD-1130] Remove ping and node RPC commands. * [NOD-1130] Dummify handleGetNetTotals. * [NOD-1130] Add NetConnection to Peer. * [NOD-1130] Fix merge errors. * [NOD-1130] Implement Peers. * [NOD-1130] Fix HandleGetConnectedPeerInfo. * [NOD-1130] Fix SendRawTransaction. * [NOD-1130] Rename addManualNode to connect and removeManualNode to disconnect. * [NOD-1130] Add a stub for AddBlock. * [NOD-1130] Fix tests. * [NOD-1130] Replace half-baked contents of RemoveConnection with a stub. * [NOD-1130] Fix merge errors. * [NOD-1130] Make golint happy. * [NOD-1130] Get rid of something weird. * [NOD-1130] Rename minerClient back to client. * [NOD-1130] Add a few fields to GetConnectedPeerInfoResult. * [NOD-1130] Rename oneTry to isPermanent. * [NOD-1130] Implement ConnectionCount in NetAdapter. * [NOD-1130] Move RawMempoolVerbose out of mempool. * [NOD-1130] Move isSynced into the mining package. * [NOD-1130] Fix a compilation error. * [NOD-1130] Make golint happy. * [NOD-1130] Fix merge errors.
123 lines
3.1 KiB
Go
123 lines
3.1 KiB
Go
package handshake
|
|
|
|
import (
|
|
"github.com/kaspanet/kaspad/addrmgr"
|
|
"github.com/kaspanet/kaspad/blockdag"
|
|
"github.com/kaspanet/kaspad/config"
|
|
"github.com/kaspanet/kaspad/netadapter"
|
|
"github.com/kaspanet/kaspad/protocol/common"
|
|
"sync"
|
|
"sync/atomic"
|
|
|
|
routerpkg "github.com/kaspanet/kaspad/netadapter/router"
|
|
peerpkg "github.com/kaspanet/kaspad/protocol/peer"
|
|
"github.com/kaspanet/kaspad/util/locks"
|
|
"github.com/kaspanet/kaspad/wire"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// HandleHandshakeContext is the interface for the context needed for the HandleHandshake flow.
|
|
type HandleHandshakeContext interface {
|
|
Config() *config.Config
|
|
NetAdapter() *netadapter.NetAdapter
|
|
DAG() *blockdag.BlockDAG
|
|
AddressManager() *addrmgr.AddrManager
|
|
StartIBDIfRequired()
|
|
AddToPeers(peer *peerpkg.Peer) error
|
|
}
|
|
|
|
// HandleHandshake sets up the handshake protocol - It sends a version message and waits for an incoming
|
|
// version message, as well as a verack for the sent version
|
|
func HandleHandshake(context HandleHandshakeContext, router *routerpkg.Router,
|
|
netConnection *netadapter.NetConnection) (peer *peerpkg.Peer, closed bool, err error) {
|
|
|
|
receiveVersionRoute, err := router.AddIncomingRoute([]wire.MessageCommand{wire.CmdVersion})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
sendVersionRoute, err := router.AddIncomingRoute([]wire.MessageCommand{wire.CmdVerAck})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// For HandleHandshake to finish, we need to get from the other node
|
|
// a version and verack messages, so we increase the wait group by 2
|
|
// and block HandleHandshake with wg.Wait().
|
|
wg := sync.WaitGroup{}
|
|
wg.Add(2)
|
|
|
|
errChanUsed := uint32(0)
|
|
errChan := make(chan error)
|
|
|
|
peer = peerpkg.New(netConnection)
|
|
|
|
var peerAddress *wire.NetAddress
|
|
spawn("HandleHandshake-ReceiveVersion", func() {
|
|
defer wg.Done()
|
|
address, err := ReceiveVersion(context, receiveVersionRoute, router.OutgoingRoute(), peer)
|
|
if err != nil {
|
|
log.Errorf("error from ReceiveVersion: %s", err)
|
|
}
|
|
if err != nil {
|
|
if atomic.AddUint32(&errChanUsed, 1) != 1 {
|
|
errChan <- err
|
|
}
|
|
return
|
|
}
|
|
peerAddress = address
|
|
})
|
|
|
|
spawn("HandleHandshake-SendVersion", func() {
|
|
defer wg.Done()
|
|
err := SendVersion(context, sendVersionRoute, router.OutgoingRoute())
|
|
if err != nil {
|
|
log.Errorf("error from SendVersion: %s", err)
|
|
}
|
|
if err != nil {
|
|
if atomic.AddUint32(&errChanUsed, 1) != 1 {
|
|
errChan <- err
|
|
}
|
|
return
|
|
}
|
|
})
|
|
|
|
select {
|
|
case err := <-errChan:
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
return nil, true, nil
|
|
case <-locks.ReceiveFromChanWhenDone(func() { wg.Wait() }):
|
|
}
|
|
|
|
err = context.AddToPeers(peer)
|
|
if err != nil {
|
|
if errors.Is(err, common.ErrPeerWithSameIDExists) {
|
|
return nil, false, err
|
|
}
|
|
panic(err)
|
|
}
|
|
|
|
peerID := peer.ID()
|
|
err = context.NetAdapter().AssociateRouterID(router, peerID)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if peerAddress != nil {
|
|
subnetworkID := peer.SubnetworkID()
|
|
context.AddressManager().AddAddress(peerAddress, peerAddress, subnetworkID)
|
|
context.AddressManager().Good(peerAddress, subnetworkID)
|
|
}
|
|
|
|
context.StartIBDIfRequired()
|
|
|
|
err = router.RemoveRoute([]wire.MessageCommand{wire.CmdVersion, wire.CmdVerAck})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return peer, false, nil
|
|
}
|