mirror of
https://github.com/yggdrasil-network/yggdrasil-go.git
synced 2025-01-07 08:06:50 +00:00
290 lines
8.4 KiB
Go
290 lines
8.4 KiB
Go
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
|
|
|
|
// Send:
|
|
// Receive a packet from the adapter
|
|
// 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
|
|
// The router then runs some sanity checks before passing it to the adapter
|
|
|
|
import (
|
|
//"bytes"
|
|
|
|
"time"
|
|
|
|
"github.com/yggdrasil-network/yggdrasil-go/src/address"
|
|
"github.com/yggdrasil-network/yggdrasil-go/src/crypto"
|
|
|
|
"github.com/Arceliar/phony"
|
|
)
|
|
|
|
// 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 phony.Inbox goroutine is responsible for managing all information related to the dht, searches, and crypto sessions.
|
|
type router struct {
|
|
phony.Inbox
|
|
core *Core
|
|
addr address.Address
|
|
subnet address.Subnet
|
|
out func([]byte) // packets we're sending to the network, link to peer's "in"
|
|
dht dht
|
|
nodeinfo nodeinfo
|
|
searches searches
|
|
sessions sessions
|
|
intf routerInterface
|
|
peer *peer
|
|
table *lookupTable // has a copy of our locator
|
|
}
|
|
|
|
// Initializes the router struct, which includes setting up channels to/from the adapter.
|
|
func (r *router) init(core *Core) {
|
|
r.core = core
|
|
r.addr = *address.AddrForNodeID(&r.dht.nodeID)
|
|
r.subnet = *address.SubnetForNodeID(&r.dht.nodeID)
|
|
r.intf.router = r
|
|
phony.Block(&r.core.peers, func() {
|
|
// FIXME don't block here!
|
|
r.peer = r.core.peers._newPeer(&r.core.boxPub, &r.core.sigPub, &crypto.BoxSharedKey{}, &r.intf)
|
|
})
|
|
r.peer.Act(r, r.peer._handleIdle)
|
|
r.out = func(bs []byte) {
|
|
r.peer.handlePacketFrom(r, bs)
|
|
}
|
|
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()
|
|
r.dht.init(r)
|
|
r.searches.init(r)
|
|
r.sessions.init(r)
|
|
}
|
|
|
|
func (r *router) updateTable(from phony.Actor, table *lookupTable) {
|
|
r.Act(from, func() {
|
|
r.table = table
|
|
r.nodeinfo.Act(r, func() {
|
|
r.nodeinfo.table = table
|
|
})
|
|
for _, ses := range r.sessions.sinfos {
|
|
sinfo := ses
|
|
sinfo.Act(r, func() {
|
|
sinfo.table = table
|
|
})
|
|
}
|
|
})
|
|
}
|
|
|
|
// Reconfigures the router and any child modules. This should only ever be run
|
|
// by the router actor.
|
|
func (r *router) reconfigure() {
|
|
// Reconfigure the router
|
|
current := r.core.config.GetCurrent()
|
|
r.core.log.Println("Reloading NodeInfo...")
|
|
if err := r.nodeinfo.setNodeInfo(current.NodeInfo, current.NodeInfoPrivacy); err != nil {
|
|
r.core.log.Errorln("Error reloading NodeInfo:", err)
|
|
} else {
|
|
r.core.log.Infoln("NodeInfo updated")
|
|
}
|
|
// Reconfigure children
|
|
r.dht.reconfigure()
|
|
r.searches.reconfigure()
|
|
r.sessions.reconfigure()
|
|
}
|
|
|
|
// Starts the tickerLoop goroutine.
|
|
func (r *router) start() error {
|
|
r.core.log.Infoln("Starting router")
|
|
go r.doMaintenance()
|
|
return nil
|
|
}
|
|
|
|
// Insert a peer info into the dht, TODO? make the dht a separate actor
|
|
func (r *router) insertPeer(from phony.Actor, info *dhtInfo) {
|
|
r.Act(from, func() {
|
|
r.dht.insertPeer(info)
|
|
})
|
|
}
|
|
|
|
// Reset sessions and DHT after the switch sees our coords change
|
|
func (r *router) reset(from phony.Actor) {
|
|
r.Act(from, func() {
|
|
r.sessions.reset()
|
|
r.dht.reset()
|
|
})
|
|
}
|
|
|
|
// TODO remove reconfigure so this is just a ticker loop
|
|
// and then find something better than a ticker loop to schedule things...
|
|
func (r *router) doMaintenance() {
|
|
phony.Block(r, func() {
|
|
// Any periodic maintenance stuff goes here
|
|
r.core.switchTable.doMaintenance(r)
|
|
r.dht.doMaintenance()
|
|
r.sessions.cleanup()
|
|
})
|
|
time.AfterFunc(time.Second, r.doMaintenance)
|
|
}
|
|
|
|
// Checks incoming traffic type and passes it to the appropriate handler.
|
|
func (r *router) _handlePacket(packet []byte) {
|
|
pType, pTypeLen := wire_decode_uint64(packet)
|
|
if pTypeLen == 0 {
|
|
return
|
|
}
|
|
switch pType {
|
|
case wire_Traffic:
|
|
r._handleTraffic(packet)
|
|
case wire_ProtocolTraffic:
|
|
r._handleProto(packet)
|
|
default:
|
|
}
|
|
}
|
|
|
|
// Handles incoming traffic, i.e. encapuslated ordinary IPv6 packets.
|
|
// Passes them to the crypto session worker to be decrypted and sent to the adapter.
|
|
func (r *router) _handleTraffic(packet []byte) {
|
|
p := wire_trafficPacket{}
|
|
if !p.decode(packet) {
|
|
return
|
|
}
|
|
sinfo, isIn := r.sessions.getSessionForHandle(&p.Handle)
|
|
if !isIn {
|
|
return
|
|
}
|
|
sinfo.recv(r, &p)
|
|
}
|
|
|
|
// Handles protocol traffic by decrypting it, checking its type, and passing it to the appropriate handler for that traffic type.
|
|
func (r *router) _handleProto(packet []byte) {
|
|
// First parse the packet
|
|
p := wire_protoTrafficPacket{}
|
|
if !p.decode(packet) {
|
|
return
|
|
}
|
|
// Now try to open the payload
|
|
var sharedKey *crypto.BoxSharedKey
|
|
if p.ToKey == r.core.boxPub {
|
|
// Try to open using our permanent key
|
|
sharedKey = r.sessions.getSharedKey(&r.core.boxPriv, &p.FromKey)
|
|
} else {
|
|
return
|
|
}
|
|
bs, isOK := crypto.BoxOpen(sharedKey, p.Payload, &p.Nonce)
|
|
if !isOK {
|
|
return
|
|
}
|
|
// Now do something with the bytes in bs...
|
|
// send dht messages to dht, sessionRefresh to sessions, data to adapter...
|
|
// For data, should check that key and IP match...
|
|
bsType, bsTypeLen := wire_decode_uint64(bs)
|
|
if bsTypeLen == 0 {
|
|
return
|
|
}
|
|
switch bsType {
|
|
case wire_SessionPing:
|
|
r._handlePing(bs, &p.FromKey)
|
|
case wire_SessionPong:
|
|
r._handlePong(bs, &p.FromKey)
|
|
case wire_NodeInfoRequest:
|
|
fallthrough
|
|
case wire_NodeInfoResponse:
|
|
r._handleNodeInfo(bs, &p.FromKey)
|
|
case wire_DHTLookupRequest:
|
|
r._handleDHTReq(bs, &p.FromKey)
|
|
case wire_DHTLookupResponse:
|
|
r._handleDHTRes(bs, &p.FromKey)
|
|
default:
|
|
}
|
|
}
|
|
|
|
// 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) {
|
|
ping := sessionPing{}
|
|
if !ping.decode(bs) {
|
|
return
|
|
}
|
|
ping.SendPermPub = *fromKey
|
|
r.sessions.handlePing(&ping)
|
|
}
|
|
|
|
// Handles session pongs (which are really pings with an extra flag to prevent acknowledgement).
|
|
func (r *router) _handlePong(bs []byte, fromKey *crypto.BoxPubKey) {
|
|
r._handlePing(bs, fromKey)
|
|
}
|
|
|
|
// Decodes dht requests and passes them to dht.handleReq to trigger a lookup/response.
|
|
func (r *router) _handleDHTReq(bs []byte, fromKey *crypto.BoxPubKey) {
|
|
req := dhtReq{}
|
|
if !req.decode(bs) {
|
|
return
|
|
}
|
|
req.Key = *fromKey
|
|
r.dht.handleReq(&req)
|
|
}
|
|
|
|
// 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) {
|
|
res := dhtRes{}
|
|
if !res.decode(bs) {
|
|
return
|
|
}
|
|
res.Key = *fromKey
|
|
r.dht.handleRes(&res)
|
|
}
|
|
|
|
// Decodes nodeinfo request
|
|
func (r *router) _handleNodeInfo(bs []byte, fromKey *crypto.BoxPubKey) {
|
|
req := nodeinfoReqRes{}
|
|
if !req.decode(bs) {
|
|
return
|
|
}
|
|
req.SendPermPub = *fromKey
|
|
r.nodeinfo.handleNodeInfo(r, &req)
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// routerInterface is a helper that implements linkInterface
|
|
type routerInterface struct {
|
|
router *router
|
|
}
|
|
|
|
func (intf *routerInterface) out(bss [][]byte) {
|
|
// Note that this is run in the peer's goroutine
|
|
intf.router.Act(intf.router.peer, func() {
|
|
for _, bs := range bss {
|
|
intf.router._handlePacket(bs)
|
|
}
|
|
})
|
|
//intf.router.peer.Act(nil, intf.router.peer._handleIdle)
|
|
intf.router.peer._handleIdle()
|
|
}
|
|
|
|
func (intf *routerInterface) linkOut(_ []byte) {}
|
|
|
|
func (intf *routerInterface) notifyQueued(seq uint64) {}
|
|
|
|
func (intf *routerInterface) close() {}
|
|
|
|
func (intf *routerInterface) name() string { return "(self)" }
|
|
|
|
func (intf *routerInterface) local() string { return "(self)" }
|
|
|
|
func (intf *routerInterface) remote() string { return "(self)" }
|
|
|
|
func (intf *routerInterface) interfaceType() string { return "self" }
|