deluge/deluge/ui/console/main.py
Calum Lind 253eb2240b
[Console] Refactor main to use async instead of callbacks
Use the new maybe_coroutine decorator to replace callbacks with inline
async/await for cleaner code.
2023-02-27 17:38:58 +00:00

726 lines
24 KiB
Python

#
# Copyright (C) 2008-2009 Ido Abramovich <ido.deluge@gmail.com>
# Copyright (C) 2009 Andrew Resch <andrewresch@gmail.com>
#
# This file is part of Deluge and is licensed under GNU General Public License 3.0, or later, with
# the additional special exception to link portions of this program with the OpenSSL library.
# See LICENSE for more details.
#
import locale
import logging
import os
import sys
import time
from twisted.internet import defer, error, reactor
import deluge.common
import deluge.component as component
from deluge.configmanager import ConfigManager
from deluge.decorators import maybe_coroutine, overrides
from deluge.ui.client import client
from deluge.ui.console.modes.addtorrents import AddTorrents
from deluge.ui.console.modes.basemode import TermResizeHandler
from deluge.ui.console.modes.cmdline import CmdLine
from deluge.ui.console.modes.eventview import EventView
from deluge.ui.console.modes.preferences import Preferences
from deluge.ui.console.modes.torrentdetail import TorrentDetail
from deluge.ui.console.modes.torrentlist.torrentlist import TorrentList
from deluge.ui.console.utils import colors
from deluge.ui.console.widgets import StatusBars
from deluge.ui.coreconfig import CoreConfig
from deluge.ui.sessionproxy import SessionProxy
log = logging.getLogger(__name__)
DEFAULT_CONSOLE_PREFS = {
'ring_bell': False,
'first_run': True,
'language': '',
'torrentview': {
'sort_primary': 'queue',
'sort_secondary': 'name',
'show_sidebar': True,
'sidebar_width': 25,
'separate_complete': True,
'move_selection': True,
'columns': {},
},
'addtorrents': {
'show_misc_files': False, # TODO: Showing/hiding this
'show_hidden_folders': False, # TODO: Showing/hiding this
'sort_column': 'date',
'reverse_sort': True,
'last_path': '~',
},
'cmdline': {
'ignore_duplicate_lines': False,
'third_tab_lists_all': False,
'torrents_per_tab_press': 15,
'save_command_history': True,
},
}
class MockConsoleLog:
def write(self, data):
pass
def flush(self):
pass
class ConsoleUI(component.Component, TermResizeHandler):
def __init__(self, options, cmds, log_stream):
component.Component.__init__(self, 'ConsoleUI')
TermResizeHandler.__init__(self)
self.options = options
self.log_stream = log_stream
# keep track of events for the log view
self.events = []
self.torrents = []
self.statusbars = None
self.modes = {}
self.active_mode = None
self.initialized = False
try:
locale.setlocale(locale.LC_ALL, '')
self.encoding = locale.getpreferredencoding()
except locale.Error:
self.encoding = sys.getdefaultencoding()
log.debug('Using encoding: %s', self.encoding)
# start up the session proxy
self.sessionproxy = SessionProxy()
client.set_disconnect_callback(self.on_client_disconnect)
# Set the interactive flag to indicate where we should print the output
self.interactive = True
self._commands = cmds
self.coreconfig = CoreConfig()
def start_ui(self):
"""Start the console UI.
Note: When running console UI reactor.run() will be called which
effectively blocks this function making the return value
insignificant. However, when running unit tests, the reacor is
replaced by a mock object, leaving the return deferred object
necessary for the tests to run properly.
Returns:
Deferred: If valid commands are provided, a deferred that fires when
all commands are executed. Else None is returned.
"""
if self.options.parsed_cmds:
# Non-Interactive mode
self.interactive = False
if not self._commands:
print('No valid console commands found')
return
deferred = self.exec_args(self.options)
reactor.run()
return deferred
# Interactive
# We use the curses.wrapper function to prevent the console from getting
# messed up if an uncaught exception is experienced.
try:
from curses import wrapper
except ImportError:
wrapper = None
if deluge.common.windows_check() and not wrapper:
print(
"""\nDeluge-console does not run in interactive mode on Windows. \n
Please use commands from the command line, e.g.:\n
deluge-console.exe help
deluge-console.exe info
deluge-console.exe "add --help"
deluge-console.exe "add -p c:\\mytorrents c:\\new.torrent"
"""
)
# We don't ever want log output to terminal when running in
# interactive mode, so insert a dummy here
self.log_stream.out = MockConsoleLog()
# Set Esc key delay to 0 to avoid a very annoying delay
# due to curses waiting in case of other key are pressed
# after ESC is pressed
os.environ.setdefault('ESCDELAY', '0')
wrapper(self.run)
@maybe_coroutine
async def quit(self):
if client.connected():
await client.disconnect()
try:
reactor.stop()
except error.ReactorNotRunning:
pass
@maybe_coroutine
async def exec_args(self, options):
"""Execute console commands from command line."""
from deluge.ui.console.cmdline.command import Commander
commander = Commander(self._commands)
try:
if not self.interactive and options.parsed_cmds[0].command == 'connect':
await commander.exec_command(options.parsed_cmds.pop(0))
else:
daemon_options = (
options.daemon_addr,
options.daemon_port,
options.daemon_user,
options.daemon_pass,
)
log.info(
'Connect: host=%s, port=%s, username=%s',
*daemon_options[0:3],
)
await client.connect(*daemon_options)
except Exception as reason:
print(
'Could not connect to daemon: %s:%s\n %s'
% (options.daemon_addr, options.daemon_port, reason)
)
commander.do_command('quit')
await self.start_console()
# Wait for RPCs in start() to finish before processing commands.
await self.started_deferred
for cmd in options.parsed_cmds:
if cmd.command in ('quit', 'exit'):
break
await commander.exec_command(cmd)
commander.do_command('quit')
def run(self, stdscr):
"""This method is called by the curses.wrapper to start the mainloop and screen.
Args:
stdscr (_curses.curses window): curses screen passed in from curses.wrapper.
"""
# We want to do an interactive session, so start up the curses screen and
# pass it the function that handles commands
colors.init_colors()
self.stdscr = stdscr
self.config = ConfigManager(
'console.conf', defaults=DEFAULT_CONSOLE_PREFS, file_version=2
)
self.config.run_converter((0, 1), 2, self._migrate_config_1_to_2)
self.statusbars = StatusBars()
from deluge.ui.console.modes.connectionmanager import ConnectionManager
self.register_mode(ConnectionManager(stdscr, self.encoding), set_mode=True)
torrentlist = self.register_mode(TorrentList(self.stdscr, self.encoding))
self.register_mode(CmdLine(self.stdscr, self.encoding))
self.register_mode(EventView(torrentlist, self.stdscr, self.encoding))
self.register_mode(
TorrentDetail(torrentlist, self.stdscr, self.config, self.encoding)
)
self.register_mode(
Preferences(torrentlist, self.stdscr, self.config, self.encoding)
)
self.register_mode(
AddTorrents(torrentlist, self.stdscr, self.config, self.encoding)
)
self.eventlog = EventLog()
self.active_mode.topbar = (
'{!status!}Deluge ' + deluge.common.get_version() + ' Console'
)
self.active_mode.bottombar = '{!status!}'
self.active_mode.refresh()
# Start the twisted mainloop
reactor.run()
@overrides(TermResizeHandler)
def on_terminal_size(self, *args):
rows, cols = super().on_terminal_size(args)
for mode in self.modes:
self.modes[mode].on_resize(rows, cols)
def register_mode(self, mode, set_mode=False):
self.modes[mode.mode_name] = mode
if set_mode:
self.set_mode(mode.mode_name)
return mode
def set_mode(self, mode_name, refresh=False):
log.debug('Setting console mode: %s', mode_name)
mode = self.modes.get(mode_name, None)
if mode is None:
log.error('Non-existent mode requested: %s', mode_name)
return
self.stdscr.erase()
if self.active_mode:
self.active_mode.pause()
d = component.pause([self.active_mode.mode_name])
def on_mode_paused(result, mode, *args):
from deluge.ui.console.widgets.popup import PopupsHandler
if isinstance(mode, PopupsHandler):
if mode.popup is not None:
# If popups are not removed, they are still referenced in the memory
# which can cause issues as the popup's screen will not be destroyed.
# This can lead to the popup border being visible for short periods
# while the current modes' screen is repainted.
log.error(
'Mode "%s" still has popups available after being paused.'
' Ensure all popups are removed on pause!',
mode.popup.title,
)
d.addCallback(on_mode_paused, self.active_mode)
reactor.removeReader(self.active_mode)
self.active_mode = mode
self.statusbars.screen = self.active_mode
# The Screen object is designed to run as a twisted reader so that it
# can use twisted's select poll for non-blocking user input.
reactor.addReader(self.active_mode)
self.stdscr.clear()
if self.active_mode._component_state == 'Stopped':
component.start([self.active_mode.mode_name])
else:
component.resume([self.active_mode.mode_name])
mode.resume()
if refresh:
mode.refresh()
return mode
def switch_mode(self, func, error_smg):
def on_stop(arg):
if arg and True in arg[0]:
func()
else:
self.messages.append(('Error', error_smg))
component.stop(['TorrentList']).addCallback(on_stop)
def is_active_mode(self, mode):
return mode == self.active_mode
@maybe_coroutine
async def start_components(self):
if not self.interactive:
return await component.start(['SessionProxy', 'ConsoleUI', 'CoreConfig'])
await component.start()
component.pause(
[
'TorrentList',
'EventView',
'AddTorrents',
'TorrentDetail',
'Preferences',
]
)
@maybe_coroutine
async def start_console(self):
self.started_deferred = defer.Deferred()
if self.initialized:
await component.stop(['SessionProxy'])
await component.start(['SessionProxy'])
else:
self.initialized = True
await self.start_components()
@maybe_coroutine
async def start(self):
result = await client.core.get_session_state()
# Maintain a list of (torrent_id, name) for use in tab completion
self.torrents = []
self.events = []
torrents = await client.core.get_torrents_status({'id': result}, ['name'])
for torrent_id, status in torrents.items():
self.torrents.append((torrent_id, status['name']))
self.started_deferred.callback(True)
# Register event handlers to keep the torrent list up-to-date
client.register_event_handler('TorrentAddedEvent', self.on_torrent_added_event)
client.register_event_handler(
'TorrentRemovedEvent', self.on_torrent_removed_event
)
def on_torrent_added_event(self, event, from_state=False):
def on_torrent_status(status):
self.torrents.append((event, status['name']))
client.core.get_torrent_status(event, ['name']).addCallback(on_torrent_status)
def on_torrent_removed_event(self, event):
for index, (tid, name) in enumerate(self.torrents):
if event == tid:
del self.torrents[index]
def match_torrents(self, strings):
torrent_ids = []
for s in strings:
torrent_ids.extend(self.match_torrent(s))
return list(set(torrent_ids))
def match_torrent(self, string):
"""
Returns a list of torrent_id matches for the string. It will search both
torrent_ids and torrent names, but will only return torrent_ids.
:param string: str, the string to match on
:returns: list of matching torrent_ids. Will return an empty list if
no matches are found.
"""
deluge.common.decode_bytes(string, self.encoding)
if string == '*' or string == '':
return [tid for tid, name in self.torrents]
match_func = '__eq__'
if string.startswith('*'):
string = string[1:]
match_func = 'endswith'
if string.endswith('*'):
match_func = '__contains__' if match_func == 'endswith' else 'startswith'
string = string[:-1]
matches = []
for tid, name in self.torrents:
deluge.common.decode_bytes(name, self.encoding)
if getattr(tid, match_func, None)(string) or getattr(
name, match_func, None
)(string):
matches.append(tid)
return matches
def get_torrent_name(self, torrent_id):
for tid, name in self.torrents:
if torrent_id == tid:
return name
return None
def set_batch_write(self, batch):
if self.interactive and isinstance(
self.active_mode, deluge.ui.console.modes.cmdline.CmdLine
):
return self.active_mode.set_batch_write(batch)
def tab_complete_torrent(self, line):
if self.interactive and isinstance(
self.active_mode, deluge.ui.console.modes.cmdline.CmdLine
):
return self.active_mode.tab_complete_torrent(line)
def tab_complete_path(
self, line, path_type='file', ext='', sort='name', dirs_first=True
):
if self.interactive and isinstance(
self.active_mode, deluge.ui.console.modes.cmdline.CmdLine
):
return self.active_mode.tab_complete_path(
line, path_type=path_type, ext=ext, sort=sort, dirs_first=dirs_first
)
def on_client_disconnect(self):
component.stop()
def write(self, s):
if self.interactive:
if isinstance(self.active_mode, deluge.ui.console.modes.cmdline.CmdLine):
self.active_mode.write(s)
else:
component.get('CmdLine').add_line(s, False)
self.events.append(s)
else:
print(colors.strip_colors(s))
def write_event(self, s):
if self.interactive:
if isinstance(self.active_mode, deluge.ui.console.modes.cmdline.CmdLine):
self.events.append(s)
self.active_mode.write(s)
else:
component.get('CmdLine').add_line(s, False)
self.events.append(s)
else:
print(colors.strip_colors(s))
def _migrate_config_1_to_2(self, config):
"""Create better structure by moving most settings out of dict root
and into sub categories. Some keys are also renamed to be consistent
with other UIs.
"""
def move_key(source, dest, source_key, dest_key=None):
if dest_key is None:
dest_key = source_key
dest[dest_key] = source[source_key]
del source[source_key]
# These are moved to 'torrentview' sub dict
for k in [
'sort_primary',
'sort_secondary',
'move_selection',
'separate_complete',
]:
move_key(config, config['torrentview'], k)
# These are moved to 'addtorrents' sub dict
for k in [
'show_misc_files',
'show_hidden_folders',
'sort_column',
'reverse_sort',
'last_path',
]:
move_key(config, config['addtorrents'], 'addtorrents_%s' % k, dest_key=k)
# These are moved to 'cmdline' sub dict
for k in [
'ignore_duplicate_lines',
'torrents_per_tab_press',
'third_tab_lists_all',
]:
move_key(config, config['cmdline'], k)
move_key(
config,
config['cmdline'],
'save_legacy_history',
dest_key='save_command_history',
)
# Add key for localization
config['language'] = DEFAULT_CONSOLE_PREFS['language']
# Migrate column settings
columns = [
'queue',
'size',
'state',
'progress',
'seeds',
'peers',
'downspeed',
'upspeed',
'eta',
'ratio',
'avail',
'added',
'tracker',
'savepath',
'downloaded',
'uploaded',
'remaining',
'owner',
'downloading_time',
'seeding_time',
'completed',
'seeds_peers_ratio',
'complete_seen',
'down_limit',
'up_limit',
'shared',
'name',
]
column_name_mapping = {
'downspeed': 'download_speed',
'upspeed': 'upload_speed',
'added': 'time_added',
'savepath': 'download_location',
'completed': 'completed_time',
'complete_seen': 'last_seen_complete',
'down_limit': 'max_download_speed',
'up_limit': 'max_upload_speed',
'downloading_time': 'active_time',
}
from deluge.ui.console.modes.torrentlist.torrentview import default_columns
# These are moved to 'torrentview.columns' sub dict
for k in columns:
column_name = column_name_mapping.get(k, k)
config['torrentview']['columns'][column_name] = {}
if k == 'name':
config['torrentview']['columns'][column_name]['visible'] = True
else:
move_key(
config,
config['torrentview']['columns'][column_name],
'show_%s' % k,
dest_key='visible',
)
move_key(
config,
config['torrentview']['columns'][column_name],
'%s_width' % k,
dest_key='width',
)
config['torrentview']['columns'][column_name]['order'] = default_columns[
column_name
]['order']
return config
class EventLog(component.Component):
"""
Prints out certain events as they are received from the core.
"""
def __init__(self):
component.Component.__init__(self, 'EventLog')
self.console = component.get('ConsoleUI')
self.prefix = '{!event!}* [%H:%M:%S] '
self.date_change_format = 'On {!yellow!}%a, %d %b %Y{!input!} %Z:'
client.register_event_handler('TorrentAddedEvent', self.on_torrent_added_event)
client.register_event_handler(
'PreTorrentRemovedEvent', self.on_torrent_removed_event
)
client.register_event_handler(
'TorrentStateChangedEvent', self.on_torrent_state_changed_event
)
client.register_event_handler(
'TorrentFinishedEvent', self.on_torrent_finished_event
)
client.register_event_handler(
'NewVersionAvailableEvent', self.on_new_version_available_event
)
client.register_event_handler(
'SessionPausedEvent', self.on_session_paused_event
)
client.register_event_handler(
'SessionResumedEvent', self.on_session_resumed_event
)
client.register_event_handler(
'ConfigValueChangedEvent', self.on_config_value_changed_event
)
client.register_event_handler(
'PluginEnabledEvent', self.on_plugin_enabled_event
)
client.register_event_handler(
'PluginDisabledEvent', self.on_plugin_disabled_event
)
self.previous_time = time.localtime(0)
def on_torrent_added_event(self, torrent_id, from_state):
if from_state:
return
def on_torrent_status(status):
self.write(
'{!green!}Torrent Added: {!info!}%s ({!cyan!}%s{!info!})'
% (status['name'], torrent_id)
)
# Write out what state the added torrent took
self.on_torrent_state_changed_event(torrent_id, status['state'])
client.core.get_torrent_status(torrent_id, ['name', 'state']).addCallback(
on_torrent_status
)
def on_torrent_removed_event(self, torrent_id):
self.write(
'{!red!}Torrent Removed: {!info!}%s ({!cyan!}%s{!info!})'
% (self.console.get_torrent_name(torrent_id), torrent_id)
)
def on_torrent_state_changed_event(self, torrent_id, state):
# It's probably a new torrent, ignore it
if not state:
return
# Modify the state string color
if state in colors.state_color:
state = colors.state_color[state] + state
t_name = self.console.get_torrent_name(torrent_id)
# Again, it's most likely a new torrent
if not t_name:
return
self.write(f'{state}: {{!info!}}{t_name} ({{!cyan!}}{torrent_id}{{!info!}})')
def on_torrent_finished_event(self, torrent_id):
if component.get('TorrentList').config['ring_bell']:
import curses.beep
curses.beep()
self.write(
'{!info!}Torrent Finished: %s ({!cyan!}%s{!info!})'
% (self.console.get_torrent_name(torrent_id), torrent_id)
)
def on_new_version_available_event(self, version):
self.write('{!input!}New Deluge version available: {!info!}%s' % (version))
def on_session_paused_event(self):
self.write('{!input!}Session Paused')
def on_session_resumed_event(self):
self.write('{!green!}Session Resumed')
def on_config_value_changed_event(self, key, value):
color = '{!white,black,bold!}'
try:
color = colors.type_color[type(value)]
except KeyError:
pass
self.write(f'ConfigValueChanged: {{!input!}}{key}: {color}{value}')
def write(self, s):
current_time = time.localtime()
date_different = False
for field in ['tm_mday', 'tm_mon', 'tm_year']:
c = getattr(current_time, field)
p = getattr(self.previous_time, field)
if c != p:
date_different = True
if date_different:
string = time.strftime(self.date_change_format)
self.console.write_event(' ')
self.console.write_event(string)
p = time.strftime(self.prefix)
self.console.write_event(p + s)
self.previous_time = current_time
def on_plugin_enabled_event(self, name):
self.write('PluginEnabled: {!info!}%s' % name)
def on_plugin_disabled_event(self, name):
self.write('PluginDisabled: {!info!}%s' % name)