deluge/library/pytorrent.py

619 lines
20 KiB
Python
Raw Normal View History

2006-12-03 18:20:15 +00:00
#
# Copyright (C) 2006 Alon Zakai ('Kripken') <kripkensteiner@gmail.com>
2006-12-05 09:22:53 +00:00
# Copyright (C) 2006 Zach Tibbitts <zach@collegegeek.org>
2006-12-03 18:20:15 +00:00
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2, or (at your option)
# any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
2006-12-05 19:03:30 +00:00
# Deluge Library, a.k.a. pytorrent, previously known as python-libtorrent:
2006-12-05 09:22:53 +00:00
#
2006-12-05 19:03:30 +00:00
# pytorrent is a Python library for torrenting, that includes
2006-12-05 09:22:53 +00:00
# pytorrent.py, which is Python code, and pytorrent_core, which is also a Python
# module, but written in C++, and includes the libtorrent torrent library. Only
# pytorrent should be visible, and only it should be imported, in the client.
2006-12-05 19:03:30 +00:00
# pytorrent_core contains mainly libtorrent-interfacing code, and a few other things
# that make most sense to write at that level. pytorrent.py contains all other
# torrent-system management: queueing, configuration management, persistent
# list of torrents, etc.
2006-12-05 09:22:53 +00:00
#
2006-12-03 18:20:15 +00:00
2006-12-05 16:19:48 +00:00
# Documentation:
# Torrents have 3 structures:
# 1. torrent_info - persistent data, like name, upload speed cap, etc.
# 2. core_torrent_state - transient state data from the core. This may take
# time to calculate, so we do if efficiently
# 3. supp_torrent_state - supplementary torrent data, from pytorrent
2006-12-03 18:20:15 +00:00
import pytorrent_core
import os, shutil
import pickle
import time
2006-12-03 19:08:51 +00:00
# Constants
TORRENTS_SUBDIR = "torrentfiles"
2006-12-05 09:22:53 +00:00
2006-12-04 16:42:35 +00:00
STATE_FILENAME = "persistent.state"
PREFS_FILENAME = "prefs.state"
DHT_FILENAME = "dht.state"
2006-12-08 10:17:50 +00:00
CACHED_DATA_EXPIRATION = 1 # seconds, like the output of time.time()
2006-12-05 09:22:53 +00:00
2006-12-04 16:42:35 +00:00
# "max_half_open" : -1,
2006-12-05 20:46:03 +00:00
DEFAULT_PREFS = {
2006-12-05 09:22:53 +00:00
"max_uploads" : 2, # a.k.a. upload slots
2006-12-04 16:42:35 +00:00
"listen_on" : [6881,9999],
"max_connections" : 80,
"use_DHT" : True,
"max_active_torrents" : -1,
"auto_seed_ratio" : -1,
"max_download_rate" : -1,
"max_upload_rate" : -1
}
2006-12-05 20:46:03 +00:00
PREF_FUNCTIONS = {
"max_uploads" : pytorrent_core.set_max_uploads,
"listen_on" : pytorrent_core.set_listen_on,
"max_connections" : pytorrent_core.set_max_connections,
"use_DHT" : None, # not a normal pref in that is is applied only on start
"max_active_torrents" : None, # no need for a function, applied constantly
"auto_seed_ratio" : None, # no need for a function, applied constantly
"max_download_rate" : pytorrent_core.set_download_rate_limit,
"max_upload_rate" : pytorrent_core.set_upload_rate_limit
}
2006-12-06 08:00:34 +00:00
# Exceptions
2006-12-05 09:22:53 +00:00
class PyTorrentError(Exception):
def __init__(self, value):
self.value = value
def __str__(self):
return repr(self.value)
2006-12-06 08:00:34 +00:00
class InvalidEncodingError(PyTorrentError):
pass
class FilesystemError(PyTorrentError):
pass
class DuplicateTorrentError(PyTorrentError):
pass
class InvalidTorrentError(PyTorrentError):
pass
2006-12-03 19:08:51 +00:00
2006-12-08 10:17:50 +00:00
# A cached data item
class cached_data:
def __init__(self, get_method, key):
self.get_method = get_method
self.key = key
self.timestamp = -1
def get(self, efficiently=True):
if self.timestamp == -1 or time.time() > self.timestamp + CACHED_DATA_EXPIRATION or \
not efficiently:
self.data = self.get_method(key)
self.timestamp = time.time()
return self.data
2006-12-05 16:19:48 +00:00
# Persistent information for a single torrent
2006-12-03 18:20:15 +00:00
2006-12-05 16:19:48 +00:00
class torrent_info:
2006-12-03 18:20:15 +00:00
def __init__(self, filename, save_dir, compact):
self.filename = filename
self.save_dir = save_dir
self.compact = compact
2006-12-05 19:52:54 +00:00
self.user_paused = False # start out unpaused
self.uploaded_memory = 0
2006-12-03 18:20:15 +00:00
2006-12-05 19:41:57 +00:00
self.file_filter = []
2006-12-03 18:20:15 +00:00
self.delete_me = False # set this to true, to delete it on next sync
# The persistent state of the torrent system. Everything in this will be pickled
class persistent_state:
def __init__(self):
2006-12-04 16:42:35 +00:00
# Torrents
self.torrents = []
2006-12-03 19:08:51 +00:00
2006-12-03 18:20:15 +00:00
# Prepare queue (queue is pickled, just like everything else)
self.queue = [] # queue[x] is the unique_ID of the x-th queue position. Simple.
# The manager for the torrent system
class manager:
2006-12-05 17:25:30 +00:00
# blank_slate mode ignores the two pickle files and DHT state file, i.e. you start
# completely fresh. When quitting, the old files will be overwritten
def __init__(self, client_ID, version, user_agent, base_dir, blank_slate=False):
2006-12-03 19:08:51 +00:00
self.base_dir = base_dir
2006-12-03 18:20:15 +00:00
# Ensure directories exist
2006-12-03 19:08:51 +00:00
if not TORRENTS_SUBDIR in os.listdir(self.base_dir):
os.mkdir(self.base_dir + "/" + TORRENTS_SUBDIR)
2006-12-03 18:20:15 +00:00
2006-12-06 08:00:34 +00:00
# Pre-initialize the core's data structures
pytorrent_core.pre_init(PyTorrentError,
InvalidEncodingError,
FilesystemError,
DuplicateTorrentError,
InvalidTorrentError)
2006-12-03 18:20:15 +00:00
# Start up the core
assert(len(version) == 4)
pytorrent_core.init(client_ID,
int(version[0]),
int(version[1]),
int(version[2]),
int(version[3]),
user_agent)
2006-12-05 14:32:17 +00:00
self.constants = pytorrent_core.constants()
2006-12-03 18:20:15 +00:00
# Unique IDs are NOT in the state, since they are temporary for each session
2006-12-05 19:41:57 +00:00
self.unique_IDs = {} # unique_ID -> a torrent_info object, i.e. persistent data
2006-12-05 16:19:48 +00:00
# Saved torrent core_states. We do not poll the core in a costly manner, necessarily
2006-12-08 10:17:50 +00:00
self.saved_core_torrent_states = {} # unique_ID -> torrent_state
2006-12-03 18:20:15 +00:00
2006-12-05 16:19:48 +00:00
# supplementary torrent states
self.supp_torrent_states = {} # unique_ID->dict of data
2006-12-05 09:22:53 +00:00
2006-12-08 10:17:50 +00:00
# Saved torrent core_states. We do not poll the core in a costly manner, necessarily
self.saved_core_torrent_peer_infos = {} # unique_ID -> torrent_state
2006-12-04 16:42:35 +00:00
# Unpickle the preferences, or create a new one
2006-12-05 17:25:30 +00:00
self.prefs = DEFAULT_PREFS
if not blank_slate:
try:
pkl_file = open(self.base_dir + "/" + PREFS_FILENAME, 'rb')
self.prefs = pickle.load(pkl_file)
pkl_file.close()
except IOError:
pass
2006-12-04 16:42:35 +00:00
2006-12-05 09:22:53 +00:00
# Apply preferences. Note that this is before any torrents are added
2006-12-04 16:42:35 +00:00
self.apply_prefs()
2006-12-05 09:22:53 +00:00
# Apply DHT, if needed. Note that this is before any torrents are added
2006-12-04 16:42:35 +00:00
if self.get_pref('use_DHT'):
2006-12-05 17:25:30 +00:00
if not blank_slate:
pytorrent_core.start_DHT(self.base_dir + "/" + DHT_FILENAME)
else:
pytorrent_core.start_DHT("")
2006-12-04 16:42:35 +00:00
2006-12-03 18:20:15 +00:00
# Unpickle the state, or create a new one
2006-12-05 17:25:30 +00:00
if not blank_slate:
try:
pkl_file = open(self.base_dir + "/" + STATE_FILENAME, 'rb')
self.state = pickle.load(pkl_file)
pkl_file.close()
# Sync with the core: tell core about torrents, and get unique_IDs
self.sync()
2006-12-05 19:41:57 +00:00
# Apply all the file filters, right after adding the torrents
self.apply_all_file_filters()
2006-12-05 17:25:30 +00:00
except IOError:
self.state = persistent_state()
else:
2006-12-03 18:20:15 +00:00
self.state = persistent_state()
def quit(self):
2006-12-05 19:52:54 +00:00
# Analyze data needed for pickling, etc.
self.pre_quitting()
2006-12-04 16:42:35 +00:00
# Pickle the prefs
2006-12-05 17:25:30 +00:00
print "Pickling prefs..."
2006-12-04 16:42:35 +00:00
output = open(self.base_dir + "/" + PREFS_FILENAME, 'wb')
pickle.dump(self.prefs, output)
output.close()
2006-12-03 18:20:15 +00:00
# Pickle the state
2006-12-05 17:25:30 +00:00
print "Pickling state..."
2006-12-04 16:42:35 +00:00
output = open(self.base_dir + "/" + STATE_FILENAME, 'wb')
2006-12-03 18:20:15 +00:00
pickle.dump(self.state, output)
output.close()
# Save fastresume data
2006-12-05 17:25:30 +00:00
print "Saving fastresume data..."
2006-12-03 18:20:15 +00:00
self.save_fastresume_data()
2006-12-04 16:42:35 +00:00
# Stop DHT, if needed
if self.get_pref('use_DHT'):
2006-12-05 17:25:30 +00:00
print "Stopping DHT..."
2006-12-04 16:42:35 +00:00
pytorrent_core.stop_DHT(self.base_dir + "/" + DHT_FILENAME)
2006-12-03 18:20:15 +00:00
# Shutdown torrent core
2006-12-05 17:25:30 +00:00
print "Quitting the core..."
2006-12-03 19:08:51 +00:00
pytorrent_core.quit()
2006-12-03 18:20:15 +00:00
2006-12-05 19:52:54 +00:00
def pre_quitting(self):
# Save the uploaded data from this session to the existing upload memory
for unique_ID in self.unique_IDs.keys():
self.unique_IDs[unique_ID].uploaded_memory = \
self.unique_IDs[unique_ID].uploaded_memory + \
2006-12-08 10:17:50 +00:00
self.get_core_torrent_state(unique_ID, False)['total_upload'] # Purposefully ineffi.
2006-12-05 19:52:54 +00:00
2006-12-05 19:03:30 +00:00
# Preference management functions
2006-12-04 16:42:35 +00:00
def get_pref(self, key):
# If we have a value, return, else fallback on default_prefs, else raise an error
# the fallback is useful if the source has newer prefs than the existing pref state,
# which was created by an old version of the source
if key in self.prefs.keys():
return self.prefs[key]
elif key in DEFAULT_PREFS:
self.prefs[key] = DEFAULT_PREFS[key]
return self.prefs[key]
else:
2006-12-05 09:22:53 +00:00
raise PyTorrentError("Asked for a pref that doesn't exist: " + key)
def set_pref(self, key, value):
# Make sure this is a valid key
if key not in DEFAULT_PREFS.keys():
raise PyTorrentError("Asked to change a pref that isn't valid: " + key)
self.prefs[key] = value
2006-12-04 16:42:35 +00:00
2006-12-05 20:46:03 +00:00
# Apply the pref, if applicable
if PREF_FUNCTIONS[key] is not None:
PREF_FUNCTIONS[key](value)
2006-12-05 14:46:04 +00:00
2006-12-05 19:03:30 +00:00
# Torrent addition and removal functions
2006-12-03 18:20:15 +00:00
def add_torrent(self, filename, save_dir, compact):
self.add_torrent_ns(filename, save_dir, compact)
return self.sync() # Syncing will create a new torrent in the core, and return it's ID
def remove_torrent(self, unique_ID, data_also):
# Save some data before we remove the torrent, needed later in this func
temp = self.unique_IDs[unique_ID]
temp_fileinfo = pytorrent_core.get_fileinfo(unique_ID)
self.remove_torrent_ns(unique_ID)
self.sync()
# Remove .torrent and .fastresume
os.remove(temp.filename)
try:
# Must be after removal of the torrent, because that saves a new .fastresume
os.remove(temp.filename + ".fastresume")
except OSError:
pass # Perhaps there never was one to begin with
# Remove data, if asked to do so
if data_also:
# Must be done AFTER the torrent is removed
# Note: can this be to the trash?
for filedata in temp_fileinfo:
filename = filedata['path']
try:
os.remove(temp.save_dir + "/" + filename)
except OSError:
pass # No file just means it wasn't downloaded, we can continue
# A separate function, because people may want to call it from time to time
def save_fastresume_data(self):
for unique_ID in self.unique_IDs:
pytorrent_core.save_fastresume(unique_ID, self.unique_IDs[unique_ID].filename)
2006-12-05 19:03:30 +00:00
# State retrieval functions
2006-12-05 14:32:17 +00:00
def get_state(self):
ret = pytorrent_core.get_session_info()
2006-12-08 10:17:50 +00:00
# Get additional data from our level
2006-12-05 14:32:17 +00:00
ret['is_listening'] = pytorrent_core.is_listening()
ret['port'] = pytorrent_core.listening_port()
if self.get_pref('use_DHT'):
ret['DHT_nodes'] = pytorrent_core.get_DHT_info()
2006-12-08 10:17:50 +00:00
2006-12-05 14:32:17 +00:00
return ret
2006-12-05 17:25:30 +00:00
# This is the EXTERNAL function, for the GUI. It returns the core_state + supp_state
2006-12-08 10:17:50 +00:00
def get_torrent_state(self, unique_ID, full=False):
ret = self.get_core_torrent_state(unique_ID, True).copy()
2006-12-05 17:25:30 +00:00
2006-12-08 10:17:50 +00:00
# Add the pytorrent-level things to the pytorrent_core data
2006-12-05 17:25:30 +00:00
if self.get_supp_torrent_state(unique_ID) is not None:
ret.update(self.get_supp_torrent_state(unique_ID))
2006-12-08 10:17:50 +00:00
# If asked, we calculate the time-costly information as well
if full:
ret['availability'] = self.calc_availability(unique_ID)
ret['swarm speed'] = self.calc_swarm_speed(unique_ID)
2006-12-05 17:25:30 +00:00
return ret
2006-12-08 10:17:50 +00:00
def get_torrent_peer_info(self, unique_ID):
# Perhaps at some time we may add info here
return get_core_torrent_peer_info(unique_ID)
2006-12-05 19:03:30 +00:00
# Queueing functions
2006-12-03 18:20:15 +00:00
def queue_up(self, unique_ID):
curr_index = self.get_queue_index(unique_ID)
if curr_index > 0:
temp = self.state.queue[curr_index - 1]
self.state.queue[curr_index - 1] = unique_ID
self.state.queue[curr_index] = temp
def queue_down(self, unique_ID):
curr_index = self.get_queue_index(unique_ID)
if curr_index < (len(self.state.queue) - 1):
temp = self.state.queue[curr_index + 1]
self.state.queue[curr_index + 1] = unique_ID
self.state.queue[curr_index] = temp
def queue_bottom(self, unique_ID):
curr_index = self.get_queue_index(unique_ID)
if curr_index < (len(self.state.queue) - 1):
self.state.queue.remove(curr_index)
self.state.queue.append(unique_ID)
def clear_completed(self):
for unique_ID in self.unique_IDs:
2006-12-08 10:17:50 +00:00
torrent_state = self.get_core_torrent_state(unique_ID)
2006-12-03 18:20:15 +00:00
if torrent_state['progress'] == 100.0:
self.remove_torrent_ns(unique_ID)
self.sync()
# Enforce the queue: pause/unpause as needed, based on queue and user_pausing
# This should be called after changes to relevant parameters (user_pausing, or
# altering max_active_torrents), or just from time to time
# ___ALL queuing code should be in this function, and ONLY here___
2006-12-05 16:19:48 +00:00
def apply_queue(self, efficient = True):
2006-12-03 18:20:15 +00:00
# Handle autoseeding - downqueue as needed
if self.auto_seed_ratio != -1:
for unique_ID in self.unique_IDs:
2006-12-08 10:17:50 +00:00
if self.get_core_torrent_state(unique_ID, efficient)['is_seed']:
torrent_state = self.get_core_torrent_state(unique_ID, efficient)
2006-12-03 18:20:15 +00:00
ratio = self.calc_ratio(unique_ID, torrent_state)
if ratio >= self.auto_seed_ratio:
self.queue_bottom(unique_ID)
# Pause and resume torrents
for index in range(len(self.state.queue)):
unique_ID = self.state.queue[index]
2006-12-03 19:08:51 +00:00
if (index < self.state.max_active_torrents or self.state_max_active_torrents == -1) \
2006-12-08 10:17:50 +00:00
and self.get_core_torrent_state(unique_ID, efficient)['is_paused'] \
2006-12-03 18:20:15 +00:00
and not self.is_user_paused(unique_ID):
pytorrent_core.resume(unique_ID)
2006-12-08 10:17:50 +00:00
elif not self.get_core_torrent_state(unique_ID, efficient)['is_paused'] and \
2006-12-03 18:20:15 +00:00
(index >= self.state.max_active_torrents or self.is_user_paused(unique_ID)):
pytorrent_core.pause(unique_ID)
2006-12-05 19:03:30 +00:00
# Event handling
2006-12-05 14:32:17 +00:00
def handle_events(self):
2006-12-05 19:03:30 +00:00
# Handle them for the backend's purposes, but still send them up in case the client
# wants to do something - show messages, for example
2006-12-05 16:19:48 +00:00
ret = []
2006-12-05 14:32:17 +00:00
event = pytorrent_core.pop_event()
while event is not None:
2006-12-05 14:46:04 +00:00
# print "EVENT: ", event
2006-12-05 14:32:17 +00:00
2006-12-05 16:19:48 +00:00
ret.append(event)
2006-12-05 14:32:17 +00:00
if event['event_type'] is self.constants['EVENT_FINISHED']:
# If we are autoseeding, then we need to apply the queue
if self.auto_seed_ratio == -1:
2006-12-05 16:19:48 +00:00
self.apply_queue(efficient = False) # To work on current data
2006-12-05 14:32:17 +00:00
elif event['event_type'] is self.constants['EVENT_TRACKER']:
2006-12-08 09:05:07 +00:00
unique_ID = event['unique_ID']
status = event['tracker_status']
message = event['message']
tracker = message[message.find('"')+1:message.rfind('"')]
self.set_supp_torrent_state_val(unique_ID,
"tracker_status",
(tracker, status))
old_state = self.get_supp_torrent_state(unique_ID)
try:
new = old_state['tracker_messages']
except KeyError:
new = {}
new[tracker] = message
self.set_supp_torrent_state_val(unique_ID,
"tracker_messages",
new)
2006-12-05 14:32:17 +00:00
event = pytorrent_core.pop_event()
2006-12-05 16:19:48 +00:00
return ret
2006-12-05 19:41:57 +00:00
# Filtering functions
def set_file_filter(self, unique_ID, file_filter):
2006-12-08 10:17:50 +00:00
assert(len(file_filter) == self.get_core_torrent_state(unique_ID, True)['num_files'])
2006-12-05 19:41:57 +00:00
self.unique_IDs[unique_ID].file_filter = file_filter[:]
pytorrent_core.set_filter_out(file_filter)
def get_file_filter(self, unique_ID):
try:
return self.unique_IDs[unique_ID].file_filter[:]
except AttributeError:
return None
2006-12-05 20:20:39 +00:00
# Called when a session starts, to apply existing filters
2006-12-05 19:41:57 +00:00
def apply_all_file_filters(self):
2006-12-05 19:52:54 +00:00
for unique_ID in self.unique_IDs.keys():
2006-12-05 19:41:57 +00:00
try:
self.set_file_filter(self.unique_IDs[unique_ID].file_filter)
except AttributeError:
pass
2006-12-05 19:03:30 +00:00
# Miscellaneous minor functions
def set_user_pause(self, unique_ID, new_value):
self.unique_IDs[unique_ID].user_paused = new_value
self.apply_queue()
def is_user_paused(self, unique_ID):
return self.unique_IDs[unique_ID].user_paused
def get_num_torrents(self):
return pytorrent_core.get_num_torrents()
def get_unique_IDs(self):
return self.unique_IDs.keys()
2006-12-05 14:32:17 +00:00
2006-12-03 18:20:15 +00:00
####################
# Internal functions
####################
2006-12-05 16:19:48 +00:00
# Efficient: use a saved state, if it hasn't expired yet
2006-12-08 10:17:50 +00:00
def get_core_torrent_state(self, unique_ID, efficiently=True):
if unique_ID not in self.saved_torrent_states.keys()
self.saved_torrent_states[unique_ID] = cached_data(pytorrent_core.get_torrent_state,
unique_ID)
2006-12-05 16:19:48 +00:00
2006-12-08 10:17:50 +00:00
return self.saved_torrent_states[unique_ID].get(efficiently)
2006-12-05 16:19:48 +00:00
def get_supp_torrent_state(self, unique_ID):
try:
return self.supp_torrent_states[unique_ID]
except KeyError:
return None
def set_supp_torrent_state_val(self, unique_ID, key, val):
try:
if self.supp_torrent_states[unique_ID] is None:
self.supp_torrent_states[unique_ID] = {}
except KeyError:
self.supp_torrent_states[unique_ID] = {}
self.supp_torrent_states[unique_ID][key] = val
2006-12-08 10:17:50 +00:00
def get_core_torrent_peer_info(self, unique_ID, efficiently=True):
if unique_ID not in self.saved_torrent_peer_infos.keys()
self.saved_torrent_peer_infos[unique_ID] = cached_data(pytorrent_core.get_peer_info,
unique_ID)
return self.saved_torrent_peer_infos[unique_ID].get(efficiently)
2006-12-03 18:20:15 +00:00
# Non-syncing functions. Used when we loop over such events, and sync manually at the end
def add_torrent_ns(self, filename, save_dir, compact):
# Cache torrent file
2006-12-05 17:25:30 +00:00
(temp, filename_short) = os.path.split(filename)
2006-12-03 18:20:15 +00:00
2006-12-05 17:25:30 +00:00
if filename_short in os.listdir(self.base_dir + "/" + TORRENTS_SUBDIR):
raise PyTorrentError("Duplicate Torrent, it appears: " + filename_short)
2006-12-03 18:20:15 +00:00
2006-12-05 17:25:30 +00:00
full_new_name = self.base_dir + "/" + TORRENTS_SUBDIR + "/" + filename_short
2006-12-03 18:20:15 +00:00
shutil.copy(filename, full_new_name)
# Create torrent object
2006-12-05 16:19:48 +00:00
new_torrent = torrent_info(full_new_name, save_dir, compact)
2006-12-03 18:20:15 +00:00
self.state.torrents.append(new_torrent)
def remove_torrent_ns(self, unique_ID):
self.unique_IDs[unique_ID].delete_me = True
# Sync the state.torrents and unique_IDs lists with the core
# ___ALL syncing code with the core is here, and ONLY here___
# Also all self-syncing is done here (various lists)
def sync(self):
ret = None # We return new added unique ID(s), or None
# Add torrents to core and unique_IDs
torrents_with_unique_ID = self.unique_IDs.values()
for torrent in self.state.torrents:
if torrent not in torrents_with_unique_ID:
2006-12-05 09:22:53 +00:00
# print "Adding torrent to core:", torrent.filename, torrent.save_dir, torrent.compact
2006-12-03 18:20:15 +00:00
unique_ID = pytorrent_core.add_torrent(torrent.filename,
torrent.save_dir,
torrent.compact)
2006-12-05 09:22:53 +00:00
# print "Got unique ID:", unique_ID
2006-12-03 18:20:15 +00:00
ret = unique_ID
self.unique_IDs[unique_ID] = torrent
# Remove torrents from core, unique_IDs and queue
to_delete = []
for torrent in self.state.torrents:
if torrent.delete_me:
pytorrent_core.remove_torrent(torrent.unique_ID, torrent.filename)
to_delete.append(torrent.unique_ID)
for unique_ID in to_delete:
self.state.torrents.remove(self.unique_IDs[unique_ID])
self.state.queue.remove(self.unique_IDs[unique_ID])
del self.unique_IDs[unique_ID]
# Add torrents to queue - at the end, of course
for unique_ID in self.unique_IDs:
if unique_ID not in self.state.queue:
self.state.queue.append(unique_ID)
assert(len(self.unique_IDs) == len(self.state.torrents))
assert(len(self.unique_IDs) == len(self.state.queue))
assert(len(self.unique_IDs) == pytorrent_core.get_num_torrents())
return ret
def get_queue_index(self, unique_ID):
return self.state.queue.index(unique_ID)
2006-12-05 17:25:30 +00:00
def apply_prefs(self):
print "Applying preferences"
2006-12-05 20:46:03 +00:00
assert(len(PREF_FUNCTIONS) == len(DEFAULT_PREFS))
2006-12-05 17:25:30 +00:00
2006-12-05 20:46:03 +00:00
for pref in PREF_FUNCTIONS.keys():
if PREF_FUNCTIONS[pref] is not None:
PREF_FUNCTIONS[pref](self.get_pref(pref))
2006-12-05 17:25:30 +00:00
2006-12-08 10:17:50 +00:00
# Calculations
2006-12-05 17:25:30 +00:00
def calc_ratio(self, unique_ID, torrent_state):
up = float(torrent_state['total_upload'] + self.unique_IDs[unique_ID].uploaded_memory)
down = float(torrent_state["total_done"])
try:
ret = float(up/down)
except:
ret = -1
return ret
2006-12-08 10:17:50 +00:00
def calc_availability(self, unique_ID):
def calc_swarm_speed(self, unique_ID):