2021-08-26 09:26:12 -04:00
|
|
|
import os
|
|
|
|
import RNS
|
|
|
|
import time
|
|
|
|
import threading
|
2021-09-17 13:04:23 -04:00
|
|
|
import subprocess
|
2021-08-26 09:26:12 -04:00
|
|
|
import RNS.vendor.umsgpack as msgpack
|
|
|
|
|
|
|
|
class Node:
|
|
|
|
JOB_INTERVAL = 5
|
2021-12-11 13:20:34 -05:00
|
|
|
START_ANNOUNCE_DELAY = 6
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
def __init__(self, app):
|
|
|
|
RNS.log("Nomad Network Node starting...", RNS.LOG_VERBOSE)
|
|
|
|
self.app = app
|
|
|
|
self.identity = self.app.identity
|
|
|
|
self.destination = RNS.Destination(self.identity, RNS.Destination.IN, RNS.Destination.SINGLE, "nomadnetwork", "node")
|
2021-12-11 13:20:34 -05:00
|
|
|
self.last_announce = time.time()
|
2023-11-29 06:43:49 -05:00
|
|
|
self.last_file_refresh = time.time()
|
|
|
|
self.last_page_refresh = time.time()
|
2021-08-26 09:26:12 -04:00
|
|
|
self.announce_interval = self.app.node_announce_interval
|
2023-11-29 06:43:49 -05:00
|
|
|
self.page_refresh_interval = self.app.page_refresh_interval
|
|
|
|
self.file_refresh_interval = self.app.file_refresh_interval
|
2021-08-26 09:26:12 -04:00
|
|
|
self.job_interval = Node.JOB_INTERVAL
|
|
|
|
self.should_run_jobs = True
|
|
|
|
self.app_data = None
|
|
|
|
self.name = self.app.node_name
|
|
|
|
|
|
|
|
self.register_pages()
|
|
|
|
self.register_files()
|
|
|
|
|
2022-05-17 10:12:00 -04:00
|
|
|
self.destination.set_link_established_callback(self.peer_connected)
|
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
if self.name == None:
|
2021-08-26 10:29:35 -04:00
|
|
|
self.name = self.app.peer_settings["display_name"]+"'s Node"
|
2021-08-26 09:26:12 -04:00
|
|
|
|
2021-08-26 10:29:35 -04:00
|
|
|
RNS.log("Node \""+self.name+"\" ready for incoming connections on "+RNS.prettyhexrep(self.destination.hash), RNS.LOG_VERBOSE)
|
|
|
|
|
|
|
|
if self.app.node_announce_at_start:
|
2021-12-11 13:20:34 -05:00
|
|
|
def delayed_announce():
|
|
|
|
time.sleep(Node.START_ANNOUNCE_DELAY)
|
|
|
|
self.announce()
|
|
|
|
|
|
|
|
da_thread = threading.Thread(target=delayed_announce)
|
|
|
|
da_thread.setDaemon(True)
|
|
|
|
da_thread.start()
|
2021-08-26 09:26:12 -04:00
|
|
|
|
2021-10-09 07:41:48 -04:00
|
|
|
job_thread = threading.Thread(target=self.__jobs)
|
|
|
|
job_thread.setDaemon(True)
|
|
|
|
job_thread.start()
|
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
def register_pages(self):
|
2024-01-07 18:10:55 -05:00
|
|
|
# TODO: Deregister previously registered pages
|
|
|
|
# that no longer exist.
|
2021-08-26 09:26:12 -04:00
|
|
|
self.servedpages = []
|
|
|
|
self.scan_pages(self.app.pagespath)
|
|
|
|
|
|
|
|
if not self.app.pagespath+"index.mu" in self.servedpages:
|
|
|
|
self.destination.register_request_handler(
|
|
|
|
"/page/index.mu",
|
|
|
|
response_generator = self.serve_default_index,
|
|
|
|
allow = RNS.Destination.ALLOW_ALL
|
|
|
|
)
|
|
|
|
|
|
|
|
for page in self.servedpages:
|
|
|
|
request_path = "/page"+page.replace(self.app.pagespath, "")
|
|
|
|
self.destination.register_request_handler(
|
|
|
|
request_path,
|
|
|
|
response_generator = self.serve_page,
|
|
|
|
allow = RNS.Destination.ALLOW_ALL
|
|
|
|
)
|
|
|
|
|
|
|
|
def register_files(self):
|
2024-01-07 18:10:55 -05:00
|
|
|
# TODO: Deregister previously registered files
|
|
|
|
# that no longer exist.
|
2021-08-26 09:26:12 -04:00
|
|
|
self.servedfiles = []
|
|
|
|
self.scan_files(self.app.filespath)
|
|
|
|
|
2021-09-10 15:33:29 -04:00
|
|
|
for file in self.servedfiles:
|
|
|
|
request_path = "/file"+file.replace(self.app.filespath, "")
|
|
|
|
self.destination.register_request_handler(
|
|
|
|
request_path,
|
|
|
|
response_generator = self.serve_file,
|
|
|
|
allow = RNS.Destination.ALLOW_ALL
|
|
|
|
)
|
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
def scan_pages(self, base_path):
|
|
|
|
files = [file for file in os.listdir(base_path) if os.path.isfile(os.path.join(base_path, file)) and file[:1] != "."]
|
|
|
|
directories = [file for file in os.listdir(base_path) if os.path.isdir(os.path.join(base_path, file)) and file[:1] != "."]
|
|
|
|
|
|
|
|
for file in files:
|
2022-04-06 14:34:49 -04:00
|
|
|
if not file.endswith(".allowed"):
|
|
|
|
self.servedpages.append(base_path+"/"+file)
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
for directory in directories:
|
|
|
|
self.scan_pages(base_path+"/"+directory)
|
|
|
|
|
|
|
|
def scan_files(self, base_path):
|
|
|
|
files = [file for file in os.listdir(base_path) if os.path.isfile(os.path.join(base_path, file)) and file[:1] != "."]
|
|
|
|
directories = [file for file in os.listdir(base_path) if os.path.isdir(os.path.join(base_path, file)) and file[:1] != "."]
|
|
|
|
|
|
|
|
for file in files:
|
|
|
|
self.servedfiles.append(base_path+"/"+file)
|
|
|
|
|
|
|
|
for directory in directories:
|
|
|
|
self.scan_files(base_path+"/"+directory)
|
|
|
|
|
2023-02-14 13:46:21 -05:00
|
|
|
def serve_page(self, path, data, request_id, link_id, remote_identity, requested_at):
|
2021-09-10 15:33:29 -04:00
|
|
|
RNS.log("Page request "+RNS.prettyhexrep(request_id)+" for: "+str(path), RNS.LOG_VERBOSE)
|
2022-05-17 10:12:00 -04:00
|
|
|
try:
|
|
|
|
self.app.peer_settings["served_page_requests"] += 1
|
|
|
|
self.app.save_peer_settings()
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Could not increase served page request count", RNS.LOG_ERROR)
|
2022-04-06 14:34:49 -04:00
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
file_path = path.replace("/page", self.app.pagespath, 1)
|
2022-04-06 14:34:49 -04:00
|
|
|
|
|
|
|
allowed_path = file_path+".allowed"
|
|
|
|
request_allowed = False
|
|
|
|
|
|
|
|
if os.path.isfile(allowed_path):
|
|
|
|
allowed_list = []
|
|
|
|
|
|
|
|
try:
|
|
|
|
if os.access(allowed_path, os.X_OK):
|
|
|
|
allowed_result = subprocess.run([allowed_path], stdout=subprocess.PIPE)
|
|
|
|
allowed_input = allowed_result.stdout
|
|
|
|
|
|
|
|
else:
|
|
|
|
fh = open(allowed_path, "rb")
|
|
|
|
allowed_input = fh.read()
|
|
|
|
fh.close()
|
|
|
|
|
|
|
|
allowed_hash_strs = allowed_input.splitlines()
|
|
|
|
|
|
|
|
for hash_str in allowed_hash_strs:
|
|
|
|
if len(hash_str) == RNS.Identity.TRUNCATED_HASHLENGTH//8*2:
|
|
|
|
try:
|
|
|
|
allowed_hash = bytes.fromhex(hash_str.decode("utf-8"))
|
|
|
|
allowed_list.append(allowed_hash)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Could not decode RNS Identity hash from: "+str(hash_str), RNS.LOG_DEBUG)
|
|
|
|
RNS.log("The contained exception was: "+str(e), RNS.LOG_DEBUG)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Error while fetching list of allowed identities for request: "+str(e), RNS.LOG_ERROR)
|
|
|
|
|
2022-04-06 15:41:58 -04:00
|
|
|
if hasattr(remote_identity, "hash") and remote_identity.hash in allowed_list:
|
2022-04-06 14:34:49 -04:00
|
|
|
request_allowed = True
|
|
|
|
else:
|
|
|
|
request_allowed = False
|
|
|
|
RNS.log("Denying request, remote identity was not in list of allowed identities", RNS.LOG_VERBOSE)
|
|
|
|
|
|
|
|
else:
|
|
|
|
request_allowed = True
|
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
try:
|
2022-04-06 14:34:49 -04:00
|
|
|
if request_allowed:
|
|
|
|
RNS.log("Serving page: "+file_path, RNS.LOG_VERBOSE)
|
|
|
|
if os.access(file_path, os.X_OK):
|
2023-02-14 13:46:21 -05:00
|
|
|
env_map = {}
|
2023-02-17 17:04:44 -05:00
|
|
|
if "PATH" in os.environ:
|
|
|
|
env_map["PATH"] = os.environ["PATH"]
|
2023-02-14 13:46:21 -05:00
|
|
|
if link_id != None:
|
|
|
|
env_map["link_id"] = RNS.hexrep(link_id, delimit=False)
|
|
|
|
if remote_identity != None:
|
|
|
|
env_map["remote_identity"] = RNS.hexrep(remote_identity.hash, delimit=False)
|
|
|
|
|
|
|
|
if data != None and isinstance(data, dict):
|
|
|
|
for e in data:
|
2023-02-15 04:09:25 -05:00
|
|
|
if isinstance(e, str) and (e.startswith("field_") or e.startswith("var_")):
|
2023-02-14 13:46:21 -05:00
|
|
|
env_map[e] = data[e]
|
|
|
|
|
|
|
|
generated = subprocess.run([file_path], stdout=subprocess.PIPE, env=env_map)
|
2022-04-06 14:34:49 -04:00
|
|
|
return generated.stdout
|
|
|
|
else:
|
|
|
|
fh = open(file_path, "rb")
|
|
|
|
response_data = fh.read()
|
|
|
|
fh.close()
|
|
|
|
return response_data
|
2021-09-17 13:04:23 -04:00
|
|
|
else:
|
2022-04-06 14:34:49 -04:00
|
|
|
RNS.log("Request denied", RNS.LOG_VERBOSE)
|
|
|
|
return DEFAULT_NOTALLOWED.encode("utf-8")
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Error occurred while handling request "+RNS.prettyhexrep(request_id)+" for: "+str(path), RNS.LOG_ERROR)
|
|
|
|
RNS.log("The contained exception was: "+str(e), RNS.LOG_ERROR)
|
|
|
|
return None
|
|
|
|
|
2021-09-10 15:33:29 -04:00
|
|
|
# TODO: Improve file handling, this will be slow for large files
|
|
|
|
def serve_file(self, path, data, request_id, remote_identity, requested_at):
|
|
|
|
RNS.log("File request "+RNS.prettyhexrep(request_id)+" for: "+str(path), RNS.LOG_VERBOSE)
|
2022-05-17 10:12:00 -04:00
|
|
|
try:
|
|
|
|
self.app.peer_settings["served_file_requests"] += 1
|
|
|
|
self.app.save_peer_settings()
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Could not increase served file request count", RNS.LOG_ERROR)
|
|
|
|
|
2021-09-10 15:33:29 -04:00
|
|
|
file_path = path.replace("/file", self.app.filespath, 1)
|
|
|
|
file_name = path.replace("/file/", "", 1)
|
|
|
|
try:
|
|
|
|
RNS.log("Serving file: "+file_path, RNS.LOG_VERBOSE)
|
|
|
|
fh = open(file_path, "rb")
|
|
|
|
file_data = fh.read()
|
|
|
|
fh.close()
|
|
|
|
return [file_name, file_data]
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Error occurred while handling request "+RNS.prettyhexrep(request_id)+" for: "+str(path), RNS.LOG_ERROR)
|
|
|
|
RNS.log("The contained exception was: "+str(e), RNS.LOG_ERROR)
|
|
|
|
return None
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
def serve_default_index(self, path, data, request_id, remote_identity, requested_at):
|
|
|
|
RNS.log("Serving default index for request "+RNS.prettyhexrep(request_id)+" for: "+str(path), RNS.LOG_VERBOSE)
|
|
|
|
return DEFAULT_INDEX.encode("utf-8")
|
|
|
|
|
|
|
|
def announce(self):
|
|
|
|
self.app_data = self.name.encode("utf-8")
|
|
|
|
self.last_announce = time.time()
|
2021-09-16 14:44:15 -04:00
|
|
|
self.app.peer_settings["node_last_announce"] = self.last_announce
|
2021-08-26 09:26:12 -04:00
|
|
|
self.destination.announce(app_data=self.app_data)
|
2021-10-03 15:06:56 -04:00
|
|
|
self.app.message_router.announce_propagation_node()
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
def __jobs(self):
|
|
|
|
while self.should_run_jobs:
|
|
|
|
now = time.time()
|
|
|
|
|
2021-10-10 07:50:57 -04:00
|
|
|
if now > self.last_announce + self.announce_interval*60:
|
2021-08-26 09:26:12 -04:00
|
|
|
self.announce()
|
2023-11-29 06:43:49 -05:00
|
|
|
|
|
|
|
if self.page_refresh_interval > 0:
|
|
|
|
if now > self.last_page_refresh + self.page_refresh_interval*60:
|
|
|
|
self.register_pages()
|
2024-01-07 18:10:55 -05:00
|
|
|
self.last_page_refresh = time.time()
|
|
|
|
|
2023-11-29 06:43:49 -05:00
|
|
|
if self.file_refresh_interval > 0:
|
|
|
|
if now > self.last_file_refresh + self.file_refresh_interval*60:
|
|
|
|
self.register_files()
|
2024-01-07 18:10:55 -05:00
|
|
|
self.last_file_refresh = time.time()
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
time.sleep(self.job_interval)
|
|
|
|
|
2022-05-17 10:12:00 -04:00
|
|
|
def peer_connected(self, link):
|
|
|
|
RNS.log("Peer connected to "+str(self.destination), RNS.LOG_VERBOSE)
|
|
|
|
try:
|
|
|
|
self.app.peer_settings["node_connects"] += 1
|
|
|
|
self.app.save_peer_settings()
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
RNS.log("Could not increase node connection count", RNS.LOG_ERROR)
|
|
|
|
|
2021-08-26 09:26:12 -04:00
|
|
|
link.set_link_closed_callback(self.peer_disconnected)
|
|
|
|
|
2022-05-17 10:12:00 -04:00
|
|
|
def peer_disconnected(self, link):
|
|
|
|
RNS.log("Peer disconnected from "+str(self.destination), RNS.LOG_VERBOSE)
|
|
|
|
pass
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
DEFAULT_INDEX = '''>Default Home Page
|
|
|
|
|
2021-08-27 13:58:14 -04:00
|
|
|
This node is serving pages, but the home page file (index.mu) was not found in the page storage directory. This is an auto-generated placeholder.
|
2021-08-26 09:26:12 -04:00
|
|
|
|
|
|
|
If you are the node operator, you can define your own home page by creating a file named `*index.mu`* in the page storage directory.
|
2022-04-06 14:34:49 -04:00
|
|
|
'''
|
|
|
|
|
|
|
|
DEFAULT_NOTALLOWED = '''>Request Not Allowed
|
|
|
|
|
|
|
|
You are not authorised to carry out the request.
|
2023-02-17 17:04:44 -05:00
|
|
|
'''
|