mirror of
https://github.com/kaspanet/kaspad.git
synced 2025-05-21 14:26:45 +00:00

* [NOD-1151] Added gRPC server for seeding peers * [NOD-1151] Fix branch after rebase * [NOD-1151] Lint infrastructure/config/config.go * [NOD-1151] Use Warnf instead of Infof * [NOD-1151] Check if a.cfg.GRPCSeed is defined * [NOD-1151] Delete Makefile - Use go generate instead of Makefile * [NOD-1151] Panic in case of GRPCSeed and DNSSeed are both defined * [NOD-1151] Add generate.go file * [NOD-1151] Allow dnsseed and grpcseed to work together Co-authored-by: Bogdan Ovsiannikov <takahawkkun@gmail.com> Co-authored-by: yaroslavr-itd <yaroslav.r@it-dimension.com> Co-authored-by: Ori Newman <orinewman1@gmail.com>
172 lines
4.7 KiB
Go
172 lines
4.7 KiB
Go
// Copyright (c) 2016 The btcsuite developers
|
|
// Use of this source code is governed by an ISC
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package dnsseed
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"math/rand"
|
|
"net"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/kaspanet/kaspad/app/appmessage"
|
|
pb2 "github.com/kaspanet/kaspad/infrastructure/network/dnsseed/pb"
|
|
"google.golang.org/grpc"
|
|
|
|
"github.com/kaspanet/kaspad/util/mstime"
|
|
|
|
"github.com/kaspanet/kaspad/util/subnetworkid"
|
|
|
|
"github.com/kaspanet/kaspad/domain/dagconfig"
|
|
)
|
|
|
|
const (
|
|
// These constants are used by the DNS seed code to pick a random last
|
|
// seen time.
|
|
secondsIn3Days int32 = 24 * 60 * 60 * 3
|
|
secondsIn4Days int32 = 24 * 60 * 60 * 4
|
|
|
|
// SubnetworkIDPrefixChar is the prefix of subnetworkID, when building a DNS seed request
|
|
SubnetworkIDPrefixChar byte = 'n'
|
|
|
|
// ServiceFlagPrefixChar is the prefix of service flag, when building a DNS seed request
|
|
ServiceFlagPrefixChar byte = 'x'
|
|
)
|
|
|
|
// OnSeed is the signature of the callback function which is invoked when DNS
|
|
// seeding is successful.
|
|
type OnSeed func(addrs []*appmessage.NetAddress)
|
|
|
|
// LookupFunc is the signature of the DNS lookup function.
|
|
type LookupFunc func(string) ([]net.IP, error)
|
|
|
|
// SeedFromDNS uses DNS seeding to populate the address manager with peers.
|
|
func SeedFromDNS(dagParams *dagconfig.Params, customSeed string, reqServices appmessage.ServiceFlag, includeAllSubnetworks bool,
|
|
subnetworkID *subnetworkid.SubnetworkID, lookupFn LookupFunc, seedFn OnSeed) {
|
|
|
|
var dnsSeeds []string
|
|
if customSeed != "" {
|
|
dnsSeeds = []string{customSeed}
|
|
} else {
|
|
dnsSeeds = dagParams.DNSSeeds
|
|
}
|
|
|
|
for _, dnsseed := range dnsSeeds {
|
|
var host string
|
|
if reqServices == appmessage.SFNodeNetwork {
|
|
host = dnsseed
|
|
} else {
|
|
host = fmt.Sprintf("%c%x.%s", ServiceFlagPrefixChar, uint64(reqServices), dnsseed)
|
|
}
|
|
|
|
if !includeAllSubnetworks {
|
|
if subnetworkID != nil {
|
|
host = fmt.Sprintf("%c%s.%s", SubnetworkIDPrefixChar, subnetworkID, host)
|
|
} else {
|
|
host = fmt.Sprintf("%c.%s", SubnetworkIDPrefixChar, host)
|
|
}
|
|
}
|
|
|
|
spawn("SeedFromDNS", func() {
|
|
randSource := rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
|
|
seedPeers, err := lookupFn(host)
|
|
if err != nil {
|
|
log.Infof("DNS discovery failed on seed %s: %s", host, err)
|
|
return
|
|
}
|
|
numPeers := len(seedPeers)
|
|
|
|
log.Infof("%d addresses found from DNS seed %s", numPeers, host)
|
|
|
|
if numPeers == 0 {
|
|
return
|
|
}
|
|
addresses := make([]*appmessage.NetAddress, len(seedPeers))
|
|
// if this errors then we have *real* problems
|
|
intPort, _ := strconv.Atoi(dagParams.DefaultPort)
|
|
for i, peer := range seedPeers {
|
|
addresses[i] = appmessage.NewNetAddressTimestamp(
|
|
// seed with addresses from a time randomly selected
|
|
// between 3 and 7 days ago.
|
|
mstime.Now().Add(-1*time.Second*time.Duration(secondsIn3Days+
|
|
randSource.Int31n(secondsIn4Days))),
|
|
0, peer, uint16(intPort))
|
|
}
|
|
|
|
seedFn(addresses)
|
|
})
|
|
}
|
|
}
|
|
|
|
// SeedFromGRPC send gRPC request to get list of peers for a given host
|
|
func SeedFromGRPC(dagParams *dagconfig.Params, host string, reqServices appmessage.ServiceFlag, includeAllSubnetworks bool,
|
|
subnetworkID *subnetworkid.SubnetworkID, seedFn OnSeed) {
|
|
|
|
spawn("SeedFromGRPC", func() {
|
|
|
|
randSource := rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
|
|
conn, err := grpc.Dial(host, grpc.WithInsecure())
|
|
client := pb2.NewPeerServiceClient(conn)
|
|
if err != nil {
|
|
log.Warnf("Failed to connect to gRPC server: %s", host)
|
|
}
|
|
|
|
var subnetID []byte
|
|
if subnetworkID != nil {
|
|
subnetID = subnetworkID.CloneBytes()
|
|
} else {
|
|
subnetID = nil
|
|
}
|
|
|
|
req := &pb2.GetPeersListRequest{
|
|
ServiceFlag: uint64(reqServices),
|
|
SubnetworkID: subnetID,
|
|
IncludeAllSubnetworks: includeAllSubnetworks,
|
|
}
|
|
res, err := client.GetPeersList(context.Background(), req)
|
|
|
|
if err != nil {
|
|
log.Infof("gRPC request to get peers failed (host=%s): %s", host, err)
|
|
return
|
|
}
|
|
|
|
seedPeers := fromProtobufAddresses(res.Addresses)
|
|
|
|
numPeers := len(seedPeers)
|
|
|
|
log.Infof("%d addresses found from DNS seed %s", numPeers, host)
|
|
|
|
if numPeers == 0 {
|
|
return
|
|
}
|
|
addresses := make([]*appmessage.NetAddress, len(seedPeers))
|
|
// if this errors then we have *real* problems
|
|
intPort, _ := strconv.Atoi(dagParams.DefaultPort)
|
|
for i, peer := range seedPeers {
|
|
addresses[i] = appmessage.NewNetAddressTimestamp(
|
|
// seed with addresses from a time randomly selected
|
|
// between 3 and 7 days ago.
|
|
mstime.Now().Add(-1*time.Second*time.Duration(secondsIn3Days+
|
|
randSource.Int31n(secondsIn4Days))),
|
|
0, peer, uint16(intPort))
|
|
}
|
|
|
|
seedFn(addresses)
|
|
})
|
|
}
|
|
|
|
func fromProtobufAddresses(proto []*pb2.NetAddress) []net.IP {
|
|
var addresses []net.IP
|
|
|
|
for _, pbAddr := range proto {
|
|
addresses = append(addresses, net.IP(pbAddr.IP))
|
|
}
|
|
|
|
return addresses
|
|
}
|