""" Contains helpers for interacting with the console. Includes definition for the class that is used to run the console. """ import atexit import cmd2 import os import readline #import string import shlex import sys from .colors import Colors from .proxy import MessageError ################### ## Helper Functions def print_errors(func): def catch(*args, **kwargs): try: func(*args, **kwargs) except (CommandError, MessageError) as e: print(str(e)) return catch def interface_loop(client): cons = Cmd(client=client) load_interface(cons) sys.argv = [] cons.cmdloop() def load_interface(cons): from .interface import test, view, decode, misc, context, mangle, macros, tags test.load_cmds(cons) view.load_cmds(cons) decode.load_cmds(cons) misc.load_cmds(cons) context.load_cmds(cons) mangle.load_cmds(cons) macros.load_cmds(cons) tags.load_cmds(cons) ########## ## Classes class SessionEnd(Exception): pass class CommandError(Exception): pass class Cmd(cmd2.Cmd): """ An object representing the console interface. Provides methods to add commands and aliases to the console. Implemented as a hack around cmd2.Cmd """ def __init__(self, *args, **kwargs): # the \x01/\x02 are to make the prompt behave properly with the readline library self.prompt = 'pappy\x01' + Colors.YELLOW + '\x02> \x01' + Colors.ENDC + '\x02' self.debug = True self.histsize = 0 if 'histsize' in kwargs: self.histsize = kwargs['histsize'] del kwargs['histsize'] if 'client' not in kwargs: raise Exception("client argument is required") self.client = kwargs['client'] self.client.console = self del kwargs['client'] self._cmds = {} self._aliases = {} atexit.register(self.save_histfile) readline.set_history_length(self.histsize) if os.path.exists('cmdhistory'): if self.histsize != 0: readline.read_history_file('cmdhistory') else: os.remove('cmdhistory') cmd2.Cmd.__init__(self, *args, **kwargs) def __dir__(self): # Hack to get cmd2 to detect that we can run a command ret = set(dir(self.__class__)) ret.update(self.__dict__.keys()) ret.update(['do_'+k for k in self._cmds.keys()]) ret.update(['help_'+k for k in self._cmds.keys()]) ret.update(['complete_'+k for k, v in self._cmds.items() if self._cmds[k][1]]) for k, v in self._aliases.items(): ret.add('do_' + k) ret.add('help_' + k) if self._cmds[self._aliases[k]][1]: ret.add('complete_'+k) return sorted(ret) def __getattr__(self, attr): def gen_helpfunc(func): def f(): if not func.__doc__: to_print = 'No help exists for function' else: lines = func.__doc__.splitlines() if len(lines) > 0 and lines[0] == '': lines = lines[1:] if len(lines) > 0 and lines[-1] == '': lines = lines[-1:] to_print = '\n'.join(l.lstrip() for l in lines) aliases = set() aliases.add(attr[5:]) for i in range(2): for k, v in self._aliases.items(): if k in aliases or v in aliases: aliases.add(k) aliases.add(v) to_print += '\nAliases: ' + ', '.join(aliases) print(to_print) return f def gen_dofunc(func, client): def f(line): args = shlex.split(line) func(client, args) return print_errors(f) if attr.startswith('do_'): command = attr[3:] if command in self._cmds: return gen_dofunc(self._cmds[command][0], self.client) elif command in self._aliases: real_command = self._aliases[command] if real_command in self._cmds: return gen_dofunc(self._cmds[real_command][0], self.client) elif attr.startswith('help_'): command = attr[5:] if command in self._cmds: return gen_helpfunc(self._cmds[command][0]) elif command in self._aliases: real_command = self._aliases[command] if real_command in self._cmds: return gen_helpfunc(self._cmds[real_command][0]) elif attr.startswith('complete_'): command = attr[9:] if command in self._cmds: if self._cmds[command][1]: return self._cmds[command][1] elif command in self._aliases: real_command = self._aliases[command] if real_command in self._cmds: if self._cmds[real_command][1]: return self._cmds[real_command][1] raise AttributeError(attr) def run_args(self, args): command = args[0] if command in self._cmds: self._cmds[command][0](self.client, args[1:]) elif command in self._aliases: real_command = self._aliases[command] if real_command in self._cmds: self._cmds[real_command][0](self.client, args[1:]) def save_histfile(self): # Write the command to the history file if self.histsize != 0: readline.set_history_length(self.histsize) readline.write_history_file('cmdhistory') def get_names(self): # Hack to get cmd to recognize do_/etc functions as functions for things # like autocomplete return dir(self) def set_cmd(self, command, func, autocomplete_func=None): """ Add a command to the console. """ self._cmds[command] = (func, autocomplete_func) def set_cmds(self, cmd_dict): """ Set multiple commands from a dictionary. Format is: {'command': (do_func, autocomplete_func)} Use autocomplete_func=None for no autocomplete function """ for command, vals in cmd_dict.items(): do_func, ac_func = vals self.set_cmd(command, do_func, ac_func) def add_alias(self, command, alias): """ Add an alias for a command. ie add_alias("foo", "f") will let you run the 'foo' command with 'f' """ if command not in self._cmds: raise KeyError() self._aliases[alias] = command def add_aliases(self, alias_list): """ Pass in a list of tuples to add them all as aliases. ie add_aliases([('foo', 'f'), ('foo', 'fo')]) will add 'f' and 'fo' as aliases for 'foo' """ for command, alias in alias_list: self.add_alias(command, alias)