5
0
mirror of https://github.com/cwinfo/yggdrasil-go.git synced 2024-11-23 17:01:36 +00:00
yggdrasil-go/src/yggdrasil/router.go

278 lines
8.6 KiB
Go
Raw Normal View History

2017-12-29 04:16:20 +00:00
package yggdrasil
// This part does most of the work to handle packets to/from yourself
// It also manages crypto and dht info
// TODO clean up old/unused code, maybe improve comments on whatever is left
2017-12-29 04:16:20 +00:00
// Send:
2019-03-28 19:09:19 +00:00
// Receive a packet from the adapter
2017-12-29 04:16:20 +00:00
// Look up session (if none exists, trigger a search)
// Hand off to session (which encrypts, etc)
// Session will pass it back to router.out, which hands it off to the self peer
// The self peer triggers a lookup to find which peer to send to next
// And then passes it to that's peer's peer.out function
// The peer.out function sends it over the wire to the matching peer
// Recv:
// A packet comes in off the wire, and goes to a peer.handlePacket
// The peer does a lookup, sees no better peer than the self
// Hands it to the self peer.out, which passes it to router.in
// If it's dht/seach/etc. traffic, the router passes it to that part
// If it's an encapsulated IPv6 packet, the router looks up the session for it
// The packet is passed to the session, which decrypts it, router.recvPacket
2019-03-28 19:09:19 +00:00
// The router then runs some sanity checks before passing it to the adapter
2017-12-29 04:16:20 +00:00
2018-06-12 22:50:08 +00:00
import (
//"bytes"
2019-04-22 22:12:13 +00:00
2018-06-12 22:50:08 +00:00
"time"
2018-01-04 22:37:51 +00:00
"github.com/yggdrasil-network/yggdrasil-go/src/address"
"github.com/yggdrasil-network/yggdrasil-go/src/crypto"
"github.com/yggdrasil-network/yggdrasil-go/src/util"
2018-06-12 22:50:08 +00:00
)
2017-12-29 04:16:20 +00:00
2019-03-28 19:09:19 +00:00
// The router struct has channels to/from the adapter device and a self peer (0), which is how messages are passed between this node and the peers/switch layer.
// The router's mainLoop goroutine is responsible for managing all information related to the dht, searches, and crypto sessions.
2017-12-29 04:16:20 +00:00
type router struct {
core *Core
2018-12-30 12:04:42 +00:00
reconfigure chan chan error
addr address.Address
subnet address.Subnet
2019-04-22 22:12:13 +00:00
in <-chan []byte // packets we received from the network, link to peer's "out"
out func([]byte) // packets we're sending to the network, link to peer's "in"
reset chan struct{} // signal that coords changed (re-init sessions/dht)
admin chan func() // pass a lambda for the admin socket to query stuff
nodeinfo nodeinfo
}
2019-03-28 19:09:19 +00:00
// Initializes the router struct, which includes setting up channels to/from the adapter.
2017-12-29 04:16:20 +00:00
func (r *router) init(core *Core) {
2018-01-04 22:37:51 +00:00
r.core = core
2018-12-30 12:04:42 +00:00
r.reconfigure = make(chan chan error, 1)
r.addr = *address.AddrForNodeID(&r.core.dht.nodeID)
r.subnet = *address.SubnetForNodeID(&r.core.dht.nodeID)
in := make(chan []byte, 1) // TODO something better than this...
self := linkInterface{
name: "(self)",
info: linkInfo{
local: "(self)",
remote: "(self)",
linkType: "self",
},
}
p := r.core.peers.newPeer(&r.core.boxPub, &r.core.sigPub, &crypto.BoxSharedKey{}, &self, nil)
p.out = func(packet []byte) { in <- packet }
2018-01-04 22:37:51 +00:00
r.in = in
out := make(chan []byte, 32)
go func() {
for packet := range out {
p.handlePacket(packet)
}
}()
out2 := make(chan []byte, 32)
go func() {
// This worker makes sure r.out never blocks
// It will buffer traffic long enough for the switch worker to take it
// If (somehow) you can send faster than the switch can receive, then this would use unbounded memory
// But crypto slows sends enough that the switch should always be able to take the packets...
var buf [][]byte
for {
buf = append(buf, <-out2)
for len(buf) > 0 {
select {
case bs := <-out2:
buf = append(buf, bs)
case out <- buf[0]:
buf = buf[1:]
}
}
}
}()
r.out = func(packet []byte) { out2 <- packet }
2018-01-04 22:37:51 +00:00
r.reset = make(chan struct{}, 1)
r.admin = make(chan func(), 32)
2019-01-16 13:23:26 +00:00
r.nodeinfo.init(r.core)
r.core.config.Mutex.RLock()
r.nodeinfo.setNodeInfo(r.core.config.Current.NodeInfo, r.core.config.Current.NodeInfoPrivacy)
r.core.config.Mutex.RUnlock()
2018-05-27 21:13:37 +00:00
}
// Starts the mainLoop goroutine.
2018-05-27 21:13:37 +00:00
func (r *router) start() error {
r.core.log.Infoln("Starting router")
2018-01-04 22:37:51 +00:00
go r.mainLoop()
2018-05-27 21:13:37 +00:00
return nil
2017-12-29 04:16:20 +00:00
}
2019-03-28 19:09:19 +00:00
// Takes traffic from the adapter and passes it to router.send, or from r.in and handles incoming traffic.
// Also adds new peer info to the DHT.
// Also resets the DHT and sesssions in the event of a coord change.
// Also does periodic maintenance stuff.
2017-12-29 04:16:20 +00:00
func (r *router) mainLoop() {
2018-01-04 22:37:51 +00:00
ticker := time.NewTicker(time.Second)
defer ticker.Stop()
for {
select {
case p := <-r.in:
r.handleIn(p)
case info := <-r.core.dht.peers:
r.core.dht.insertPeer(info)
2018-01-04 22:37:51 +00:00
case <-r.reset:
r.core.sessions.reset()
r.core.dht.reset()
2018-01-04 22:37:51 +00:00
case <-ticker.C:
{
// Any periodic maintenance stuff goes here
r.core.switchTable.doMaintenance()
2018-01-04 22:37:51 +00:00
r.core.dht.doMaintenance()
r.core.sessions.cleanup()
util.GetBytes() // To slowly drain things
2018-01-04 22:37:51 +00:00
}
case f := <-r.admin:
f()
2018-12-30 12:04:42 +00:00
case e := <-r.reconfigure:
current := r.core.config.GetCurrent()
e <- r.nodeinfo.setNodeInfo(current.NodeInfo, current.NodeInfoPrivacy)
2018-01-04 22:37:51 +00:00
}
}
2017-12-29 04:16:20 +00:00
}
// Checks incoming traffic type and passes it to the appropriate handler.
2017-12-29 04:16:20 +00:00
func (r *router) handleIn(packet []byte) {
2018-01-04 22:37:51 +00:00
pType, pTypeLen := wire_decode_uint64(packet)
if pTypeLen == 0 {
return
}
switch pType {
case wire_Traffic:
r.handleTraffic(packet)
case wire_ProtocolTraffic:
r.handleProto(packet)
2018-06-12 22:50:08 +00:00
default:
2018-01-04 22:37:51 +00:00
}
2017-12-29 04:16:20 +00:00
}
// Handles incoming traffic, i.e. encapuslated ordinary IPv6 packets.
2019-03-28 19:09:19 +00:00
// Passes them to the crypto session worker to be decrypted and sent to the adapter.
2017-12-29 04:16:20 +00:00
func (r *router) handleTraffic(packet []byte) {
defer util.PutBytes(packet)
2018-01-04 22:37:51 +00:00
p := wire_trafficPacket{}
if !p.decode(packet) {
return
}
2018-06-02 20:21:05 +00:00
sinfo, isIn := r.core.sessions.getSessionForHandle(&p.Handle)
2018-01-04 22:37:51 +00:00
if !isIn {
return
}
select {
case sinfo.fromRouter <- &p: // FIXME ideally this should be front drop
default:
util.PutBytes(p.Payload)
}
2017-12-29 04:16:20 +00:00
}
// Handles protocol traffic by decrypting it, checking its type, and passing it to the appropriate handler for that traffic type.
2017-12-29 04:16:20 +00:00
func (r *router) handleProto(packet []byte) {
2018-01-04 22:37:51 +00:00
// First parse the packet
p := wire_protoTrafficPacket{}
if !p.decode(packet) {
return
}
// Now try to open the payload
var sharedKey *crypto.BoxSharedKey
2018-06-02 20:21:05 +00:00
if p.ToKey == r.core.boxPub {
2018-01-04 22:37:51 +00:00
// Try to open using our permanent key
2018-06-02 20:21:05 +00:00
sharedKey = r.core.sessions.getSharedKey(&r.core.boxPriv, &p.FromKey)
2018-01-04 22:37:51 +00:00
} else {
return
}
bs, isOK := crypto.BoxOpen(sharedKey, p.Payload, &p.Nonce)
2018-01-04 22:37:51 +00:00
if !isOK {
return
}
// Now do something with the bytes in bs...
2019-03-28 19:09:19 +00:00
// send dht messages to dht, sessionRefresh to sessions, data to adapter...
2018-01-04 22:37:51 +00:00
// For data, should check that key and IP match...
bsType, bsTypeLen := wire_decode_uint64(bs)
if bsTypeLen == 0 {
return
}
switch bsType {
case wire_SessionPing:
2018-06-02 20:21:05 +00:00
r.handlePing(bs, &p.FromKey)
2018-01-04 22:37:51 +00:00
case wire_SessionPong:
2018-06-02 20:21:05 +00:00
r.handlePong(bs, &p.FromKey)
2018-12-15 22:37:11 +00:00
case wire_NodeInfoRequest:
fallthrough
2018-12-15 22:37:11 +00:00
case wire_NodeInfoResponse:
r.handleNodeInfo(bs, &p.FromKey)
2018-01-04 22:37:51 +00:00
case wire_DHTLookupRequest:
2018-06-02 20:21:05 +00:00
r.handleDHTReq(bs, &p.FromKey)
2018-01-04 22:37:51 +00:00
case wire_DHTLookupResponse:
2018-06-02 20:21:05 +00:00
r.handleDHTRes(bs, &p.FromKey)
default:
util.PutBytes(packet)
2018-01-04 22:37:51 +00:00
}
2017-12-29 04:16:20 +00:00
}
// Decodes session pings from wire format and passes them to sessions.handlePing where they either create or update a session.
func (r *router) handlePing(bs []byte, fromKey *crypto.BoxPubKey) {
2018-01-04 22:37:51 +00:00
ping := sessionPing{}
if !ping.decode(bs) {
return
}
ping.SendPermPub = *fromKey
2018-01-04 22:37:51 +00:00
r.core.sessions.handlePing(&ping)
2017-12-29 04:16:20 +00:00
}
// Handles session pongs (which are really pings with an extra flag to prevent acknowledgement).
func (r *router) handlePong(bs []byte, fromKey *crypto.BoxPubKey) {
2018-01-04 22:37:51 +00:00
r.handlePing(bs, fromKey)
2017-12-29 04:16:20 +00:00
}
// Decodes dht requests and passes them to dht.handleReq to trigger a lookup/response.
func (r *router) handleDHTReq(bs []byte, fromKey *crypto.BoxPubKey) {
2018-01-04 22:37:51 +00:00
req := dhtReq{}
if !req.decode(bs) {
return
}
req.Key = *fromKey
2018-01-04 22:37:51 +00:00
r.core.dht.handleReq(&req)
2017-12-29 04:16:20 +00:00
}
// Decodes dht responses and passes them to dht.handleRes to update the DHT table and further pass them to the search code (if applicable).
func (r *router) handleDHTRes(bs []byte, fromKey *crypto.BoxPubKey) {
2018-01-04 22:37:51 +00:00
res := dhtRes{}
if !res.decode(bs) {
return
}
res.Key = *fromKey
2018-01-04 22:37:51 +00:00
r.core.dht.handleRes(&res)
2017-12-29 04:16:20 +00:00
}
2018-12-15 22:37:11 +00:00
// Decodes nodeinfo request
func (r *router) handleNodeInfo(bs []byte, fromKey *crypto.BoxPubKey) {
2018-12-15 22:37:11 +00:00
req := nodeinfoReqRes{}
if !req.decode(bs) {
return
}
req.SendPermPub = *fromKey
2019-01-14 19:05:16 +00:00
r.nodeinfo.handleNodeInfo(&req)
}
// Passed a function to call.
// This will send the function to r.admin and block until it finishes.
// It's used by the admin socket to ask the router mainLoop goroutine about information in the session or dht structs, which cannot be read safely from outside that goroutine.
func (r *router) doAdmin(f func()) {
// Pass this a function that needs to be run by the router's main goroutine
// It will pass the function to the router and wait for the router to finish
done := make(chan struct{})
newF := func() {
f()
close(done)
}
r.admin <- newF
<-done
}