2018-01-26 20:21:46 +00:00
|
|
|
from wrapper.toxcore_enums_and_consts import TOX_FILE_KIND, TOX_FILE_CONTROL
|
2016-06-11 21:40:58 +00:00
|
|
|
from os.path import basename, getsize, exists, dirname
|
|
|
|
from os import remove, rename, chdir
|
2018-05-16 21:02:22 +00:00
|
|
|
from time import time
|
2018-01-26 20:21:46 +00:00
|
|
|
from wrapper.tox import Tox
|
2018-05-16 16:04:02 +00:00
|
|
|
from common.event import Event
|
2016-03-16 20:11:18 +00:00
|
|
|
|
|
|
|
|
2018-05-16 16:04:02 +00:00
|
|
|
FILE_TRANSFER_STATE = {
|
2016-03-17 17:46:18 +00:00
|
|
|
'RUNNING': 0,
|
2016-06-16 21:10:26 +00:00
|
|
|
'PAUSED_BY_USER': 1,
|
|
|
|
'CANCELLED': 2,
|
2016-03-17 17:46:18 +00:00
|
|
|
'FINISHED': 3,
|
2016-06-16 21:10:26 +00:00
|
|
|
'PAUSED_BY_FRIEND': 4,
|
2016-06-17 10:31:48 +00:00
|
|
|
'INCOMING_NOT_STARTED': 5,
|
2018-05-18 16:40:34 +00:00
|
|
|
'OUTGOING_NOT_STARTED': 6,
|
|
|
|
'UNSENT': 7
|
2016-03-17 17:46:18 +00:00
|
|
|
}
|
|
|
|
|
2016-06-17 10:31:48 +00:00
|
|
|
ACTIVE_FILE_TRANSFERS = (0, 1, 4, 5, 6)
|
2016-06-16 21:10:26 +00:00
|
|
|
|
2016-06-17 10:31:48 +00:00
|
|
|
PAUSED_FILE_TRANSFERS = (1, 4, 5, 6)
|
2016-06-16 21:10:26 +00:00
|
|
|
|
2016-06-17 10:31:48 +00:00
|
|
|
DO_NOT_SHOW_ACCEPT_BUTTON = (2, 3, 4, 6)
|
2016-06-16 21:10:26 +00:00
|
|
|
|
|
|
|
SHOW_PROGRESS_BAR = (0, 1, 4)
|
|
|
|
|
2016-06-08 19:53:41 +00:00
|
|
|
|
2016-12-24 19:05:29 +00:00
|
|
|
def is_inline(file_name):
|
2018-05-16 11:47:14 +00:00
|
|
|
allowed_inlines = ('toxygen_inline.png', 'utox-inline.png', 'sticker.png')
|
|
|
|
|
|
|
|
return file_name in allowed_inlines or file_name.startswith('qTox_Screenshot_')
|
|
|
|
|
2016-12-24 19:05:29 +00:00
|
|
|
|
2018-05-16 16:04:02 +00:00
|
|
|
class FileTransfer:
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Superclass for file transfers
|
|
|
|
"""
|
2016-03-19 11:41:01 +00:00
|
|
|
|
2016-03-21 17:19:13 +00:00
|
|
|
def __init__(self, path, tox, friend_number, size, file_number=None):
|
2016-03-16 20:11:18 +00:00
|
|
|
self._path = path
|
|
|
|
self._tox = tox
|
|
|
|
self._friend_number = friend_number
|
2018-05-17 12:20:47 +00:00
|
|
|
self._state = FILE_TRANSFER_STATE['RUNNING']
|
2016-03-17 17:46:18 +00:00
|
|
|
self._file_number = file_number
|
2016-06-17 10:31:48 +00:00
|
|
|
self._creation_time = None
|
2016-03-21 18:53:02 +00:00
|
|
|
self._size = float(size)
|
2016-03-21 17:19:13 +00:00
|
|
|
self._done = 0
|
2018-05-16 16:04:02 +00:00
|
|
|
self._state_changed_event = Event()
|
|
|
|
self._finished_event = Event()
|
2018-05-16 21:02:22 +00:00
|
|
|
self._file_id = self._file = None
|
2016-03-16 20:11:18 +00:00
|
|
|
|
2016-03-21 17:19:13 +00:00
|
|
|
def set_state_changed_handler(self, handler):
|
2018-05-16 16:04:02 +00:00
|
|
|
self._state_changed_event += handler
|
2016-03-19 11:41:01 +00:00
|
|
|
|
2016-08-03 20:32:11 +00:00
|
|
|
def set_transfer_finished_handler(self, handler):
|
2018-05-16 16:04:02 +00:00
|
|
|
self._finished_event += handler
|
2016-08-03 20:32:11 +00:00
|
|
|
|
2016-03-17 17:46:18 +00:00
|
|
|
def get_file_number(self):
|
|
|
|
return self._file_number
|
|
|
|
|
2018-05-16 16:04:02 +00:00
|
|
|
file_number = property(get_file_number)
|
|
|
|
|
2018-05-17 12:20:47 +00:00
|
|
|
def get_state(self):
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
def set_state(self, value):
|
|
|
|
self._state = value
|
|
|
|
self._signal()
|
|
|
|
|
|
|
|
state = property(get_state, set_state)
|
|
|
|
|
2016-03-17 17:46:18 +00:00
|
|
|
def get_friend_number(self):
|
|
|
|
return self._friend_number
|
|
|
|
|
2018-05-16 16:04:02 +00:00
|
|
|
friend_number = property(get_friend_number)
|
|
|
|
|
|
|
|
def get_file_id(self):
|
2016-07-30 18:43:28 +00:00
|
|
|
return self._file_id
|
|
|
|
|
2018-05-16 16:04:02 +00:00
|
|
|
file_id = property(get_file_id)
|
|
|
|
|
2016-07-30 12:44:45 +00:00
|
|
|
def get_path(self):
|
|
|
|
return self._path
|
|
|
|
|
2018-05-16 16:04:02 +00:00
|
|
|
path = property(get_path)
|
|
|
|
|
|
|
|
def get_size(self):
|
|
|
|
return self._size
|
|
|
|
|
|
|
|
size = property(get_size)
|
|
|
|
|
2016-03-19 17:58:42 +00:00
|
|
|
def cancel(self):
|
|
|
|
self.send_control(TOX_FILE_CONTROL['CANCEL'])
|
2018-05-16 21:02:22 +00:00
|
|
|
if self._file is not None:
|
2016-03-23 15:22:32 +00:00
|
|
|
self._file.close()
|
2018-05-17 12:20:47 +00:00
|
|
|
self._signal()
|
2016-03-19 17:58:42 +00:00
|
|
|
|
2016-03-23 14:27:05 +00:00
|
|
|
def cancelled(self):
|
2018-05-16 21:02:22 +00:00
|
|
|
if self._file is not None:
|
2016-03-23 15:22:32 +00:00
|
|
|
self._file.close()
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['CANCELLED']
|
2016-05-08 19:29:50 +00:00
|
|
|
|
|
|
|
def pause(self, by_friend):
|
|
|
|
if not by_friend:
|
|
|
|
self.send_control(TOX_FILE_CONTROL['PAUSE'])
|
2016-05-09 15:32:29 +00:00
|
|
|
else:
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['PAUSED_BY_FRIEND']
|
2016-03-23 14:27:05 +00:00
|
|
|
|
2016-03-17 17:46:18 +00:00
|
|
|
def send_control(self, control):
|
|
|
|
if self._tox.file_control(self._friend_number, self._file_number, control):
|
|
|
|
self.state = control
|
|
|
|
|
|
|
|
def get_file_id(self):
|
|
|
|
return self._tox.file_get_file_id(self._friend_number, self._file_number)
|
|
|
|
|
2018-05-17 12:20:47 +00:00
|
|
|
def _signal(self):
|
2018-05-16 16:04:02 +00:00
|
|
|
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)
|
|
|
|
|
2016-03-23 14:27:05 +00:00
|
|
|
# -----------------------------------------------------------------------------------------------------------------
|
|
|
|
# Send file
|
|
|
|
# -----------------------------------------------------------------------------------------------------------------
|
|
|
|
|
2016-03-16 20:11:18 +00:00
|
|
|
|
|
|
|
class SendTransfer(FileTransfer):
|
2016-03-19 17:58:42 +00:00
|
|
|
|
2016-03-18 13:20:07 +00:00
|
|
|
def __init__(self, path, tox, friend_number, kind=TOX_FILE_KIND['DATA'], file_id=None):
|
|
|
|
if path is not None:
|
2018-05-17 20:31:48 +00:00
|
|
|
fl = open(path, 'rb')
|
2016-03-21 17:19:13 +00:00
|
|
|
size = getsize(path)
|
|
|
|
else:
|
2018-05-17 20:31:48 +00:00
|
|
|
fl = None
|
2016-03-21 17:19:13 +00:00
|
|
|
size = 0
|
2018-04-26 20:54:39 +00:00
|
|
|
super().__init__(path, tox, friend_number, size)
|
2018-05-17 20:31:48 +00:00
|
|
|
self._file = fl
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['OUTGOING_NOT_STARTED']
|
2016-03-21 17:19:13 +00:00
|
|
|
self._file_number = tox.file_send(friend_number, kind, size, file_id,
|
2016-06-23 12:04:36 +00:00
|
|
|
bytes(basename(path), 'utf-8') if path else b'')
|
2016-07-30 18:43:28 +00:00
|
|
|
self._file_id = self.get_file_id()
|
2016-03-16 20:11:18 +00:00
|
|
|
|
|
|
|
def send_chunk(self, position, size):
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Send chunk
|
|
|
|
:param position: start position in file
|
|
|
|
:param size: chunk max size
|
|
|
|
"""
|
2016-06-17 11:53:33 +00:00
|
|
|
if self._creation_time is None:
|
|
|
|
self._creation_time = time()
|
2016-03-18 13:50:32 +00:00
|
|
|
if size:
|
|
|
|
self._file.seek(position)
|
|
|
|
data = self._file.read(size)
|
2016-03-21 20:51:29 +00:00
|
|
|
self._tox.file_send_chunk(self._friend_number, self._file_number, position, data)
|
2016-03-21 17:19:13 +00:00
|
|
|
self._done += size
|
2018-05-17 21:06:14 +00:00
|
|
|
self._signal()
|
2016-03-18 13:50:32 +00:00
|
|
|
else:
|
2018-05-17 12:20:47 +00:00
|
|
|
if self._file is not None:
|
2016-05-02 18:54:32 +00:00
|
|
|
self._file.close()
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['FINISHED']
|
|
|
|
self._finished()
|
2016-03-16 20:11:18 +00:00
|
|
|
|
|
|
|
|
2016-03-18 13:20:07 +00:00
|
|
|
class SendAvatar(SendTransfer):
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Send avatar to friend. Doesn't need file transfer item
|
|
|
|
"""
|
2016-03-19 17:58:42 +00:00
|
|
|
|
2016-03-18 13:20:07 +00:00
|
|
|
def __init__(self, path, tox, friend_number):
|
|
|
|
if path is None:
|
2018-04-26 20:54:39 +00:00
|
|
|
avatar_hash = None
|
2016-03-18 13:20:07 +00:00
|
|
|
else:
|
|
|
|
with open(path, 'rb') as fl:
|
2018-04-26 20:54:39 +00:00
|
|
|
avatar_hash = Tox.hash(fl.read())
|
2018-05-03 21:17:48 +00:00
|
|
|
super().__init__(path, tox, friend_number, TOX_FILE_KIND['AVATAR'], avatar_hash)
|
2016-03-18 13:20:07 +00:00
|
|
|
|
|
|
|
|
2016-03-23 14:27:05 +00:00
|
|
|
class SendFromBuffer(FileTransfer):
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Send inline image
|
|
|
|
"""
|
2016-03-23 14:27:05 +00:00
|
|
|
|
|
|
|
def __init__(self, tox, friend_number, data, file_name):
|
2018-04-26 20:54:39 +00:00
|
|
|
super().__init__(None, tox, friend_number, len(data))
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['OUTGOING_NOT_STARTED']
|
2016-03-23 14:27:05 +00:00
|
|
|
self._data = data
|
2016-06-22 08:48:00 +00:00
|
|
|
self._file_number = tox.file_send(friend_number, TOX_FILE_KIND['DATA'],
|
|
|
|
len(data), None, bytes(file_name, 'utf-8'))
|
2016-03-23 14:27:05 +00:00
|
|
|
|
2016-04-13 21:46:28 +00:00
|
|
|
def get_data(self):
|
|
|
|
return self._data
|
|
|
|
|
2018-05-18 09:26:02 +00:00
|
|
|
data = property(get_data)
|
|
|
|
|
2016-03-23 14:27:05 +00:00
|
|
|
def send_chunk(self, position, size):
|
2016-06-17 11:53:33 +00:00
|
|
|
if self._creation_time is None:
|
|
|
|
self._creation_time = time()
|
2016-03-23 14:27:05 +00:00
|
|
|
if size:
|
|
|
|
data = self._data[position:position + size]
|
2016-06-21 20:43:43 +00:00
|
|
|
self._tox.file_send_chunk(self._friend_number, self._file_number, position, data)
|
2016-03-23 14:27:05 +00:00
|
|
|
self._done += size
|
|
|
|
else:
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['FINISHED']
|
|
|
|
self._finished()
|
2018-05-17 12:20:47 +00:00
|
|
|
self._signal()
|
2016-03-23 14:27:05 +00:00
|
|
|
|
2016-06-11 21:40:58 +00:00
|
|
|
|
|
|
|
class SendFromFileBuffer(SendTransfer):
|
|
|
|
|
|
|
|
def __init__(self, *args):
|
2018-04-26 20:54:39 +00:00
|
|
|
super().__init__(*args)
|
2016-06-11 21:40:58 +00:00
|
|
|
|
|
|
|
def send_chunk(self, position, size):
|
2018-04-26 20:54:39 +00:00
|
|
|
super().send_chunk(position, size)
|
2016-06-11 21:40:58 +00:00
|
|
|
if not size:
|
|
|
|
chdir(dirname(self._path))
|
|
|
|
remove(self._path)
|
|
|
|
|
2016-03-23 14:27:05 +00:00
|
|
|
# -----------------------------------------------------------------------------------------------------------------
|
|
|
|
# Receive file
|
|
|
|
# -----------------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
2016-03-16 20:11:18 +00:00
|
|
|
class ReceiveTransfer(FileTransfer):
|
2016-03-19 17:58:42 +00:00
|
|
|
|
2016-07-30 18:43:28 +00:00
|
|
|
def __init__(self, path, tox, friend_number, size, file_number, position=0):
|
2018-04-26 20:54:39 +00:00
|
|
|
super().__init__(path, tox, friend_number, size, file_number)
|
2016-08-04 09:56:47 +00:00
|
|
|
self._file = open(self._path, 'wb')
|
2016-07-30 18:43:28 +00:00
|
|
|
self._file_size = position
|
|
|
|
self._file.truncate(position)
|
|
|
|
self._missed = set()
|
|
|
|
self._file_id = self.get_file_id()
|
2016-07-30 19:51:25 +00:00
|
|
|
self._done = position
|
2016-03-16 20:11:18 +00:00
|
|
|
|
2016-03-19 17:58:42 +00:00
|
|
|
def cancel(self):
|
2018-04-26 20:54:39 +00:00
|
|
|
super().cancel()
|
2016-03-19 17:58:42 +00:00
|
|
|
remove(self._path)
|
|
|
|
|
2016-07-30 18:43:28 +00:00
|
|
|
def total_size(self):
|
|
|
|
self._missed.add(self._file_size)
|
2018-04-26 20:54:39 +00:00
|
|
|
|
2016-07-30 18:43:28 +00:00
|
|
|
return min(self._missed)
|
|
|
|
|
2016-03-16 20:11:18 +00:00
|
|
|
def write_chunk(self, position, data):
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Incoming chunk
|
|
|
|
:param position: position in file to save data
|
|
|
|
:param data: raw data (string)
|
|
|
|
"""
|
2016-06-17 11:53:33 +00:00
|
|
|
if self._creation_time is None:
|
|
|
|
self._creation_time = time()
|
2016-03-21 17:19:13 +00:00
|
|
|
if data is None:
|
|
|
|
self._file.close()
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['FINISHED']
|
|
|
|
self._finished()
|
2016-06-17 19:35:05 +00:00
|
|
|
else:
|
|
|
|
data = bytearray(data)
|
2016-03-21 17:19:13 +00:00
|
|
|
if self._file_size < position:
|
2016-03-17 15:55:18 +00:00
|
|
|
self._file.seek(0, 2)
|
2016-06-22 13:09:44 +00:00
|
|
|
self._file.write(b'\0' * (position - self._file_size))
|
2016-07-30 18:43:28 +00:00
|
|
|
self._missed.add(self._file_size)
|
|
|
|
else:
|
|
|
|
self._missed.discard(position)
|
2016-03-16 21:07:09 +00:00
|
|
|
self._file.seek(position)
|
2016-03-17 17:46:18 +00:00
|
|
|
self._file.write(data)
|
2016-03-21 18:53:02 +00:00
|
|
|
l = len(data)
|
|
|
|
if position + l > self._file_size:
|
|
|
|
self._file_size = position + l
|
|
|
|
self._done += l
|
2018-05-17 12:20:47 +00:00
|
|
|
self._signal()
|
2016-03-18 13:20:07 +00:00
|
|
|
|
|
|
|
|
2016-04-12 13:11:10 +00:00
|
|
|
class ReceiveToBuffer(FileTransfer):
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Inline image - save in buffer not in file system
|
|
|
|
"""
|
2016-04-12 13:11:10 +00:00
|
|
|
|
|
|
|
def __init__(self, tox, friend_number, size, file_number):
|
2018-04-26 20:54:39 +00:00
|
|
|
super().__init__(None, tox, friend_number, size, file_number)
|
2016-06-22 08:48:00 +00:00
|
|
|
self._data = bytes()
|
2016-04-12 13:11:10 +00:00
|
|
|
self._data_size = 0
|
|
|
|
|
|
|
|
def get_data(self):
|
|
|
|
return self._data
|
|
|
|
|
2018-05-18 09:26:02 +00:00
|
|
|
data = property(get_data)
|
|
|
|
|
2016-04-12 13:11:10 +00:00
|
|
|
def write_chunk(self, position, data):
|
2016-06-17 11:53:33 +00:00
|
|
|
if self._creation_time is None:
|
|
|
|
self._creation_time = time()
|
2016-04-12 13:11:10 +00:00
|
|
|
if data is None:
|
2018-05-16 16:04:02 +00:00
|
|
|
self.state = FILE_TRANSFER_STATE['FINISHED']
|
|
|
|
self._finished()
|
2016-04-12 13:11:10 +00:00
|
|
|
else:
|
2016-06-22 08:48:00 +00:00
|
|
|
data = bytes(data)
|
2016-04-12 13:11:10 +00:00
|
|
|
l = len(data)
|
|
|
|
if self._data_size < position:
|
2016-06-22 08:48:00 +00:00
|
|
|
self._data += (b'\0' * (position - self._data_size))
|
2016-04-12 13:11:10 +00:00
|
|
|
self._data = self._data[:position] + data + self._data[position + l:]
|
|
|
|
if position + l > self._data_size:
|
|
|
|
self._data_size = position + l
|
|
|
|
self._done += l
|
2018-05-17 12:20:47 +00:00
|
|
|
self._signal()
|
2016-04-12 13:11:10 +00:00
|
|
|
|
|
|
|
|
2016-03-18 13:20:07 +00:00
|
|
|
class ReceiveAvatar(ReceiveTransfer):
|
2016-04-14 12:01:59 +00:00
|
|
|
"""
|
|
|
|
Get friend's avatar. Doesn't need file transfer item
|
|
|
|
"""
|
2016-04-02 11:41:06 +00:00
|
|
|
MAX_AVATAR_SIZE = 512 * 1024
|
2016-03-19 17:58:42 +00:00
|
|
|
|
2018-05-10 20:54:51 +00:00
|
|
|
def __init__(self, path, tox, friend_number, size, file_number):
|
|
|
|
full_path = path + '.tmp'
|
|
|
|
super().__init__(full_path, tox, friend_number, size, file_number)
|
2016-04-02 11:41:06 +00:00
|
|
|
if size > self.MAX_AVATAR_SIZE:
|
|
|
|
self.send_control(TOX_FILE_CONTROL['CANCEL'])
|
2016-06-22 13:09:44 +00:00
|
|
|
self._file.close()
|
2018-05-10 20:54:51 +00:00
|
|
|
remove(full_path)
|
2016-05-25 13:05:16 +00:00
|
|
|
elif not size:
|
|
|
|
self.send_control(TOX_FILE_CONTROL['CANCEL'])
|
|
|
|
self._file.close()
|
2018-05-10 20:54:51 +00:00
|
|
|
remove(full_path)
|
2016-04-02 11:41:06 +00:00
|
|
|
elif exists(path):
|
2016-05-25 13:05:16 +00:00
|
|
|
hash = self.get_file_id()
|
2016-06-21 11:58:11 +00:00
|
|
|
with open(path, 'rb') as fl:
|
2016-05-25 13:05:16 +00:00
|
|
|
data = fl.read()
|
|
|
|
existing_hash = Tox.hash(data)
|
|
|
|
if hash == existing_hash:
|
2016-03-18 13:20:07 +00:00
|
|
|
self.send_control(TOX_FILE_CONTROL['CANCEL'])
|
2016-06-22 13:09:44 +00:00
|
|
|
self._file.close()
|
2018-05-10 20:54:51 +00:00
|
|
|
remove(full_path)
|
2016-03-18 13:20:07 +00:00
|
|
|
else:
|
2016-05-25 13:05:16 +00:00
|
|
|
self.send_control(TOX_FILE_CONTROL['RESUME'])
|
2016-03-18 13:20:07 +00:00
|
|
|
else:
|
|
|
|
self.send_control(TOX_FILE_CONTROL['RESUME'])
|
2016-04-28 19:57:32 +00:00
|
|
|
|
|
|
|
def write_chunk(self, position, data):
|
2018-07-02 19:53:07 +00:00
|
|
|
if data is None:
|
2016-04-28 19:57:32 +00:00
|
|
|
avatar_path = self._path[:-4]
|
|
|
|
if exists(avatar_path):
|
2016-06-11 21:40:58 +00:00
|
|
|
chdir(dirname(avatar_path))
|
2016-04-28 19:57:32 +00:00
|
|
|
remove(avatar_path)
|
|
|
|
rename(self._path, avatar_path)
|
2018-07-02 19:53:07 +00:00
|
|
|
super().write_chunk(position, data)
|