from wrapper.toxcore_enums_and_consts import TOX_FILE_KIND, TOX_FILE_CONTROL from os.path import basename, getsize, exists, dirname from os import remove, rename, chdir from time import time from wrapper.tox import Tox from common.event import Event from middleware.threads import invoke_in_main_thread FILE_TRANSFER_STATE = { 'RUNNING': 0, 'PAUSED_BY_USER': 1, 'CANCELLED': 2, 'FINISHED': 3, 'PAUSED_BY_FRIEND': 4, 'INCOMING_NOT_STARTED': 5, 'OUTGOING_NOT_STARTED': 6, 'UNSENT': 7 } ACTIVE_FILE_TRANSFERS = (0, 1, 4, 5, 6) PAUSED_FILE_TRANSFERS = (1, 4, 5, 6) DO_NOT_SHOW_ACCEPT_BUTTON = (2, 3, 4, 6) SHOW_PROGRESS_BAR = (0, 1, 4) def is_inline(file_name): allowed_inlines = ('toxygen_inline.png', 'utox-inline.png', 'sticker.png') return file_name in allowed_inlines or file_name.startswith('qTox_Image_') class FileTransfer: """ Superclass for file transfers """ def __init__(self, path, tox, friend_number, size, file_number=None): self._path = path self._tox = tox self._friend_number = friend_number self._state = FILE_TRANSFER_STATE['RUNNING'] self._file_number = file_number self._creation_time = None self._size = float(size) self._done = 0 self._state_changed_event = Event() self._finished_event = Event() self._file_id = self._file = None def set_state_changed_handler(self, handler): self._state_changed_event += lambda *args: invoke_in_main_thread(handler, *args) def set_transfer_finished_handler(self, handler): self._finished_event += lambda *args: invoke_in_main_thread(handler, *args) def get_file_number(self): return self._file_number file_number = property(get_file_number) def get_state(self): return self._state def set_state(self, value): self._state = value self._signal() state = property(get_state, set_state) def get_friend_number(self): return self._friend_number friend_number = property(get_friend_number) def get_file_id(self): return self._file_id file_id = property(get_file_id) def get_path(self): return self._path path = property(get_path) def get_size(self): return self._size size = property(get_size) def cancel(self): self.send_control(TOX_FILE_CONTROL['CANCEL']) if self._file is not None: self._file.close() self._signal() def cancelled(self): if self._file is not None: self._file.close() self.set_state(FILE_TRANSFER_STATE['CANCELLED']) def pause(self, by_friend): if not by_friend: self.send_control(TOX_FILE_CONTROL['PAUSE']) else: self.set_state(FILE_TRANSFER_STATE['PAUSED_BY_FRIEND']) def send_control(self, control): if self._tox.file_control(self._friend_number, self._file_number, control): self.set_state(control) def get_file_id(self): return self._tox.file_get_file_id(self._friend_number, self._file_number) def _signal(self): percentage = self._done / self._size if self._size else 0 if self._creation_time is None or not percentage: t = -1 else: t = ((time() - self._creation_time) / percentage) * (1 - percentage) self._state_changed_event(self.state, percentage, int(t)) def _finished(self): self._finished_event(self._friend_number, self._file_number) # ----------------------------------------------------------------------------------------------------------------- # Send file # ----------------------------------------------------------------------------------------------------------------- class SendTransfer(FileTransfer): def __init__(self, path, tox, friend_number, kind=TOX_FILE_KIND['DATA'], file_id=None): if path is not None: fl = open(path, 'rb') size = getsize(path) else: fl = None size = 0 super().__init__(path, tox, friend_number, size) self._file = fl self.state = FILE_TRANSFER_STATE['OUTGOING_NOT_STARTED'] self._file_number = tox.file_send(friend_number, kind, size, file_id, bytes(basename(path), 'utf-8') if path else b'') self._file_id = self.get_file_id() def send_chunk(self, position, size): """ Send chunk :param position: start position in file :param size: chunk max size """ if self._creation_time is None: self._creation_time = time() if size: self._file.seek(position) data = self._file.read(size) self._tox.file_send_chunk(self._friend_number, self._file_number, position, data) self._done += size self._signal() else: if self._file is not None: self._file.close() self.state = FILE_TRANSFER_STATE['FINISHED'] self._finished() class SendAvatar(SendTransfer): """ Send avatar to friend. Doesn't need file transfer item """ def __init__(self, path, tox, friend_number): if path is None: avatar_hash = None else: with open(path, 'rb') as fl: avatar_hash = Tox.hash(fl.read()) super().__init__(path, tox, friend_number, TOX_FILE_KIND['AVATAR'], avatar_hash) class SendFromBuffer(FileTransfer): """ Send inline image """ def __init__(self, tox, friend_number, data, file_name): super().__init__(None, tox, friend_number, len(data)) self.state = FILE_TRANSFER_STATE['OUTGOING_NOT_STARTED'] self._data = data self._file_number = tox.file_send(friend_number, TOX_FILE_KIND['DATA'], len(data), None, bytes(file_name, 'utf-8')) def get_data(self): return self._data data = property(get_data) def send_chunk(self, position, size): if self._creation_time is None: self._creation_time = time() if size: data = self._data[position:position + size] self._tox.file_send_chunk(self._friend_number, self._file_number, position, data) self._done += size else: self.state = FILE_TRANSFER_STATE['FINISHED'] self._finished() self._signal() class SendFromFileBuffer(SendTransfer): def __init__(self, *args): super().__init__(*args) def send_chunk(self, position, size): super().send_chunk(position, size) if not size: chdir(dirname(self._path)) remove(self._path) # ----------------------------------------------------------------------------------------------------------------- # Receive file # ----------------------------------------------------------------------------------------------------------------- class ReceiveTransfer(FileTransfer): def __init__(self, path, tox, friend_number, size, file_number, position=0): super().__init__(path, tox, friend_number, size, file_number) self._file = open(self._path, 'wb') self._file_size = position self._file.truncate(position) self._missed = set() self._file_id = self.get_file_id() self._done = position def cancel(self): super().cancel() remove(self._path) def total_size(self): self._missed.add(self._file_size) return min(self._missed) def write_chunk(self, position, data): """ Incoming chunk :param position: position in file to save data :param data: raw data (string) """ if self._creation_time is None: self._creation_time = time() if data is None: self._file.close() self.state = FILE_TRANSFER_STATE['FINISHED'] self._finished() else: data = bytearray(data) if self._file_size < position: self._file.seek(0, 2) self._file.write(b'\0' * (position - self._file_size)) self._missed.add(self._file_size) else: self._missed.discard(position) self._file.seek(position) self._file.write(data) l = len(data) if position + l > self._file_size: self._file_size = position + l self._done += l self._signal() class ReceiveToBuffer(FileTransfer): """ Inline image - save in buffer not in file system """ def __init__(self, tox, friend_number, size, file_number): super().__init__(None, tox, friend_number, size, file_number) self._data = bytes() self._data_size = 0 def get_data(self): return self._data data = property(get_data) def write_chunk(self, position, data): if self._creation_time is None: self._creation_time = time() if data is None: self.state = FILE_TRANSFER_STATE['FINISHED'] self._finished() else: data = bytes(data) l = len(data) if self._data_size < position: self._data += (b'\0' * (position - self._data_size)) self._data = self._data[:position] + data + self._data[position + l:] if position + l > self._data_size: self._data_size = position + l self._done += l self._signal() class ReceiveAvatar(ReceiveTransfer): """ Get friend's avatar. Doesn't need file transfer item """ MAX_AVATAR_SIZE = 512 * 1024 def __init__(self, path, tox, friend_number, size, file_number): full_path = path + '.tmp' super().__init__(full_path, tox, friend_number, size, file_number) if size > self.MAX_AVATAR_SIZE: self.send_control(TOX_FILE_CONTROL['CANCEL']) self._file.close() remove(full_path) elif not size: self.send_control(TOX_FILE_CONTROL['CANCEL']) self._file.close() remove(full_path) elif exists(path): hash = self.get_file_id() with open(path, 'rb') as fl: data = fl.read() existing_hash = Tox.hash(data) if hash == existing_hash: self.send_control(TOX_FILE_CONTROL['CANCEL']) self._file.close() remove(full_path) else: self.send_control(TOX_FILE_CONTROL['RESUME']) else: self.send_control(TOX_FILE_CONTROL['RESUME']) def write_chunk(self, position, data): if data is None: avatar_path = self._path[:-4] if exists(avatar_path): chdir(dirname(avatar_path)) remove(avatar_path) rename(self._path, avatar_path) super().write_chunk(position, data)