[Core] Refactor add method in tm

This commit is contained in:
Calum Lind 2015-08-30 23:57:55 +01:00
parent 7414737cbf
commit 1acd6e4c1c
2 changed files with 124 additions and 192 deletions

View File

@ -233,15 +233,15 @@ class Torrent(object):
# Load values from state if we have it # Load values from state if we have it
if state: if state:
self.set_trackers(state.trackers) self.set_trackers(state.trackers)
self.filename = state.filename
self.is_finished = state.is_finished self.is_finished = state.is_finished
self.filename = state.filename
last_sess_prepend = "[Error from Previous Session] " last_sess_prepend = "[Error from Previous Session] "
if state.error_statusmsg and not state.error_statusmsg.startswith(last_sess_prepend): if state.error_statusmsg and not state.error_statusmsg.startswith(last_sess_prepend):
self.error_statusmsg = last_sess_prepend + state.error_statusmsg self.error_statusmsg = last_sess_prepend + state.error_statusmsg
else: else:
self.error_statusmsg = state.error_statusmsg self.error_statusmsg = state.error_statusmsg
else: else:
self.trackers = [tracker for tracker in self.handle.trackers()] self.set_trackers()
self.is_finished = False self.is_finished = False
# Use infohash as fallback. # Use infohash as fallback.
if not filename: if not filename:
@ -553,7 +553,7 @@ class Torrent(object):
# End Options methods # # End Options methods #
def set_trackers(self, trackers): def set_trackers(self, trackers=None):
"""Sets the trackers for this torrent. """Sets the trackers for this torrent.
Args: Args:

View File

@ -280,9 +280,8 @@ class TorrentManager(component.Component):
if log.isEnabledFor(logging.DEBUG): if log.isEnabledFor(logging.DEBUG):
log.debug("Attempting to extract torrent_info from %s", filepath) log.debug("Attempting to extract torrent_info from %s", filepath)
try: try:
with open(filepath, "rb") as _file: torrent_info = lt.torrent_info(filepath)
torrent_info = lt.torrent_info(lt.bdecode(_file.read())) except RuntimeError as ex:
except (IOError, RuntimeError) as ex:
log.warning("Unable to open torrent file %s: %s", filepath, ex) log.warning("Unable to open torrent file %s: %s", filepath, ex)
else: else:
return torrent_info return torrent_info
@ -307,107 +306,58 @@ class TorrentManager(component.Component):
Returns None if adding was unsuccessful. Returns None if adding was unsuccessful.
""" """
if torrent_info is None and state is None and filedump is None and magnet is None: if torrent_info is None and filedump is None and magnet is None:
log.debug("You must specify a valid torrent_info, torrent state or magnet.") log.error("You must specify a valid torrent_info, torrent state or magnet.")
return return
add_torrent_params = {} if filedump:
if filedump is not None:
try: try:
torrent_info = lt.torrent_info(lt.bdecode(filedump)) torrent_info = lt.torrent_info(lt.bdecode(filedump))
except RuntimeError as ex: except RuntimeError as ex:
log.error("Unable to decode torrent file!: %s", ex) log.error("Unable to add torrent, decoding filedump failed: %s", ex)
# XXX: Probably should raise an exception here...
return return
if torrent_info is None and state: add_torrent_params = {}
# We have no torrent_info so we need to add the torrent with information
# from the state object.
# Populate the options dict from state
options = TorrentOptions()
options["max_connections"] = state.max_connections
options["max_upload_slots"] = state.max_upload_slots
options["max_upload_speed"] = state.max_upload_speed
options["max_download_speed"] = state.max_download_speed
options["prioritize_first_last_pieces"] = state.prioritize_first_last
options["sequential_download"] = state.sequential_download
options["file_priorities"] = state.file_priorities
storage_mode = state.storage_mode
options["pre_allocate_storage"] = (storage_mode == "allocate")
options["download_location"] = state.save_path
options["auto_managed"] = state.auto_managed
options["stop_at_ratio"] = state.stop_at_ratio
options["stop_ratio"] = state.stop_ratio
options["remove_at_ratio"] = state.remove_at_ratio
options["move_completed"] = state.move_completed
options["move_completed_path"] = state.move_completed_path
options["add_paused"] = state.paused
options["shared"] = state.shared
options["super_seeding"] = state.super_seeding
options["priority"] = state.priority
options["owner"] = state.owner
options["name"] = state.name
torrent_info = self.get_torrent_info_from_file(
os.path.join(self.state_dir, state.torrent_id + ".torrent"))
if torrent_info: if torrent_info:
add_torrent_params["ti"] = torrent_info add_torrent_params["ti"] = torrent_info
elif state.magnet: name = torrent_info.name()
magnet = state.magnet if not name:
name = torrent_info.file_at(0).path.replace("\\", "/", 1).split("/", 1)[0]
add_torrent_params["name"] = name
torrent_id = str(torrent_info.info_hash())
elif magnet:
magnet_info = get_magnet_info(magnet)
if magnet_info:
add_torrent_params["url"] = utf8_encoded(magnet)
add_torrent_params["name"] = magnet_info["name"]
torrent_id = magnet_info["info_hash"]
else: else:
log.error("Unable to add torrent!") log.error("Unable to add magnet, invalid magnet info: %s", magnet)
return return
if resume_data:
add_torrent_params["resume_data"] = resume_data # Check for existing torrent in session.
else: if torrent_id in self.get_torrent_list():
# We have a torrent_info object or magnet uri so we're not loading from state. log.warning("Unable to add torrent (%s), already in session", torrent_id)
if torrent_info: if torrent_info:
add_torrent_id = str(torrent_info.info_hash()) log.info("Adding any new trackers to torrent (%s) already in session...", torrent_id)
# If this torrent id is already in the session, merge any additional trackers. # Don't merge trackers if either torrent has private flag set.
if add_torrent_id in self.get_torrent_list(): if torrent_info.priv() or self[torrent_id].get_status(["private"])["private"]:
log.info("Merging trackers for torrent (%s) already in session...", add_torrent_id) log.info("Adding trackers aborted: Torrent has private flag set.")
# Don't merge trackers if either torrent has private flag set
if self[add_torrent_id].get_status(["private"])["private"]:
log.info("Merging trackers abandoned: Torrent has private flag set.")
return
add_torrent_trackers = []
for value in torrent_info.trackers():
tracker = {}
tracker["url"] = value.url
tracker["tier"] = value.tier
add_torrent_trackers.append(tracker)
torrent_trackers = {}
tracker_list = []
for tracker in self[add_torrent_id].get_status(["trackers"])["trackers"]:
torrent_trackers[(tracker["url"])] = tracker
tracker_list.append(tracker)
added_tracker = 0
for tracker in add_torrent_trackers:
if tracker['url'] not in torrent_trackers:
tracker_list.append(tracker)
added_tracker += 1
if added_tracker:
log.info("%s tracker(s) merged into torrent.", added_tracker)
self[add_torrent_id].set_trackers(tracker_list)
return
# Check if options is None and load defaults
if options is None:
options = TorrentOptions()
else: else:
for tracker in torrent_info.trackers():
self[torrent_id].handle.add_tracker({"url": tracker.url, "tier": tracker.tier})
# Update torrent.trackers from lt.
self[torrent_id].set_trackers()
return
# Load default options and update if needed.
_options = TorrentOptions() _options = TorrentOptions()
if options:
_options.update(options) _options.update(options)
options = _options options = _options
# Check for renamed files and if so, rename them in the torrent_info # Check for renamed files and if so, rename them in the torrent_info before adding.
# before adding to the session.
if options["mapped_files"]: if options["mapped_files"]:
for index, fname in options["mapped_files"].items(): for index, fname in options["mapped_files"].items():
fname = sanitize_filepath(decode_string(fname)) fname = sanitize_filepath(decode_string(fname))
@ -417,91 +367,74 @@ class TorrentManager(component.Component):
except TypeError: except TypeError:
torrent_info.rename_file(index, utf8_encoded(fname)) torrent_info.rename_file(index, utf8_encoded(fname))
if options["pre_allocate_storage"]: if not options["owner"]:
storage_mode = "allocate" options["owner"] = component.get("RPCServer").get_session_user()
else: if not component.get("AuthManager").has_account(options["owner"]):
storage_mode = "sparse" options["owner"] = "localclient"
add_torrent_params["ti"] = torrent_info
if log.isEnabledFor(logging.DEBUG): if log.isEnabledFor(logging.DEBUG):
log.debug("options: %s", options) log.debug("options: %s", options)
if not options["owner"]: # Fill in the rest of the add_torrent_params dictionary.
options["owner"] = component.get("RPCServer").get_session_user()
account_exists = component.get("AuthManager").has_account(options["owner"])
if not account_exists:
options["owner"] = "localclient"
# Fill in the rest of the add_torrent_params dictionary
add_torrent_params["save_path"] = utf8_encoded(options["download_location"]) add_torrent_params["save_path"] = utf8_encoded(options["download_location"])
if options["name"]:
try: add_torrent_params["name"] = options["name"]
add_torrent_params["storage_mode"] = lt.storage_mode_t.names["storage_mode_" + storage_mode] if options["pre_allocate_storage"]:
except KeyError: add_torrent_params["storage_mode"] = lt.storage_mode_t.storage_mode_allocate
add_torrent_params["storage_mode"] = lt.storage_mode_t.storage_mode_sparse if resume_data:
add_torrent_params["resume_data"] = resume_data
default_flags = (lt.add_torrent_params_flags_t.flag_paused | default_flags = (lt.add_torrent_params_flags_t.flag_paused |
lt.add_torrent_params_flags_t.flag_auto_managed | lt.add_torrent_params_flags_t.flag_auto_managed |
lt.add_torrent_params_flags_t.flag_update_subscribe | lt.add_torrent_params_flags_t.flag_update_subscribe |
lt.add_torrent_params_flags_t.flag_apply_ip_filter) lt.add_torrent_params_flags_t.flag_apply_ip_filter)
# Set flags: enable duplicate_is_error and disable auto_managed # Set flags: enable duplicate_is_error & override_resume_data, disable auto_managed.
add_torrent_params["flags"] = ((default_flags add_torrent_params["flags"] = ((default_flags
| lt.add_torrent_params_flags_t.flag_duplicate_is_error) | lt.add_torrent_params_flags_t.flag_duplicate_is_error
| lt.add_torrent_params_flags_t.flag_override_resume_data)
^ lt.add_torrent_params_flags_t.flag_auto_managed) ^ lt.add_torrent_params_flags_t.flag_auto_managed)
if options["seed_mode"]: if options["seed_mode"]:
add_torrent_params["flags"] |= lt.add_torrent_params_flags_t.flag_seed_mode add_torrent_params["flags"] |= lt.add_torrent_params_flags_t.flag_seed_mode
if magnet:
magnet_info = get_magnet_info(magnet)
if magnet_info:
add_torrent_params['url'] = utf8_encoded(magnet)
add_torrent_params['name'] = magnet_info['name']
if options['name']:
add_torrent_params['name'] = options['name']
elif torrent_info:
name = torrent_info.name()
if not name:
name = torrent_info.file_at(0).path.replace("\\", "/", 1).split("/", 1)[0]
add_torrent_params['name'] = name
# We need to pause the AlertManager momentarily to prevent alerts # We need to pause the AlertManager momentarily to prevent alerts
# for this torrent being generated before a Torrent object is created. # for this torrent being generated before a Torrent object is created.
component.pause("AlertManager") component.pause("AlertManager")
handle = None
try: try:
handle = self.session.add_torrent(add_torrent_params) handle = self.session.add_torrent(add_torrent_params)
except RuntimeError as ex: if not handle.is_valid():
log.warning("Error adding torrent: %s", ex) raise InvalidTorrentError("Torrent handle is invalid!")
except (RuntimeError, InvalidTorrentError) as ex:
if not handle or not handle.is_valid(): log.error("Unable to add torrent to session: %s", ex)
log.debug("torrent handle is invalid!")
# The torrent was not added to the session
component.resume("AlertManager") component.resume("AlertManager")
return return
if log.isEnabledFor(logging.DEBUG): # Create a Torrent object and add to the dictionary.
log.debug("handle id: %s", str(handle.info_hash()))
# Set auto_managed to False because the torrent is paused
handle.auto_managed(False)
# Create a Torrent object
torrent = Torrent(handle, options, state, filename, magnet) torrent = Torrent(handle, options, state, filename, magnet)
# Add the torrent object to the dictionary
self.torrents[torrent.torrent_id] = torrent self.torrents[torrent.torrent_id] = torrent
if self.config["queue_new_to_top"]:
handle.queue_position_top()
component.resume("AlertManager") component.resume("AlertManager")
# Resume the torrent if needed # Add to queued torrents set.
self.queued_torrents.add(torrent.torrent_id)
if self.config["queue_new_to_top"]:
self.queue_top()
# Resume the torrent if needed.
if not options["add_paused"]: if not options["add_paused"]:
torrent.resume() torrent.resume()
# Add to queued torrents set # Emit torrent_added signal
self.queued_torrents.add(torrent.torrent_id) from_state = state is not None
component.get("EventManager").emit(TorrentAddedEvent(torrent.torrent_id, from_state))
if log.isEnabledFor(logging.DEBUG):
log.debug("Torrent added: %s", str(handle.info_hash()))
if log.isEnabledFor(logging.INFO):
name_and_owner = torrent.get_status(["name", "owner"])
log.info("Torrent %s from user \"%s\" %s",
name_and_owner["name"],
name_and_owner["owner"],
from_state and "loaded" or "added")
# Write the .torrent file to the state directory # Write the .torrent file to the state directory
if filedump: if filedump:
@ -511,16 +444,6 @@ class TorrentManager(component.Component):
# Save the session state # Save the session state
self.save_state() self.save_state()
# Emit torrent_added signal
from_state = state is not None
component.get("EventManager").emit(TorrentAddedEvent(torrent.torrent_id, from_state))
if log.isEnabledFor(logging.INFO):
name_and_owner = torrent.get_status(["name", "owner"])
log.info("Torrent %s from user \"%s\" %s",
name_and_owner["name"],
name_and_owner["owner"],
from_state and "loaded" or "added")
return torrent.torrent_id return torrent.torrent_id
def remove(self, torrent_id, remove_data=False, save_state=True): def remove(self, torrent_id, remove_data=False, save_state=True):
@ -590,20 +513,17 @@ class TorrentManager(component.Component):
def load_state(self): def load_state(self):
"""Load the state of the TorrentManager from the torrents.state file""" """Load the state of the TorrentManager from the torrents.state file"""
filename = "torrents.state" torrents_state = os.path.join(self.state_dir, "torrents.state")
filepath = os.path.join(self.state_dir, filename) for filepath in (torrents_state, torrents_state + ".bak"):
filepath_bak = filepath + ".bak" log.info("Loading torrent state: %s", filepath)
for _filepath in (filepath, filepath_bak):
log.info("Opening %s for load: %s", filename, _filepath)
try: try:
with open(_filepath, "rb") as _file: with open(filepath, "rb") as _file:
state = cPickle.load(_file) state = cPickle.load(_file)
except (IOError, EOFError, cPickle.UnpicklingError) as ex: except (IOError, EOFError, cPickle.UnpicklingError) as ex:
log.warning("Unable to load %s: %s", _filepath, ex) log.warning("Unable to load %s: %s", filepath, ex)
state = None state = None
else: else:
log.info("Successfully loaded %s: %s", filename, _filepath) log.info("Successfully loaded %s", filepath)
break break
if state is None: if state is None:
@ -621,10 +541,9 @@ class TorrentManager(component.Component):
else: else:
setattr(t_state, attr, getattr(state_tmp, attr, None)) setattr(t_state, attr, getattr(state_tmp, attr, None))
except AttributeError as ex: except AttributeError as ex:
log.exception("Unable to update state file to a compatible version: %s", ex) log.error("Unable to update state file to a compatible version: %s", ex)
# Reorder the state.torrents list to add torrents in the correct queue # Reorder the state.torrents list to add torrents in the correct queue order.
# order.
state.torrents.sort(key=operator.attrgetter("queue"), reverse=self.config["queue_new_to_top"]) state.torrents.sort(key=operator.attrgetter("queue"), reverse=self.config["queue_new_to_top"])
resume_data = self.load_resume_data_file() resume_data = self.load_resume_data_file()
@ -632,15 +551,28 @@ class TorrentManager(component.Component):
# This speeds up startup loading the torrents by quite a lot for some reason (~40%) # This speeds up startup loading the torrents by quite a lot for some reason (~40%)
self.alerts.wait_on_handler = True self.alerts.wait_on_handler = True
for torrent_state in state.torrents: for t_state in state.torrents:
# Populate the options dict from state
options = TorrentOptions()
for option in options:
try: try:
self.add(state=torrent_state, save_state=False, options[option] = getattr(t_state, option)
resume_data=resume_data.get(torrent_state.torrent_id)) except AttributeError:
except AttributeError as ex: pass
log.error("Torrent state file is either corrupt or incompatible! %s", ex) # Manually update unmatched attributes
import traceback options["download_location"] = t_state.save_path
traceback.print_exc() options["pre_allocate_storage"] = t_state.storage_mode == "allocate"
break options["prioritize_first_last_pieces"] = t_state.prioritize_first_last
options["add_paused"] = t_state.paused
magnet = t_state.magnet
torrent_info = self.get_torrent_info_from_file(
os.path.join(self.state_dir, t_state.torrent_id + ".torrent"))
if torrent_info:
magnet = None
self.add(torrent_info=torrent_info, state=t_state, options=options, save_state=False,
magnet=magnet, resume_data=resume_data.get(t_state.torrent_id))
self.alerts.wait_on_handler = False self.alerts.wait_on_handler = False
log.info("Finished loading %d torrents.", len(state.torrents)) log.info("Finished loading %d torrents.", len(state.torrents))