Reticulum/RNS/Transport.py

126 lines
4.0 KiB
Python
Raw Normal View History

import RNS
2016-06-03 19:02:02 +02:00
class Transport:
2018-03-19 16:39:08 +01:00
# Constants
BROADCAST = 0x00;
TRANSPORT = 0x01;
RELAY = 0x02;
TUNNEL = 0x03;
types = [BROADCAST, TRANSPORT, RELAY, TUNNEL]
2018-03-20 12:32:41 +01:00
interfaces = []
destinations = []
2018-04-16 17:13:39 +02:00
pending_links = []
active_links = []
2018-03-19 20:51:26 +01:00
packet_hashlist = []
2016-06-03 19:02:02 +02:00
@staticmethod
2018-04-16 17:13:39 +02:00
def outbound(packet):
Transport.cacheRaw(packet.raw)
2018-03-20 12:32:41 +01:00
for interface in Transport.interfaces:
if interface.OUT:
2018-04-16 17:13:39 +02:00
should_transmit = True
if packet.destination.type == RNS.Destination.LINK:
if interface != packet.destination.attached_interface:
should_transmit = False
if should_transmit:
RNS.log("Transmitting "+str(len(packet.raw))+" bytes via: "+str(interface), RNS.LOG_DEBUG)
interface.processOutgoing(packet.raw)
2016-06-03 19:02:02 +02:00
2018-03-19 20:51:26 +01:00
@staticmethod
def inbound(raw, interface=None):
packet_hash = RNS.Identity.fullHash(raw)
RNS.log(str(interface)+" received packet with hash "+RNS.prettyhexrep(packet_hash), RNS.LOG_DEBUG)
2018-03-19 20:51:26 +01:00
if not packet_hash in Transport.packet_hashlist:
Transport.packet_hashlist.append(packet_hash)
packet = RNS.Packet(None, raw)
2018-03-19 20:51:26 +01:00
packet.unpack()
2018-03-20 12:32:41 +01:00
packet.packet_hash = packet_hash
2018-04-16 17:13:39 +02:00
packet.receiving_interface = interface
2018-03-19 20:51:26 +01:00
if packet.packet_type == RNS.Packet.ANNOUNCE:
if RNS.Identity.validateAnnounce(packet):
2018-03-20 12:32:41 +01:00
Transport.cache(packet)
2018-04-16 17:13:39 +02:00
if packet.packet_type == RNS.Packet.LINKREQUEST:
2018-03-20 12:32:41 +01:00
for destination in Transport.destinations:
2018-03-19 20:51:26 +01:00
if destination.hash == packet.destination_hash and destination.type == packet.destination_type:
2018-03-20 12:32:41 +01:00
packet.destination = destination
destination.receive(packet)
Transport.cache(packet)
2018-04-16 17:13:39 +02:00
if packet.packet_type == RNS.Packet.RESOURCE:
if packet.destination_type == RNS.Destination.LINK:
for link in Transport.active_links:
if link.link_id == packet.destination_hash:
link.receive(packet)
Transport.cache(packet)
else:
for destination in Transport.destinations:
if destination.hash == packet.destination_hash and destination.type == packet.destination_type:
packet.destination = destination
destination.receive(packet)
Transport.cache(packet)
2018-03-20 12:32:41 +01:00
if packet.packet_type == RNS.Packet.PROOF:
2018-04-16 17:13:39 +02:00
if packet.header_type == RNS.Packet.HEADER_3:
# This is a link request proof, forward
# to a waiting link request
for link in Transport.pending_links:
if link.link_id == packet.destination_hash:
link.validateProof(packet)
else:
for destination in Transport.destinations:
if destination.hash == packet.destination_hash:
if destination.proofcallback != None:
destination.proofcallback(packet)
# TODO: add universal proof handling
2018-03-19 20:51:26 +01:00
2016-06-03 19:02:02 +02:00
@staticmethod
def registerDestination(destination):
destination.MTU = RNS.Reticulum.MTU
if destination.direction == RNS.Destination.IN:
2018-03-20 12:32:41 +01:00
Transport.destinations.append(destination)
2018-04-16 17:13:39 +02:00
@staticmethod
def registerLink(link):
RNS.log("Registering link "+str(link))
if link.initiator:
Transport.pending_links.append(link)
else:
Transport.active_links.append(link)
@staticmethod
def activateLink(link):
RNS.log("Activating link "+str(link))
if link in Transport.pending_links:
Transport.pending_links.remove(link)
Transport.active_links.append(link)
link.status = RNS.Link.ACTIVE
else:
RNS.log("Attempted to activate a link that was not in the pending table", RNS.LOG_ERROR)
@staticmethod
def shouldCache(packet):
# TODO: Implement sensible rules for which
# packets to cache
return False
2018-03-20 12:32:41 +01:00
@staticmethod
def cache(packet):
2018-04-16 17:13:39 +02:00
if RNS.Transport.shouldCache(packet):
RNS.Transport.cacheRaw(packet.raw)
2018-03-20 12:32:41 +01:00
@staticmethod
def cacheRaw(raw):
try:
file = open(RNS.Reticulum.cachepath+"/"+RNS.hexrep(RNS.Identity.fullHash(raw), delimit=False), "w")
2018-03-20 12:32:41 +01:00
file.write(raw)
file.close()
RNS.log("Wrote packet "+RNS.prettyhexrep(RNS.Identity.fullHash(raw))+" to cache", RNS.LOG_DEBUG)
2018-03-20 12:32:41 +01:00
except Exception as e:
RNS.log("Error writing packet to cache", RNS.LOG_ERROR)
RNS.log("The contained exception was: "+str(e))