| 1 | # |
| 2 | # manage.py |
| 3 | # |
| 4 | # Copyright (C) 2008-2009 Ido Abramovich <ido.deluge@gmail.com> |
| 5 | # Copyright (C) 2009 Andrew Resch <andrewresch@gmail.com> |
| 6 | # |
| 7 | # Deluge is free software. |
| 8 | # |
| 9 | # You may redistribute it and/or modify it under the terms of the |
| 10 | # GNU General Public License, as published by the Free Software |
| 11 | # Foundation; either version 3 of the License, or (at your option) |
| 12 | # any later version. |
| 13 | # |
| 14 | # deluge is distributed in the hope that it will be useful, |
| 15 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 16 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. |
| 17 | # See the GNU General Public License for more details. |
| 18 | # |
| 19 | # You should have received a copy of the GNU General Public License |
| 20 | # along with deluge. If not, write to: |
| 21 | # The Free Software Foundation, Inc., |
| 22 | # 51 Franklin Street, Fifth Floor |
| 23 | # Boston, MA 02110-1301, USA. |
| 24 | # |
| 25 | # In addition, as a special exception, the copyright holders give |
| 26 | # permission to link the code of portions of this program with the OpenSSL |
| 27 | # library. |
| 28 | # You must obey the GNU General Public License in all respects for all of |
| 29 | # the code used other than OpenSSL. If you modify file(s) with this |
| 30 | # exception, you may extend this exception to your version of the file(s), |
| 31 | # but you are not obligated to do so. If you do not wish to do so, delete |
| 32 | # this exception statement from your version. If you delete this exception |
| 33 | # statement from all source files in the program, then also delete it here. |
| 34 | # |
| 35 | # |
| 36 | |
| 37 | from deluge.ui.console.main import BaseCommand |
| 38 | import deluge.ui.console.colors as colors |
| 39 | from deluge.ui.client import client |
| 40 | import deluge.component as component |
| 41 | from deluge.log import LOG as log |
| 42 | |
| 43 | from optparse import make_option |
| 44 | |
| 45 | |
| 46 | # torrent.py's Torrent.get_status() renames some of the options. |
| 47 | # This maps the getter names to the setter names |
| 48 | get_torrent_options = { |
| 49 | 'max_download_speed': 'max_download_speed', |
| 50 | 'max_upload_speed': 'max_upload_speed', |
| 51 | 'max_connections': 'max_connections', |
| 52 | 'max_upload_slots': 'max_upload_slots', |
| 53 | 'prioritize_first_last': 'prioritize_first_last_pieces', |
| 54 | 'is_auto_managed': 'auto_managed', |
| 55 | 'stop_at_ratio': 'stop_at_ratio', |
| 56 | 'stop_ratio': 'stop_ratio', |
| 57 | 'remove_at_ratio': 'remove_at_ratio', |
| 58 | 'move_on_completed': 'move_completed', |
| 59 | 'move_on_completed_path': 'move_completed_path', |
| 60 | #'file_priorities': 'file_priorities', |
| 61 | #'compact': 'compact_allocation', |
| 62 | #'save_path': 'download_location' |
| 63 | } |
| 64 | |
| 65 | # These are the things that can be set, as far as I can tell. The |
| 66 | # ones that aren't commented out are the ones that look like they |
| 67 | # probably will behave as expected if set. |
| 68 | # |
| 69 | # Each value is [ type, getter name, help text ] |
| 70 | set_torrent_options = { |
| 71 | # These are handled by torrent.py's Torrent.set_options() |
| 72 | 'auto_managed': [bool, None, "Makes torrent obey deluge's queue settings. (see FAQ for details)."], |
| 73 | #'download_location': str, # Probably not useful to set |
| 74 | #'file_priorities': ???, # Not a simple value to set, probably needs its own command |
| 75 | 'max_connections': [int, None, 'Maximum number of connections to use for this torrent.'], |
| 76 | 'max_download_speed': [float, None, 'Maximum total download speed to allow this torrent to use (KiB/s).'], |
| 77 | 'max_upload_slots': [int, None, 'Maximum number of connections to allow this torrent to use for uploading.'], |
| 78 | 'max_upload_speed': [float, None, 'Maximum total upload speed to allow this torrent to use (KiB/s).'], |
| 79 | 'prioritize_first_last_pieces': [bool, None, 'Whether to download the first and last piece of the torrent first.\nNOTE: Only has effect if the torrent contains a single file.'], # Only has effect if torrent contains a single file. Can only be set, not cleared. |
| 80 | |
| 81 | # These are "handled" by being set directly on the options dict |
| 82 | 'stop_at_ratio': [bool, None, 'Whether to stop seeding when share ratio reaches "stop_ratio".'], |
| 83 | 'stop_ratio': [float, None, 'The ratio at which to stop seeding (if "stop_at_ratio" is True).'], |
| 84 | 'remove_at_ratio': [bool, None, 'Whether to remove torrent when share ratio reaches "stop_ratio".'], |
| 85 | 'move_completed': [bool, None, 'Whether to move the downloaded data when downloading is complete.'], |
| 86 | 'move_completed_path': [str, None, 'Where to move completed data to (if "move_completed" is True).'], |
| 87 | |
| 88 | #'compact_allocation': bool, # Unclear what setting this would do |
| 89 | #'add_paused': ???, # Not returned by get_status, unclear what setting it would do |
| 90 | #'mapped_files': ??? # Not returned by get_status, unclear what setting it would do |
| 91 | } |
| 92 | for k,v in get_torrent_options.items(): |
| 93 | set_torrent_options[v][1] = k |
| 94 | |
| 95 | |
| 96 | def layout_option_help(opt, text): |
| 97 | base_indent = 7 |
| 98 | hanging_indent = 15 |
| 99 | offset_indent = 20 - len(opt) |
| 100 | split_text = ('\n' + ' ' * hanging_indent).join(text.split('\n')) |
| 101 | if offset_indent > 0: |
| 102 | split_text = ' ' * offset_indent + split_text |
| 103 | else: |
| 104 | split_text = '\n' + ' ' * hanging_indent + split_text |
| 105 | return ' ' * base_indent + opt + ': ' + split_text |
| 106 | |
| 107 | class Command(BaseCommand): |
| 108 | """Show and set per-torrent options""" |
| 109 | |
| 110 | option_help = [ layout_option_help(k, v[2]) for k,v in set_torrent_options.items() ] |
| 111 | option_help.sort() |
| 112 | |
| 113 | option_list = BaseCommand.option_list + ( |
| 114 | make_option('-s', '--set', action='store', nargs=2, dest='set', |
| 115 | help='set value for key'), |
| 116 | ) |
| 117 | usage = '''Usage: manage <torrent-id> [<torrent-id> ...] [<key1> [<key2> ...]] |
| 118 | manage <torrent-id> [<torrent-id> ...] --set <key> <value> |
| 119 | |
| 120 | The torrent-id * (a single asterisk) means "all loaded torrents". |
| 121 | The value -1 means unlimited (for max_connections, max_download_speed, max_upload_slots and max_upload_speed). |
| 122 | |
| 123 | Available keys: |
| 124 | ''' + '\n'.join(option_help) |
| 125 | |
| 126 | |
| 127 | def handle(self, *args, **options): |
| 128 | self.console = component.get('ConsoleUI') |
| 129 | if options['set']: |
| 130 | return self._set_option(*args, **options) |
| 131 | else: |
| 132 | return self._get_option(*args, **options) |
| 133 | |
| 134 | |
| 135 | def _get_option(self, *args, **options): |
| 136 | |
| 137 | def on_torrents_status(status): |
| 138 | for torrentid, data in status.items(): |
| 139 | self.console.write('\n') |
| 140 | if 'name' in data: |
| 141 | self.console.write('{!info!}Name: {!input!}%s' % data.get('name')) |
| 142 | self.console.write('{!info!}ID: {!input!}%s' % torrentid) |
| 143 | for k, v in data.items(): |
| 144 | if k != 'name': |
| 145 | displayname = get_torrent_options.get(k, '???' + k) |
| 146 | self.console.write('{!info!}%s: {!input!}%s' % (displayname, v)) |
| 147 | |
| 148 | def on_torrents_status_fail(reason): |
| 149 | self.console.write('{!error!}Failed to get torrent data.') |
| 150 | |
| 151 | torrent_ids = [] |
| 152 | request_options = [] |
| 153 | |
| 154 | for arg in args: |
| 155 | if arg in set_torrent_options: |
| 156 | request_options.append(set_torrent_options[arg][1]) |
| 157 | else: |
| 158 | if arg == '*': |
| 159 | ids = self.console.match_torrent('') |
| 160 | else: |
| 161 | ids = self.console.match_torrent(arg) |
| 162 | if not ids: |
| 163 | self.console.write("{!error!}The argument '" + arg + "' is not a recognized option nor did it match any torrents") |
| 164 | return |
| 165 | torrent_ids.extend(ids) |
| 166 | |
| 167 | if not torrent_ids: |
| 168 | self.console.write('{!error!}No torrents mentioned. To request info on all torrents use "manage * [<key>...]".') |
| 169 | return |
| 170 | |
| 171 | if not request_options: |
| 172 | request_options = [ opt for opt in get_torrent_options ] |
| 173 | request_options.append('name') |
| 174 | |
| 175 | d = client.core.get_torrents_status({'id': torrent_ids}, request_options) |
| 176 | d.addCallback(on_torrents_status) |
| 177 | d.addErrback(on_torrents_status_fail) |
| 178 | return d |
| 179 | |
| 180 | |
| 181 | def _set_prioritize_first_last_pieces(self, torrent_ids, val): |
| 182 | |
| 183 | def on_option_set(status): |
| 184 | self.console.write('{!success!}Torrent option successfully updated.') |
| 185 | |
| 186 | def on_set_option_failed(reason): |
| 187 | self.console.write("{!error!}Error setting torrent option: %s" % reason) |
| 188 | |
| 189 | def on_got_info(status): |
| 190 | single_ids = [] |
| 191 | multi_ids = [] |
| 192 | error_ids = [] |
| 193 | for tid in torrent_ids: |
| 194 | if tid not in status: |
| 195 | error_ids.append(tid) |
| 196 | elif len(status[tid]['files']) > 1: |
| 197 | multi_ids.append(tid) |
| 198 | else: |
| 199 | single_ids.append(tid) |
| 200 | if multi_ids: |
| 201 | torrent_names = [ self.console.get_torrent_name(tid) for tid in multi_ids ] |
| 202 | self.console.write('{!error!}Not setting prioritize_first_last_pieces to ' + str(val) + |
| 203 | ' (multiple files in torrent) for torrents: ' + ', '.join(torrent_names)) |
| 204 | if error_ids: |
| 205 | torrent_names = [ self.console.get_torrent_name(tid) for tid in error_ids ] |
| 206 | self.console.write('{!error!}Will try to set prioritize_first_last_pieces to ' + str(val) + |
| 207 | ', but may fail silently for torrents: ' + ', '.join(torrent_names)) |
| 208 | |
| 209 | use_ids = single_ids + error_ids |
| 210 | if use_ids: |
| 211 | torrent_names = [ self.console.get_torrent_name(tid) for tid in single_ids ] |
| 212 | self.console.write('{!info!}Setting prioritize_first_last_pieces to %s for torrent(s): %s' % (val, ', '.join(torrent_names))) |
| 213 | d = client.core.set_torrent_options(use_ids, {'prioritize_first_last_pieces': val}) |
| 214 | d.addCallback(on_option_set) |
| 215 | d.addErrback(on_set_option_failed) |
| 216 | else: |
| 217 | self.console.write('{!error!}No valid torrents to set prioritize_first_last_pieces on') |
| 218 | |
| 219 | def on_info_failed(reason): |
| 220 | self.console.write("{!error!}Error getting torrent info (for setting prioritize_last_first_pieces): %s" % reason) |
| 221 | |
| 222 | d = client.core.get_torrents_status({"id": torrent_ids}, [ "files" ]) |
| 223 | d.addCallback(on_got_info) |
| 224 | d.addErrback(on_info_failed) |
| 225 | |
| 226 | def _set_option(self, *args, **options): |
| 227 | torrent_ids = [] |
| 228 | for arg in args: |
| 229 | if arg == '*': |
| 230 | ids = self.console.match_torrent('') |
| 231 | else: |
| 232 | ids = self.console.match_torrent(arg) |
| 233 | if not ids: |
| 234 | self.console.write("{!error!}The argument '" + arg + "' did not match any torrents") |
| 235 | return |
| 236 | torrent_ids.extend(ids) |
| 237 | if not torrent_ids: |
| 238 | self.console.write('{!error!}No torrents mentioned.') |
| 239 | return |
| 240 | key = options['set'][0] |
| 241 | val = options['set'][1] |
| 242 | |
| 243 | if key not in set_torrent_options: |
| 244 | self.console.write("{!error!}The key '%s' is invalid!" % key) |
| 245 | return |
| 246 | |
| 247 | val = set_torrent_options[key][0](val) |
| 248 | |
| 249 | if key == 'prioritize_first_last_pieces': |
| 250 | self._set_prioritize_first_last_pieces(torrent_ids, val) |
| 251 | return |
| 252 | |
| 253 | def on_option_set(result): |
| 254 | self.console.write('{!success!}Torrent option successfully updated.') |
| 255 | |
| 256 | def on_set_option_failed(reason): |
| 257 | self.console.write("{!error!}Error setting torrent option: %s" % reason) |
| 258 | |
| 259 | torrent_names = [ self.console.get_torrent_name(tid) for tid in torrent_ids ] |
| 260 | self.console.write('{!info!}Setting %s to %s for torrent(s): %s' % (key, val, ', '.join(torrent_names))) |
| 261 | d = client.core.set_torrent_options(torrent_ids, {key: val}) |
| 262 | d.addCallback(on_option_set) |
| 263 | d.addErrback(on_set_option_failed) |
| 264 | |
| 265 | def complete(self, text): |
| 266 | torrents = component.get('ConsoleUI').tab_complete_torrent(text) |
| 267 | options = [ x for x in set_torrent_options if x.startswith(text) ] |
| 268 | # This should probably only return options immediately after --set. |
| 269 | return torrents + options |