2021-05-13 10:38:22 -04:00
|
|
|
import os
|
|
|
|
import RNS
|
|
|
|
import LXMF
|
2021-05-15 15:08:30 -04:00
|
|
|
import time
|
2021-08-26 11:17:40 -04:00
|
|
|
import nomadnet
|
2021-05-13 10:38:22 -04:00
|
|
|
import RNS.vendor.umsgpack as msgpack
|
|
|
|
|
|
|
|
class Directory:
|
2021-08-26 09:26:12 -04:00
|
|
|
ANNOUNCE_STREAM_MAXLENGTH = 64
|
2021-05-15 15:08:30 -04:00
|
|
|
|
2021-08-26 10:17:01 -04:00
|
|
|
aspect_filter = "nomadnetwork.node"
|
|
|
|
@staticmethod
|
|
|
|
def received_announce(destination_hash, announced_identity, app_data):
|
|
|
|
app = nomadnet.NomadNetworkApp.get_shared_instance()
|
|
|
|
destination_hash_text = RNS.hexrep(destination_hash, delimit=False)
|
|
|
|
|
2021-08-26 11:17:40 -04:00
|
|
|
associated_peer = RNS.Destination.hash_from_name_and_identity("lxmf.delivery", announced_identity)
|
|
|
|
|
|
|
|
app.directory.node_announce_received(destination_hash, app_data, associated_peer)
|
2021-10-03 12:44:00 -04:00
|
|
|
app.autoselect_propagation_node()
|
2021-08-26 10:17:01 -04:00
|
|
|
|
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
def __init__(self, app):
|
|
|
|
self.directory_entries = {}
|
2021-05-15 15:08:30 -04:00
|
|
|
self.announce_stream = []
|
2021-05-13 10:38:22 -04:00
|
|
|
self.app = app
|
|
|
|
self.load_from_disk()
|
|
|
|
|
2021-08-26 10:17:01 -04:00
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
def save_to_disk(self):
|
|
|
|
try:
|
|
|
|
packed_list = []
|
|
|
|
for source_hash in self.directory_entries:
|
|
|
|
e = self.directory_entries[source_hash]
|
2021-10-03 12:44:00 -04:00
|
|
|
packed_list.append((e.source_hash, e.display_name, e.trust_level, e.hosts_node, e.preferred_delivery))
|
2021-05-15 15:08:30 -04:00
|
|
|
|
|
|
|
directory = {
|
|
|
|
"entry_list": packed_list,
|
|
|
|
"announce_stream": self.announce_stream
|
|
|
|
}
|
2021-05-13 10:38:22 -04:00
|
|
|
|
|
|
|
file = open(self.app.directorypath, "wb")
|
2021-05-15 15:08:30 -04:00
|
|
|
file.write(msgpack.packb(directory))
|
2021-05-13 10:38:22 -04:00
|
|
|
file.close()
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Could not write directory to disk. Then contained exception was: "+str(e), RNS.LOG_ERROR)
|
|
|
|
|
|
|
|
def load_from_disk(self):
|
|
|
|
if os.path.isfile(self.app.directorypath):
|
|
|
|
try:
|
|
|
|
file = open(self.app.directorypath, "rb")
|
2021-05-15 15:08:30 -04:00
|
|
|
unpacked_directory = msgpack.unpackb(file.read())
|
|
|
|
unpacked_list = unpacked_directory["entry_list"]
|
2021-05-13 10:38:22 -04:00
|
|
|
file.close()
|
|
|
|
|
|
|
|
entries = {}
|
|
|
|
for e in unpacked_list:
|
2021-05-15 15:08:30 -04:00
|
|
|
if len(e) > 3:
|
|
|
|
hosts_node = e[3]
|
|
|
|
else:
|
|
|
|
hosts_node = False
|
|
|
|
|
2021-10-03 12:44:00 -04:00
|
|
|
if len(e) > 4:
|
|
|
|
preferred_delivery = e[4]
|
|
|
|
else:
|
|
|
|
preferred_delivery = None
|
|
|
|
|
|
|
|
entries[e[0]] = DirectoryEntry(e[0], e[1], e[2], hosts_node, preferred_delivery=preferred_delivery)
|
2021-05-13 10:38:22 -04:00
|
|
|
|
|
|
|
self.directory_entries = entries
|
2021-08-26 10:17:01 -04:00
|
|
|
|
2021-08-26 11:17:40 -04:00
|
|
|
self.announce_stream = unpacked_directory["announce_stream"]
|
2021-08-26 10:17:01 -04:00
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Could not load directory from disk. The contained exception was: "+str(e), RNS.LOG_ERROR)
|
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
def lxmf_announce_received(self, source_hash, app_data):
|
2021-08-27 13:58:14 -04:00
|
|
|
if app_data != None:
|
|
|
|
timestamp = time.time()
|
|
|
|
self.announce_stream.insert(0, (timestamp, source_hash, app_data, False))
|
|
|
|
while len(self.announce_stream) > Directory.ANNOUNCE_STREAM_MAXLENGTH:
|
|
|
|
self.announce_stream.pop()
|
|
|
|
self.app.ui.main_display.sub_displays.network_display.directory_change_callback()
|
2021-08-26 09:26:12 -04:00
|
|
|
|
2021-08-26 11:17:40 -04:00
|
|
|
def node_announce_received(self, source_hash, app_data, associated_peer):
|
2021-08-27 13:58:14 -04:00
|
|
|
if app_data != None:
|
|
|
|
timestamp = time.time()
|
|
|
|
self.announce_stream.insert(0, (timestamp, source_hash, app_data, True))
|
|
|
|
while len(self.announce_stream) > Directory.ANNOUNCE_STREAM_MAXLENGTH:
|
|
|
|
self.announce_stream.pop()
|
|
|
|
|
|
|
|
if self.trust_level(associated_peer) == DirectoryEntry.TRUSTED:
|
|
|
|
node_entry = DirectoryEntry(source_hash, display_name=app_data.decode("utf-8"), trust_level=DirectoryEntry.TRUSTED, hosts_node=True)
|
|
|
|
self.remember(node_entry)
|
2021-09-25 11:19:26 -04:00
|
|
|
|
|
|
|
self.app.ui.main_display.sub_displays.network_display.directory_change_callback()
|
2021-08-26 15:37:27 -04:00
|
|
|
|
|
|
|
def remove_announce_with_timestamp(self, timestamp):
|
|
|
|
selected_announce = None
|
|
|
|
for announce in self.announce_stream:
|
|
|
|
if announce[0] == timestamp:
|
|
|
|
selected_announce = announce
|
|
|
|
|
|
|
|
if selected_announce != None:
|
|
|
|
self.announce_stream.remove(selected_announce)
|
2021-08-26 11:17:40 -04:00
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
def display_name(self, source_hash):
|
|
|
|
if source_hash in self.directory_entries:
|
|
|
|
return self.directory_entries[source_hash].display_name
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def simplest_display_str(self, source_hash):
|
|
|
|
trust_level = self.trust_level(source_hash)
|
|
|
|
if trust_level == DirectoryEntry.WARNING or trust_level == DirectoryEntry.UNTRUSTED:
|
|
|
|
return "<"+RNS.hexrep(source_hash, delimit=False)+">"
|
|
|
|
else:
|
|
|
|
if source_hash in self.directory_entries:
|
|
|
|
return self.directory_entries[source_hash].display_name
|
|
|
|
else:
|
|
|
|
return "<"+RNS.hexrep(source_hash, delimit=False)+">"
|
|
|
|
|
|
|
|
def trust_level(self, source_hash, announced_display_name=None):
|
|
|
|
if source_hash in self.directory_entries:
|
|
|
|
if announced_display_name == None:
|
|
|
|
return self.directory_entries[source_hash].trust_level
|
|
|
|
else:
|
|
|
|
for entry in self.directory_entries:
|
|
|
|
e = self.directory_entries[entry]
|
|
|
|
if e.display_name == announced_display_name:
|
|
|
|
if e.source_hash != source_hash:
|
|
|
|
return DirectoryEntry.WARNING
|
|
|
|
|
|
|
|
return self.directory_entries[source_hash].trust_level
|
|
|
|
else:
|
|
|
|
return DirectoryEntry.UNKNOWN
|
|
|
|
|
2021-10-03 12:44:00 -04:00
|
|
|
def preferred_delivery(self, source_hash):
|
|
|
|
if source_hash in self.directory_entries:
|
|
|
|
return self.directory_entries[source_hash].preferred_delivery
|
|
|
|
else:
|
|
|
|
return DirectoryEntry.DIRECT
|
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
def remember(self, entry):
|
|
|
|
self.directory_entries[entry.source_hash] = entry
|
|
|
|
|
2021-08-26 15:37:27 -04:00
|
|
|
identity = RNS.Identity.recall(entry.source_hash)
|
2021-10-08 03:32:45 -04:00
|
|
|
if identity != None:
|
|
|
|
associated_node = RNS.Destination.hash_from_name_and_identity("nomadnetwork.node", identity)
|
|
|
|
if associated_node in self.directory_entries:
|
|
|
|
node_entry = self.directory_entries[associated_node]
|
|
|
|
node_entry.trust_level = entry.trust_level
|
2021-08-26 15:37:27 -04:00
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
def forget(self, source_hash):
|
|
|
|
if source_hash in self.directory_entries:
|
|
|
|
self.directory_entries.pop(source_hash)
|
|
|
|
|
|
|
|
def find(self, source_hash):
|
|
|
|
if source_hash in self.directory_entries:
|
|
|
|
return self.directory_entries[source_hash]
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
2021-05-14 16:12:42 -04:00
|
|
|
def is_known(self, source_hash):
|
|
|
|
try:
|
|
|
|
self.source_identity = RNS.Identity.recall(source_hash)
|
|
|
|
|
|
|
|
if self.source_identity:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
return False
|
|
|
|
|
2021-05-15 15:08:30 -04:00
|
|
|
def known_nodes(self):
|
|
|
|
node_list = []
|
|
|
|
for eh in self.directory_entries:
|
|
|
|
e = self.directory_entries[eh]
|
|
|
|
if e.hosts_node:
|
|
|
|
node_list.append(e)
|
|
|
|
|
|
|
|
return node_list
|
|
|
|
|
|
|
|
def number_of_known_nodes(self):
|
|
|
|
return len(self.known_nodes())
|
|
|
|
|
|
|
|
def number_of_known_peers(self, lookback_seconds=None):
|
|
|
|
unique_hashes = []
|
|
|
|
cutoff_time = time.time()-lookback_seconds
|
|
|
|
for entry in self.announce_stream:
|
|
|
|
if not entry[1] in unique_hashes:
|
|
|
|
if lookback_seconds == None or entry[0] > cutoff_time:
|
|
|
|
unique_hashes.append(entry[1])
|
|
|
|
|
|
|
|
return len(unique_hashes)
|
2021-05-13 10:38:22 -04:00
|
|
|
|
|
|
|
class DirectoryEntry:
|
|
|
|
WARNING = 0x00
|
|
|
|
UNTRUSTED = 0x01
|
|
|
|
UNKNOWN = 0x02
|
|
|
|
TRUSTED = 0xFF
|
|
|
|
|
2021-10-03 12:44:00 -04:00
|
|
|
DIRECT = 0x01
|
|
|
|
PROPAGATED = 0x02
|
|
|
|
|
|
|
|
def __init__(self, source_hash, display_name=None, trust_level=UNKNOWN, hosts_node=False, preferred_delivery=None):
|
2021-05-13 10:38:22 -04:00
|
|
|
if len(source_hash) == RNS.Identity.TRUNCATED_HASHLENGTH//8:
|
|
|
|
self.source_hash = source_hash
|
|
|
|
self.display_name = display_name
|
|
|
|
if display_name == None:
|
|
|
|
display_name = source_hash
|
|
|
|
|
2021-10-03 12:44:00 -04:00
|
|
|
if preferred_delivery == None:
|
|
|
|
self.preferred_delivery = DirectoryEntry.DIRECT
|
|
|
|
else:
|
|
|
|
self.preferred_delivery = preferred_delivery
|
|
|
|
|
2021-05-13 10:38:22 -04:00
|
|
|
self.trust_level = trust_level
|
2021-05-15 15:08:30 -04:00
|
|
|
self.hosts_node = hosts_node
|
2021-05-13 10:38:22 -04:00
|
|
|
else:
|
|
|
|
raise TypeError("Attempt to add invalid source hash to directory")
|