Added libs

This commit is contained in:
Lucas
2026-01-25 13:55:46 +10:00
parent 575c682afc
commit f70af3c4ea
229 changed files with 26983 additions and 0 deletions

View File

@@ -0,0 +1 @@
{"file.py": 1757577571.3943994, "fileTransfer.py": 1756117680.0193572}

109
fileTransfer/file.py Normal file
View File

@@ -0,0 +1,109 @@
import sys, os
import math
from enum import Enum, auto
from libs.app.common.logging import get_logger
from libs.fspn.utils.observable import Observable
from libs.fspn.utils.wrapper_util import threaded
from libs.fspn.utils.sha256_util import hash_bytes, hash_file
logger = get_logger()
class FileEvents(Enum):
ON_FILE_COMPLETED = auto()
ON_FILE_ERROR = auto()
ON_FILE_UPDATE = auto()
ON_FILE_APPROVED = auto()
# TODO Add try except and retries
class File(Observable):
def __init__(self, folder, name, size, chunk_size, hash, connection_id, sending, file_transfer, to_module) -> None:
self.id = (hash, connection_id)
self.name = name
self.folder = folder
self.size = size
self.chunk_size = chunk_size
self.parts = [None] * math.ceil(size/chunk_size)
self.hash = hash
self.status = "WAITING"
self.sending = sending
self.connection_id = connection_id
from libs.fileTransfer.fileTransfer import FileTransfer
self.file_transfer:FileTransfer = file_transfer
self.to_module = to_module
self.output_path = os.path.join(self.folder, self.name + '.download')
self.final_path = self.get_final_path()
def get_final_path(self):
base_name, ext = os.path.splitext(self.name)
count = 0
while True:
if count == 0:
filename = f"{base_name}{ext}"
else:
filename = f"{base_name}({count}){ext}"
final_path = os.path.join(self.folder, filename)
if not os.path.exists(final_path):
return final_path
count += 1
def approve_transfer(self, approved):
if approved:
self.status = "TRANSFERING"
if self.sending:
self.start_send()
else:
# TODO Check if file exists and same hash
if not os.path.exists(self.output_path):
os.makedirs(self.folder, exist_ok=True)
with open(self.output_path, 'wb') as f:
f.truncate(self.size)
else:
self.status = "CANCELED"
self.fire_event(FileEvents.ON_FILE_APPROVED.name, approved=approved)
@threaded
def start_send(self):
logger.warning("Start Sending")
f = open(os.path.join(self.folder, self.name), 'rb')
for part in range(len(self.parts)):
f.seek(part * self.chunk_size)
data = f.read(self.chunk_size)
part_hash = hash_bytes(data)
logger.debug(f"Sending part {part}")
# time.sleep(0.5)
self.file_transfer.send_file_part(self, part, part_hash, data)
f.close()
@threaded
def write_part(self, data, part, hash):
check_hash = hash_bytes(data)
logger.debug(f"Writing part {part}")
if check_hash == hash:
part_offset = part * self.chunk_size
with open(self.output_path, 'r+b') as f:
f.seek(part_offset)
f.write(data)
self.parts[part] = True
# TODO Send ack
self.update_status()
else:
# TODO Request part again
logger.error("HASH PART ERROR")
def update_status(self):
if all(x for x in self.parts):
os.rename(self.output_path, self.final_path)
if hash_file(self.final_path) == self.hash:
logger.info(f'File {self.name} downloaded!')
self.status = "COMPLETED"
self.fire_event(FileEvents.ON_FILE_COMPLETED.name, final_path=self.final_path)
else:
self.status = "CORRUPTED"
logger.warning(f'File {self.name} corrupted')
self.fire_event(FileEvents.ON_FILE_ERROR.name)
else:
self.fire_event(FileEvents.ON_FILE_UPDATE.name)

View File

@@ -0,0 +1,100 @@
import sys, os
import math
from enum import Enum, auto
from pathlib import Path
import time
from libs.noSys.noSysModule import NoSysModule
from libs.app.common.logging import get_logger
from libs.app.common.paths import ROOT_DIR
from libs.fspn.utils.observable import Observable
from libs.fspn.utils.wrapper_util import threaded
from libs.fspn.utils.sha256_util import hash_bytes, hash_file
from .file import File, FileEvents
logger = get_logger()
class FileTransferEvents(Enum):
ON_RECEIVING_ = auto()
class FileTransfer(NoSysModule):
def __init__(self, nosys_core):
super().__init__(nosys_core)
self.files:dict[tuple[str,str], File] = {}
def send_file(self, file_path, connection_id, to_module):
path = Path(file_path)
file_folder = path.parent
file_name = path.name
file_size = path.stat().st_size
chunk_size = self.get_dynamic_chunk_size(file_size)
file_hash = hash_file(file_path)
file = File(file_folder, file_name, file_size, chunk_size, file_hash, connection_id, True, self, to_module)
file.subscribe_event(FileEvents.ON_FILE_APPROVED.name, self.on_file_approved)
self.files[file.id] = file
self.send_file_info(connection_id, file_name, file_size, chunk_size, file_hash, to_module)
def get_dynamic_chunk_size(self, file_size: int) -> int:
min_chunk = 64 * 1024 # 64 KB
max_chunk = 2 * 1024 * 1024 # 2 MB
target_chunks = 500
ideal_chunk = file_size // target_chunks
return max(min_chunk, min(ideal_chunk, max_chunk))
def on_module_message(self, event):
if event.meta:
action = event.meta["action"]
else:
action = event.data['action']
handler_action = getattr(self, 'on_'+action)
handler_action(event)
def send_file_info(self, connection_id, name, size, chunk_size, hash, to_module):
body = {"action":"file_info", "name":name, "size":size, "chunk_size":chunk_size, "hash":hash, "module":{"package": to_module[0], "name":to_module[1]}}
self.nosys_core.dispatcher.send_message(body, connection_id, self.id)
def on_file_info(self, event):
payload = event.data
folder = os.path.join(ROOT_DIR, "files")
to_module = (payload['module']['package'], payload['module']['name'])
file = File(folder, payload["name"], payload["size"], payload["chunk_size"], payload["hash"], event.connection.id, False, self, to_module)
file.subscribe_event(FileEvents.ON_FILE_APPROVED.name, self.on_file_approved)
self.files[file.id] = file
self.fire_event(f"{FileTransferEvents.ON_RECEIVING_.name}{to_module[0]}_{to_module[1]}", file=file)
def subscribe_module_file_events(self, module:tuple[str, str], callback):
self.subscribe_event(f"{FileTransferEvents.ON_RECEIVING_.name}{module[0]}_{module[1]}", callback)
def on_file_approved(self, event):
file:File = event.source
approved = event.approved
if file.sending:
logger.debug(f"Peer file approved {approved}")
else:
self.send_file_transfer_approved(file, approved)
def send_file_transfer_approved(self, file:File, approved):
body = {"action":"file_transfer_approved", "file_hash":file.hash, "approved":approved}
self.nosys_core.dispatcher.send_message(body, file.connection_id, self.id)
def on_file_transfer_approved(self, event):
payload = event.data
file_id = (payload["file_hash"], event.connection.id)
file:File = self.files[file_id]
file.approve_transfer(payload["approved"])
def send_file_part(self, file:File, part, part_hash, data):
meta = {"action":"file_part", "part":part, "part_hash":part_hash, "file_hash": file.hash}
self.nosys_core.dispatcher.send_binary(data, file.connection_id, self.id, meta)
def on_file_part(self, event):
meta = event.meta
data = event.data
file_id = (meta["file_hash"], event.connection.id)
file:File = self.files[file_id]
file.write_part(data, meta["part"], meta["part_hash"])
# TODO get_file_part, ack_file_part

Binary file not shown.

10
fileTransfer/info.json Normal file
View File

@@ -0,0 +1,10 @@
{
"id": "fileTransfer",
"version": 0.007,
"modules": [
{
"id": "fileTransfer",
"version": 0
}
]
}