# # core.py # # Copyright (C) 2007-2009 Andrew Resch # Copyright (C) 2011 Pedro Algarvio # # Deluge is free software. # # You may redistribute it and/or modify it under the terms of the # GNU General Public License, as published by the Free Software # Foundation; either version 3 of the License, or (at your option) # any later version. # # deluge 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 deluge. If not, write to: # The Free Software Foundation, Inc., # 51 Franklin Street, Fifth Floor # Boston, MA 02110-1301, USA. # # In addition, as a special exception, the copyright holders give # permission to link the code of portions of this program with the OpenSSL # library. # You must obey the GNU General Public License in all respects for all of # the code used other than OpenSSL. If you modify file(s) with this # exception, you may extend this exception to your version of the file(s), # but you are not obligated to do so. If you do not wish to do so, delete # this exception statement from your version. If you delete this exception # statement from all source files in the program, then also delete it here. # # from deluge._libtorrent import lt import os import glob import base64 import logging import threading import tempfile from urlparse import urljoin import twisted.web.client import twisted.web.error from deluge.httpdownloader import download_file import deluge.configmanager import deluge.common import deluge.component as component from deluge.event import * from deluge.error import * from deluge.core.authmanager import AUTH_LEVEL_ADMIN, AUTH_LEVEL_NONE from deluge.core.authmanager import AUTH_LEVELS_MAPPING, AUTH_LEVELS_MAPPING_REVERSE from deluge.core.torrentmanager import TorrentManager from deluge.core.pluginmanager import PluginManager from deluge.core.alertmanager import AlertManager from deluge.core.filtermanager import FilterManager from deluge.core.preferencesmanager import PreferencesManager from deluge.core.authmanager import AuthManager from deluge.core.eventmanager import EventManager from deluge.core.rpcserver import export log = logging.getLogger(__name__) class Core(component.Component): def __init__(self, listen_interface=None): log.debug("Core init..") component.Component.__init__(self, "Core") # Start the libtorrent session log.info("Starting libtorrent %s session..", lt.version) # Create the client fingerprint version = [int(value.split("-")[0]) for value in deluge.common.get_version().split(".")] while len(version) < 4: version.append(0) # Note: All libtorrent python bindings to set plugins/extensions need to be disabled # due to GIL issue. https://code.google.com/p/libtorrent/issues/detail?id=369 # Setting session flags to 1 enables all libtorrent default plugins self.session = lt.session(lt.fingerprint("DE", *version), flags=1) # Load the session state if available self.__load_session_state() # Set the user agent self.settings = lt.session_settings() self.settings.user_agent = "Deluge/%(deluge_version)s Libtorrent/%(lt_version)s" % \ { 'deluge_version': deluge.common.get_version(), 'lt_version': self.get_libtorrent_version().rpartition(".")[0] } # Increase the alert queue size so that alerts don't get lost self.settings.alert_queue_size = 10000 # Set session settings self.settings.send_redundant_have = True if deluge.common.windows_check(): self.settings.disk_io_write_mode = \ lt.io_buffer_mode_t.disable_os_cache self.settings.disk_io_read_mode = \ lt.io_buffer_mode_t.disable_os_cache self.session.set_settings(self.settings) # Load metadata extension # Note: All libtorrent python bindings to set plugins/extensions need to be disabled # due to GIL issue. https://code.google.com/p/libtorrent/issues/detail?id=369 # self.session.add_extension(lt.create_metadata_plugin) # self.session.add_extension(lt.create_ut_metadata_plugin) # self.session.add_extension(lt.create_smart_ban_plugin) # Create the components self.eventmanager = EventManager() self.preferencesmanager = PreferencesManager() self.alertmanager = AlertManager() self.pluginmanager = PluginManager(self) self.torrentmanager = TorrentManager() self.filtermanager = FilterManager(self) self.authmanager = AuthManager() # New release check information self.new_release = None # Get the core config self.config = deluge.configmanager.ConfigManager("core.conf") self.config.save() # If there was an interface value from the command line, use it, but # store the one in the config so we can restore it on shutdown self.__old_interface = None if listen_interface: self.__old_interface = self.config["listen_interface"] self.config["listen_interface"] = listen_interface def start(self): """Starts the core""" # New release check information self.__new_release = None def stop(self): log.debug("Core stopping...") # Save the DHT state if necessary if self.config["dht"]: self.save_dht_state() # Save the libtorrent session state self.__save_session_state() # We stored a copy of the old interface value if self.__old_interface: self.config["listen_interface"] = self.__old_interface # Make sure the config file has been saved self.config.save() def shutdown(self): pass def __save_session_state(self): """Saves the libtorrent session state""" try: session_state = deluge.configmanager.get_config_dir("session.state") open(session_state, "wb").write(lt.bencode(self.session.save_state())) except Exception, e: log.warning("Failed to save lt state: %s", e) def __load_session_state(self): """Loads the libtorrent session state""" try: session_state = deluge.configmanager.get_config_dir("session.state") self.session.load_state(lt.bdecode(open(session_state, "rb").read())) except Exception, e: log.warning("Failed to load lt state: %s", e) def save_dht_state(self): """Saves the dht state to a file""" try: dht_data = open(deluge.configmanager.get_config_dir("dht.state"), "wb") dht_data.write(lt.bencode(self.session.dht_state())) dht_data.close() except Exception, e: log.warning("Failed to save dht state: %s", e) def get_new_release(self): log.debug("get_new_release") from urllib2 import urlopen try: self.new_release = urlopen( "http://download.deluge-torrent.org/version-1.0").read().strip() except Exception, e: log.debug("Unable to get release info from website: %s", e) return self.check_new_release() def check_new_release(self): if self.new_release: log.debug("new_release: %s", self.new_release) if deluge.common.VersionSplit(self.new_release) > deluge.common.VersionSplit(deluge.common.get_version()): component.get("EventManager").emit(NewVersionAvailableEvent(self.new_release)) return self.new_release return False # Exported Methods @export def add_torrent_file(self, filename, filedump, options): """ Adds a torrent file to the session. :param filename: the filename of the torrent :type filename: string :param filedump: a base64 encoded string of the torrent file contents :type filedump: string :param options: the options to apply to the torrent on add :type options: dict :returns: the torrent_id as a str or None :rtype: string """ try: filedump = base64.decodestring(filedump) except Exception, e: log.error("There was an error decoding the filedump string!") log.exception(e) try: torrent_id = self.torrentmanager.add( filedump=filedump, options=options, filename=filename ) except Exception, e: log.error("There was an error adding the torrent file %s", filename) log.exception(e) torrent_id = None return torrent_id @export def add_torrent_url(self, url, options, headers=None): """ Adds a torrent from a url. Deluge will attempt to fetch the torrent from url prior to adding it to the session. :param url: the url pointing to the torrent file :type url: string :param options: the options to apply to the torrent on add :type options: dict :param headers: any optional headers to send :type headers: dict :returns: a Deferred which returns the torrent_id as a str or None """ log.info("Attempting to add url %s", url) def on_download_success(filename): # We got the file, so add it to the session f = open(filename, "rb") data = f.read() f.close() try: os.remove(filename) except Exception, e: log.warning("Couldn't remove temp file: %s", e) return self.add_torrent_file( filename, base64.encodestring(data), options ) def on_download_fail(failure): if failure.check(twisted.web.error.PageRedirect): new_url = urljoin(url, failure.getErrorMessage().split(" to ")[1]) result = download_file( new_url, tempfile.mkstemp()[1], headers=headers, force_filename=True ) result.addCallbacks(on_download_success, on_download_fail) elif failure.check(twisted.web.client.PartialDownloadError): result = download_file( url, tempfile.mkstemp()[1], headers=headers, force_filename=True, allow_compression=False ) result.addCallbacks(on_download_success, on_download_fail) else: # Log the error and pass the failure onto the client log.error("Error occured downloading torrent from %s", url) log.error("Reason: %s", failure.getErrorMessage()) result = failure return result d = download_file( url, tempfile.mkstemp()[1], headers=headers, force_filename=True ) d.addCallbacks(on_download_success, on_download_fail) return d @export def add_torrent_magnet(self, uri, options): """ Adds a torrent from a magnet link. :param uri: the magnet link :type uri: string :param options: the options to apply to the torrent on add :type options: dict :returns: the torrent_id :rtype: string """ log.debug("Attempting to add by magnet uri: %s", uri) return self.torrentmanager.add(magnet=uri, options=options) @export def remove_torrent(self, torrent_id, remove_data): """ Removes a torrent from the session. :param torrent_id: the torrent_id of the torrent to remove :type torrent_id: string :param remove_data: if True, remove the data associated with this torrent :type remove_data: boolean :returns: True if removed successfully :rtype: bool :raises InvalidTorrentError: if the torrent_id does not exist in the session """ log.debug("Removing torrent %s from the core.", torrent_id) return self.torrentmanager.remove(torrent_id, remove_data) @export def get_session_status(self, keys): """ Gets the session status values for 'keys', these keys are taking from libtorrent's session status. See: http://www.rasterbar.com/products/libtorrent/manual.html#status :param keys: the keys for which we want values :type keys: list :returns: a dictionary of {key: value, ...} :rtype: dict """ status = {} session_status = self.session.status() for key in keys: status[key] = getattr(session_status, key) return status @export def get_cache_status(self): """ Returns a dictionary of the session's cache status. :returns: the cache status :rtype: dict """ status = self.session.get_cache_status() cache = {} for attr in dir(status): if attr.startswith("_"): continue cache[attr] = getattr(status, attr) # Add in a couple ratios try: cache["write_hit_ratio"] = float((cache["blocks_written"] - cache["writes"])) / float(cache["blocks_written"]) except ZeroDivisionError: cache["write_hit_ratio"] = 0.0 try: cache["read_hit_ratio"] = float(cache["blocks_read_hit"]) / float(cache["blocks_read"]) except ZeroDivisionError: cache["read_hit_ratio"] = 0.0 return cache @export def force_reannounce(self, torrent_ids): log.debug("Forcing reannouncment to: %s", torrent_ids) for torrent_id in torrent_ids: self.torrentmanager[torrent_id].force_reannounce() @export def pause_torrent(self, torrent_ids): log.debug("Pausing: %s", torrent_ids) for torrent_id in torrent_ids: if not self.torrentmanager[torrent_id].pause(): log.warning("Error pausing torrent %s", torrent_id) @export def connect_peer(self, torrent_id, ip, port): log.debug("adding peer %s to %s", ip, torrent_id) if not self.torrentmanager[torrent_id].connect_peer(ip, port): log.warning("Error adding peer %s:%s to %s", ip, port, torrent_id) @export def move_storage(self, torrent_ids, dest): log.debug("Moving storage %s to %s", torrent_ids, dest) for torrent_id in torrent_ids: if not self.torrentmanager[torrent_id].move_storage(dest): log.warning("Error moving torrent %s to %s", torrent_id, dest) @export def pause_all_torrents(self): """Pause all torrents in the session""" for torrent in self.torrentmanager.torrents.values(): torrent.pause() @export def resume_all_torrents(self): """Resume all torrents in the session""" for torrent in self.torrentmanager.torrents.values(): torrent.resume() component.get("EventManager").emit(SessionResumedEvent()) @export def resume_torrent(self, torrent_ids): log.debug("Resuming: %s", torrent_ids) for torrent_id in torrent_ids: self.torrentmanager[torrent_id].resume() def create_torrent_status(self, torrent_id, torrent_keys, plugin_keys, diff=False, update=False): try: status = self.torrentmanager[torrent_id].get_status(torrent_keys, diff, update=update) except KeyError: import traceback traceback.print_exc() # Torrent was probaly removed meanwhile return {} # Ask the plugin manager to fill in the plugin keys if len(plugin_keys) > 0: status.update(self.pluginmanager.get_status(torrent_id, plugin_keys)) return status @export def get_torrent_status(self, torrent_id, keys, diff=False): torrent_keys, plugin_keys = self.torrentmanager.separate_keys(keys, [torrent_id]) return self.create_torrent_status(torrent_id, torrent_keys, plugin_keys, diff=diff, update=True) @export def get_torrents_status(self, filter_dict, keys, diff=False): """ returns all torrents , optionally filtered by filter_dict. """ torrent_ids = self.filtermanager.filter_torrent_ids(filter_dict) status_dict = {}.fromkeys(torrent_ids) d = self.torrentmanager.torrents_status_update(torrent_ids, keys, diff=False) def add_plugin_fields(args): status_dict, plugin_keys = args # Ask the plugin manager to fill in the plugin keys if len(plugin_keys) > 0: for key in status_dict.keys(): status_dict[key].update(self.pluginmanager.get_status(key, plugin_keys)) return status_dict d.addCallback(add_plugin_fields) return d @export def get_filter_tree(self , show_zero_hits=True, hide_cat=None): """ returns {field: [(value,count)] } for use in sidebar(s) """ return self.filtermanager.get_filter_tree(show_zero_hits, hide_cat) @export def get_session_state(self): """Returns a list of torrent_ids in the session.""" # Get the torrent list from the TorrentManager return self.torrentmanager.get_torrent_list() @export def get_config(self): """Get all the preferences as a dictionary""" return self.config.config @export def get_config_value(self, key): """Get the config value for key""" try: value = self.config[key] except KeyError: return None return value @export def get_config_values(self, keys): """Get the config values for the entered keys""" config = {} for key in keys: try: config[key] = self.config[key] except KeyError: pass return config @export def set_config(self, config): """Set the config with values from dictionary""" # Load all the values into the configuration for key in config.keys(): if isinstance(config[key], basestring): config[key] = config[key].encode("utf8") self.config[key] = config[key] @export def get_listen_port(self): """Returns the active listen port""" return self.session.listen_port() @export def get_num_connections(self): """Returns the current number of connections""" return self.session.num_connections() @export def get_available_plugins(self): """Returns a list of plugins available in the core""" return self.pluginmanager.get_available_plugins() @export def get_enabled_plugins(self): """Returns a list of enabled plugins in the core""" return self.pluginmanager.get_enabled_plugins() @export def enable_plugin(self, plugin): self.pluginmanager.enable_plugin(plugin) return None @export def disable_plugin(self, plugin): self.pluginmanager.disable_plugin(plugin) return None @export def force_recheck(self, torrent_ids): """Forces a data recheck on torrent_ids""" for torrent_id in torrent_ids: self.torrentmanager[torrent_id].force_recheck() @export def set_torrent_options(self, torrent_ids, options): """Sets the torrent options for torrent_ids""" for torrent_id in torrent_ids: self.torrentmanager[torrent_id].set_options(options) @export def set_torrent_trackers(self, torrent_id, trackers): """Sets a torrents tracker list. trackers will be [{"url", "tier"}]""" return self.torrentmanager[torrent_id].set_trackers(trackers) @export def set_torrent_max_connections(self, torrent_id, value): """Sets a torrents max number of connections""" return self.torrentmanager[torrent_id].set_max_connections(value) @export def set_torrent_max_upload_slots(self, torrent_id, value): """Sets a torrents max number of upload slots""" return self.torrentmanager[torrent_id].set_max_upload_slots(value) @export def set_torrent_max_upload_speed(self, torrent_id, value): """Sets a torrents max upload speed""" return self.torrentmanager[torrent_id].set_max_upload_speed(value) @export def set_torrent_max_download_speed(self, torrent_id, value): """Sets a torrents max download speed""" return self.torrentmanager[torrent_id].set_max_download_speed(value) @export def set_torrent_file_priorities(self, torrent_id, priorities): """Sets a torrents file priorities""" return self.torrentmanager[torrent_id].set_file_priorities(priorities) @export def set_torrent_prioritize_first_last(self, torrent_id, value): """Sets a higher priority to the first and last pieces""" return self.torrentmanager[torrent_id].set_prioritize_first_last(value) @export def set_torrent_sequential_download(self, torrent_id, value): """Toggle sequencial pieces download""" return self.torrentmanager[torrent_id].set_sequential_download(value) @export def set_torrent_auto_managed(self, torrent_id, value): """Sets the auto managed flag for queueing purposes""" return self.torrentmanager[torrent_id].set_auto_managed(value) @export def set_torrent_stop_at_ratio(self, torrent_id, value): """Sets the torrent to stop at 'stop_ratio'""" return self.torrentmanager[torrent_id].set_stop_at_ratio(value) @export def set_torrent_stop_ratio(self, torrent_id, value): """Sets the ratio when to stop a torrent if 'stop_at_ratio' is set""" return self.torrentmanager[torrent_id].set_stop_ratio(value) @export def set_torrent_remove_at_ratio(self, torrent_id, value): """Sets the torrent to be removed at 'stop_ratio'""" return self.torrentmanager[torrent_id].set_remove_at_ratio(value) @export def set_torrent_move_completed(self, torrent_id, value): """Sets the torrent to be moved when completed""" return self.torrentmanager[torrent_id].set_move_completed(value) @export def set_torrent_move_completed_path(self, torrent_id, value): """Sets the path for the torrent to be moved when completed""" return self.torrentmanager[torrent_id].set_move_completed_path(value) @export(AUTH_LEVEL_ADMIN) def set_torrents_owner(self, torrent_ids, username): """Set's the torrent owner. :param torrent_id: the torrent_id of the torrent to remove :type torrent_id: string :param username: the new owner username :type username: string :raises DelugeError: if the username is not known """ if not self.authmanager.has_account(username): raise DelugeError("Username \"%s\" is not known." % username) if isinstance(torrent_ids, basestring): torrent_ids = [torrent_ids] for torrent_id in torrent_ids: self.torrentmanager[torrent_id].set_owner(username) return None @export def set_torrents_shared(self, torrent_ids, shared): if isinstance(torrent_ids, basestring): torrent_ids = [torrent_ids] for torrent_id in torrent_ids: self.torrentmanager[torrent_id].set_options({"shared": shared}) @export def get_path_size(self, path): """Returns the size of the file or folder 'path' and -1 if the path is unaccessible (non-existent or insufficient privs)""" return deluge.common.get_path_size(path) @export def create_torrent(self, path, tracker, piece_length, comment, target, webseeds, private, created_by, trackers, add_to_session): log.debug("creating torrent..") threading.Thread(target=self._create_torrent_thread, args=( path, tracker, piece_length, comment, target, webseeds, private, created_by, trackers, add_to_session)).start() def _create_torrent_thread(self, path, tracker, piece_length, comment, target, webseeds, private, created_by, trackers, add_to_session): import deluge.metafile deluge.metafile.make_meta_file( path, tracker, piece_length, comment=comment, target=target, webseeds=webseeds, private=private, created_by=created_by, trackers=trackers) log.debug("torrent created!") if add_to_session: options = {} options["download_location"] = os.path.split(path)[0] self.add_torrent_file(os.path.split(target)[1], open(target, "rb").read(), options) @export def upload_plugin(self, filename, filedump): """This method is used to upload new plugins to the daemon. It is used when connecting to the daemon remotely and installing a new plugin on the client side. 'plugin_data' is a xmlrpc.Binary object of the file data, ie, plugin_file.read()""" try: filedump = base64.decodestring(filedump) except Exception, e: log.error("There was an error decoding the filedump string!") log.exception(e) return f = open(os.path.join(deluge.configmanager.get_config_dir(), "plugins", filename), "wb") f.write(filedump) f.close() component.get("CorePluginManager").scan_for_plugins() @export def rescan_plugins(self): """ Rescans the plugin folders for new plugins """ component.get("CorePluginManager").scan_for_plugins() @export def rename_files(self, torrent_id, filenames): """ Rename files in torrent_id. Since this is an asynchronous operation by libtorrent, watch for the TorrentFileRenamedEvent to know when the files have been renamed. :param torrent_id: the torrent_id to rename files :type torrent_id: string :param filenames: a list of index, filename pairs :type filenames: ((index, filename), ...) :raises InvalidTorrentError: if torrent_id is invalid """ if torrent_id not in self.torrentmanager.torrents: raise InvalidTorrentError("torrent_id is not in session") self.torrentmanager[torrent_id].rename_files(filenames) @export def rename_folder(self, torrent_id, folder, new_folder): """ Renames the 'folder' to 'new_folder' in 'torrent_id'. Watch for the TorrentFolderRenamedEvent which is emitted when the folder has been renamed successfully. :param torrent_id: the torrent to rename folder in :type torrent_id: string :param folder: the folder to rename :type folder: string :param new_folder: the new folder name :type new_folder: string :raises InvalidTorrentError: if the torrent_id is invalid """ if torrent_id not in self.torrentmanager.torrents: raise InvalidTorrentError("torrent_id is not in session") self.torrentmanager[torrent_id].rename_folder(folder, new_folder) @export def queue_top(self, torrent_ids): log.debug("Attempting to queue %s to top", torrent_ids) # torrent_ids must be sorted in reverse before moving to preserve order for torrent_id in sorted(torrent_ids, key=self.torrentmanager.get_queue_position, reverse=True): try: # If the queue method returns True, then we should emit a signal if self.torrentmanager.queue_top(torrent_id): component.get("EventManager").emit(TorrentQueueChangedEvent()) except KeyError: log.warning("torrent_id: %s does not exist in the queue", torrent_id) @export def queue_up(self, torrent_ids): log.debug("Attempting to queue %s to up", torrent_ids) torrents = ((self.torrentmanager.get_queue_position(torrent_id), torrent_id) for torrent_id in torrent_ids) torrent_moved = True prev_queue_position = None #torrent_ids must be sorted before moving. for queue_position, torrent_id in sorted(torrents): # Move the torrent if and only if there is space (by not moving it we preserve the order) if torrent_moved or queue_position - prev_queue_position > 1: try: torrent_moved = self.torrentmanager.queue_up(torrent_id) except KeyError: log.warning("torrent_id: %s does not exist in the queue", torrent_id) # If the torrent moved, then we should emit a signal if torrent_moved: component.get("EventManager").emit(TorrentQueueChangedEvent()) else: prev_queue_position = queue_position @export def queue_down(self, torrent_ids): log.debug("Attempting to queue %s to down", torrent_ids) torrents = ((self.torrentmanager.get_queue_position(torrent_id), torrent_id) for torrent_id in torrent_ids) torrent_moved = True prev_queue_position = None #torrent_ids must be sorted before moving. for queue_position, torrent_id in sorted(torrents, reverse=True): # Move the torrent if and only if there is space (by not moving it we preserve the order) if torrent_moved or prev_queue_position - queue_position > 1: try: torrent_moved = self.torrentmanager.queue_down(torrent_id) except KeyError: log.warning("torrent_id: %s does not exist in the queue", torrent_id) # If the torrent moved, then we should emit a signal if torrent_moved: component.get("EventManager").emit(TorrentQueueChangedEvent()) else: prev_queue_position = queue_position @export def queue_bottom(self, torrent_ids): log.debug("Attempting to queue %s to bottom", torrent_ids) # torrent_ids must be sorted before moving to preserve order for torrent_id in sorted(torrent_ids, key=self.torrentmanager.get_queue_position): try: # If the queue method returns True, then we should emit a signal if self.torrentmanager.queue_bottom(torrent_id): component.get("EventManager").emit(TorrentQueueChangedEvent()) except KeyError: log.warning("torrent_id: %s does not exist in the queue", torrent_id) @export def glob(self, path): return glob.glob(path) @export def test_listen_port(self): """ Checks if the active port is open :returns: True if the port is open, False if not :rtype: bool """ from twisted.web.client import getPage d = getPage("http://deluge-torrent.org/test_port.php?port=%s" % self.get_listen_port(), timeout=30) def on_get_page(result): return bool(int(result)) def logError(failure): log.warning("Error testing listen port: %s", failure) d.addCallback(on_get_page) d.addErrback(logError) return d @export def get_free_space(self, path=None): """ Returns the number of free bytes at path :param path: the path to check free space at, if None, use the default download location :type path: string :returns: the number of free bytes at path :rtype: int :raises InvalidPathError: if the path is invalid """ if not path: path = self.config["download_location"] try: return deluge.common.free_space(path) except InvalidPathError: return 0 @export def get_libtorrent_version(self): """ Returns the libtorrent version. :returns: the version :rtype: string """ return lt.version @export(AUTH_LEVEL_ADMIN) def get_known_accounts(self): return self.authmanager.get_known_accounts() @export(AUTH_LEVEL_NONE) def get_auth_levels_mappings(self): return (AUTH_LEVELS_MAPPING, AUTH_LEVELS_MAPPING_REVERSE) @export(AUTH_LEVEL_ADMIN) def create_account(self, username, password, authlevel): return self.authmanager.create_account(username, password, authlevel) @export(AUTH_LEVEL_ADMIN) def update_account(self, username, password, authlevel): return self.authmanager.update_account(username, password, authlevel) @export(AUTH_LEVEL_ADMIN) def remove_account(self, username): return self.authmanager.remove_account(username)