def __init__(self, check_rc_file=True, stdin=None, stdout=None, log_handler=None): OverrideCmd.__init__(self, stdin=stdin, stdout=stdout) if stdin is not None: self.use_rawinput = False # No 'use_rawinput' will cause problems with the ipy command so disable it for now self.__disabled_commands__.append('ipy') if not its.on_linux: self.__hidden_commands__.append('prep_driver') self.__hidden_commands__.append('cd') self.__hidden_commands__.append('exploit') self.last_module = None self.log_handler = log_handler if self.log_handler is None: self.__disabled_commands__.append('logging') self.logger = logging.getLogger(self.__package__ + '.interpreter') self.frmwk = Framework(stdout=stdout) self.print_exception = self.frmwk.print_exception self.print_error = self.frmwk.print_error self.print_good = self.frmwk.print_good self.print_line = self.frmwk.print_line self.print_status = self.frmwk.print_status if check_rc_file: check_rc_file = os.path.join(self.frmwk.directories.user_data, 'console.rc') if os.path.isfile(check_rc_file) and os.access( check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) elif isinstance(check_rc_file, str): if os.path.isfile(check_rc_file) and os.access( check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) else: self.logger.error('could not access resource file: ' + check_rc_file) self.print_error('Could not access resource file: ' + check_rc_file) try: import readline readline.read_history_file(self.frmwk.directories.user_data + 'history.txt') readline.set_completer_delims( readline.get_completer_delims().replace('/', '')) except (ImportError, IOError): pass
def __init__(self, check_rc_file=True, stdin=None, stdout=None, log_handler=None): OverrideCmd.__init__(self, stdin=stdin, stdout=stdout) if stdin is not None: self.use_rawinput = False # No 'use_rawinput' will cause problems with the ipy command so disable it for now self.__disabled_commands__.append('ipy') if not its.on_linux: self.__hidden_commands__.append('prep_driver') self.__hidden_commands__.append('cd') self.__hidden_commands__.append('exploit') self.last_module = None self.log_handler = log_handler if self.log_handler is None: self.__disabled_commands__.append('logging') self.logger = logging.getLogger(self.__package__ + '.interpreter') self.frmwk = Framework(stdout=stdout) self.print_exception = self.frmwk.print_exception self.print_error = self.frmwk.print_error self.print_good = self.frmwk.print_good self.print_line = self.frmwk.print_line self.print_status = self.frmwk.print_status if check_rc_file: if check_rc_file: check_rc_file = self.frmwk.directories.user_data + 'console.rc' if os.path.isfile(check_rc_file) and os.access(check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) elif isinstance(check_rc_file, str): if os.path.isfile(check_rc_file) and os.access(check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) else: self.logger.error('could not access resource file: ' + check_rc_file) self.print_error('Could not access resource file: ' + check_rc_file) try: import readline readline.read_history_file(self.frmwk.directories.user_data + 'history.txt') readline.set_completer_delims(readline.get_completer_delims().replace('/', '')) except (ImportError, IOError): pass
class InteractiveInterpreter(OverrideCmd): __doc__ = 'The core interpreter for the program' __name__ = 'termineter' prompt = __name__ + ' > ' ruler = '+' doc_header = 'Type help <command> For Information\nList Of Available Commands:' def __init__(self, check_rc_file=True, stdin=None, stdout=None, log_handler=None): OverrideCmd.__init__(self, stdin=stdin, stdout=stdout) if stdin is not None: self.use_rawinput = False # No 'use_rawinput' will cause problems with the ipy command so disable it for now self.__disabled_commands__.append('ipy') if not its.on_linux: self.__hidden_commands__.append('prep_driver') self.__hidden_commands__.append('cd') self.__hidden_commands__.append('exploit') self.last_module = None self.log_handler = log_handler if self.log_handler is None: self.__disabled_commands__.append('logging') self.logger = logging.getLogger(self.__package__ + '.interpreter') self.frmwk = Framework(stdout=stdout) self.print_exception = self.frmwk.print_exception self.print_error = self.frmwk.print_error self.print_good = self.frmwk.print_good self.print_line = self.frmwk.print_line self.print_status = self.frmwk.print_status if check_rc_file: check_rc_file = os.path.join(self.frmwk.directories.user_data, 'console.rc') if os.path.isfile(check_rc_file) and os.access( check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) elif isinstance(check_rc_file, str): if os.path.isfile(check_rc_file) and os.access( check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) else: self.logger.error('could not access resource file: ' + check_rc_file) self.print_error('Could not access resource file: ' + check_rc_file) try: import readline readline.read_history_file(self.frmwk.directories.user_data + 'history.txt') readline.set_completer_delims( readline.get_completer_delims().replace('/', '')) except (ImportError, IOError): pass @property def intro(self): intro = os.linesep intro += ' ______ _ __ ' + os.linesep intro += ' /_ __/__ ______ _ (_)__ ___ / /____ ____' + os.linesep intro += ' / / / -_) __/ \' \/ / _ \/ -_) __/ -_) __/' + os.linesep intro += ' /_/ \__/_/ /_/_/_/_/_//_/\__/\__/\__/_/ ' + os.linesep intro += os.linesep fmt_string = " <[ {0:<18} {1:>18}" intro += fmt_string.format(self.__name__, 'v' + __version__ + '') + os.linesep intro += fmt_string.format('model:', __codename__) + os.linesep intro += fmt_string.format('loaded modules:', len( self.frmwk.modules)) + os.linesep #if self.frmwk.rfcat_available: # intro += fmt_string.format('rfcat:', 'enabled') + os.linesep #else: # intro += fmt_string.format('rfcat:', 'disabled') + os.linesep return intro @property def prompt(self): if self.frmwk.current_module: if self.frmwk.use_colors: return self.__name__ + ' (\033[1;33m' + self.frmwk.current_module.name + '\033[1;m) > ' else: return self.__name__ + ' (' + self.frmwk.current_module.name + ') > ' else: return self.__name__ + ' > ' def run_rc_file(self, rc_file): if os.path.isfile(rc_file) and os.access(rc_file, os.R_OK): self.logger.info('processing "' + rc_file + '" for commands') for line in open(rc_file, 'r'): line = line.strip() if not len(line) or line[0] == '#': continue if line.startswith('print_'): line = line[6:] print_type, message = line.split(' ', 1) if print_type in ('error', 'good', 'line', 'status'): getattr(self, 'print_' + print_type)(message) continue self.print_line(self.prompt + line.strip()) self.onecmd(line.strip()) else: self.logger.error('invalid rc file: ' + rc_file) return False return True @staticmethod def serve(addr, run_once=False, log_level=None, use_ssl=False, ssl_cert=None): import ssl __package__ = '.'.join( InteractiveInterpreter.__module__.split('.')[:-1]) logger = logging.getLogger(__package__ + '.interpreter.server') srv_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) srv_sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) srv_sock.bind(addr) logger.debug('listening for connections on: ' + addr[0] + ':' + str(addr[1])) srv_sock.listen(1) while True: try: (clt_sock, clt_addr) = srv_sock.accept() except KeyboardInterrupt: break logger.info('received connection from: ' + clt_addr[0] + ':' + str(clt_addr[1])) if use_ssl: ssl_sock = ssl.wrap_socket(clt_sock, server_side=True, certfile=ssl_cert) ins = ssl_sock.makefile('r', 1) outs = ssl_sock.makefile('w', 1) else: ins = clt_sock.makefile('r', 1) outs = clt_sock.makefile('w', 1) log_stream = logging.StreamHandler(outs) if log_level is not None: log_stream.setLevel(log_level) log_stream.setFormatter( logging.Formatter("%(levelname)-8s %(message)s")) logging.getLogger('').addHandler(log_stream) interpreter = InteractiveInterpreter(check_rc_file=False, stdin=ins, stdout=outs) try: interpreter.cmdloop() except socket.error: log_stream.close() logging.getLogger('').removeHandler(log_stream) logger.warning( 'received a socket error during the main interpreter loop') continue log_stream.flush() log_stream.close() logging.getLogger('').removeHandler(log_stream) outs.close() ins.close() clt_sock.shutdown(socket.SHUT_RDWR) clt_sock.close() del clt_sock if run_once: break srv_sock.shutdown(socket.SHUT_RDWR) srv_sock.close() def do_back(self, args): """Stop using a module""" self.frmwk.current_module = None def do_banner(self, args): """Print the banner""" self.print_line(self.intro) def do_cd(self, args): """Change the current working directory""" path = args.split(' ')[0] if not path: self.print_error('must specify a path') return if not os.path.isdir(path): self.print_error('invalid path') return os.chdir(path) def complete_cd(self, text, line, begidx, endidx): return complete_path(text, allow_files=False) def do_connect(self, args): """Connect the serial interface""" args = shlex.split(args) if self.frmwk.is_serial_connected(): self.print_status('Already connected') return missing_options = self.frmwk.options.get_missing_options() if missing_options: self.print_error('The following options must be set: ' + ', '.join(missing_options)) return try: self.frmwk.serial_connect() except Exception as error: self.print_exception(error) return self.print_good('Successfully connected and the device is responding') if len(args) and args[0] == '-l': if self.frmwk.serial_login(): self.print_good('Successfully authenticated to the device') else: self.print_error('Failed to authenticate to the device') def do_disconnect(self, args): """Disconnect the serial interface""" args = shlex.split(args) if not self.frmwk.is_serial_connected(): self.print_error('Not connected') return result = self.frmwk.serial_disconnect() if result: self.print_good('Successfully disconnected') else: self.print_error( 'An error occurred while closing the serial interface') if len(args) and args[0] == '-r': missing_options = self.frmwk.options.get_missing_options() if missing_options: self.print_error('The following options must be set: ' + ', '.join(missing_options)) return try: self.frmwk.serial_connect() except Exception as error: self.print_exception(error) return self.print_good( 'Successfully reconnected and the device is responding') def do_exit(self, args): """Exit The Interpreter""" quotes = ('I\'ll be back.', 'Hasta la vista, baby.', 'Come with me if you want to live.', 'Where\'s John Connor?') self.logger.info('received exit command, now exiting') self.print_status(random.choice(quotes)) try: import readline readline.write_history_file(self.frmwk.directories.user_data + 'history.txt') except (ImportError, IOError): pass return True def do_exploit(self, args): """Run the currently selected module""" self.do_run(args) def do_help(self, args): super(InteractiveInterpreter, self).do_help(args) self.print_line('') def do_logging(self, args): """Set and show logging options""" args = shlex.split(args) if not args: args.append('show') elif not args[0] in ['show', 'set', '-h']: self.print_error('Invalid parameter "' + args[0] + '", use "logging -h" for more information') return if args[0] == '-h': self.print_status( 'Valid parameters for the "logging" command are: show, set') return elif self.log_handler is None: self.print_error('No log handler is defined') return if args[0] == 'show': loglvl = self.log_handler.level self.print_status('Effective logging level is: ' + ({ 10: 'DEBUG', 20: 'INFO', 30: 'WARNING', 40: 'ERROR', 50: 'CRITICAL' }.get(loglvl) or 'UNKNOWN')) elif args[0] == 'set': if len(args) == 1: self.print_error( 'Missing log level, valid options are: debug, info, warning, error, critical' ) return new_lvl = args[1].upper() if new_lvl in ['DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL']: self.log_handler.setLevel(getattr(logging, new_lvl)) self.print_status( 'Successfully changed the logging level to: ' + new_lvl) else: self.print_error( 'Missing log level, valid options are: debug, info, warning, error, critical' ) def complete_logging(self, text, line, begidx, endidx): return [ i for i in ['set', 'show', 'debug', 'info', 'warning', 'error', 'critical'] if i.startswith(text.lower()) ] def do_info(self, args): """Show module information""" args = shlex.split(args) if not args and self.frmwk.current_module is None: self.print_error('Must select module to show information') return if len(args) and args[0]: if args[0] in self.frmwk.modules.keys(): module = self.frmwk.modules[args[0]] else: self.print_error('Invalid module name') return else: module = self.frmwk.current_module self.print_line('') self.print_line(' Name: ' + module.name) if len(module.author) == 1: self.print_line(' Author: ' + module.author[0]) elif len(module.author) > 1: self.print_line(' Authors: ' + module.author[0]) for additional_author in module.author[1:]: self.print_line(' ' + additional_author) self.print_line(' Version: ' + str(module.version)) self.print_line('') self.print_line('Basic Options: ') longest_name = 16 longest_value = 10 for option_name, option_def in module.options.items(): longest_name = max(longest_name, len(option_name)) longest_value = max(longest_value, len(str(module.options[option_name]))) fmt_string = " {0:<" + str(longest_name) + "} {1:<" + str( longest_value) + "} {2}" self.print_line(fmt_string.format('Name', 'Value', 'Description')) self.print_line(fmt_string.format('----', '-----', '-----------')) for option_name in module.options.keys(): option_value = module.options[option_name] if option_value is None: option_value = '' option_desc = module.options.get_option_help(option_name) self.print_line( fmt_string.format(option_name, str(option_value), option_desc)) self.print_line('') self.print_line('Description:') for line in textwrap.wrap(textwrap.dedent(module.detailed_description), 78): self.print_line(' ' + line) self.print_line('') def complete_info(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)] def do_ipy(self, args): """Start an interactive Python interpreter""" import c1218.data import c1219.data from c1219.access.general import C1219GeneralAccess from c1219.access.security import C1219SecurityAccess from c1219.access.log import C1219LogAccess from c1219.access.telephone import C1219TelephoneAccess vars = { '__version__': __version__, 'C1218Packet': c1218.data.C1218Packet, 'C1218ReadRequest': c1218.data.C1218ReadRequest, 'C1218WriteRequest': c1218.data.C1218WriteRequest, 'C1219ProcedureInit': c1219.data.C1219ProcedureInit, 'C1219GeneralAccess': C1219GeneralAccess, 'C1219SecurityAccess': C1219SecurityAccess, 'C1219LogAccess': C1219LogAccess, 'C1219TelephoneAccess': C1219TelephoneAccess, 'frmwk': self.frmwk, 'os': os, 'sys': sys } banner = 'Python ' + sys.version + ' on ' + sys.platform + os.linesep banner += os.linesep banner += 'The framework instance is in the \'frmwk\' variable.' if self.frmwk.is_serial_connected(): vars['conn'] = self.frmwk.serial_connection banner += os.linesep banner += 'The connection instance is in the \'conn\' variable.' pyconsole = code.InteractiveConsole(vars) savestdin = os.dup(sys.stdin.fileno()) savestdout = os.dup(sys.stdout.fileno()) savestderr = os.dup(sys.stderr.fileno()) try: pyconsole.interact(banner) except SystemExit: sys.stdin = os.fdopen(savestdin, 'r', 0) sys.stdout = os.fdopen(savestdout, 'w', 0) sys.stderr = os.fdopen(savestderr, 'w', 0) def do_prep_driver(self, args): """Prep the optical probe driver""" args = shlex.split(args) if len(args) != 2: self.print_line('Usage:') self.print_line(' prep_driver VVVV PPPP') self.print_line('') self.print_line( 'Where VVVV and PPPP are the 4 hex digits of the vendor and product IDs respectively' ) return if os.getuid(): self.print_error('Must be running as root to prep the driver') return vendor, product = args if vendor.startswith('0x'): vendor = vendor[2:] if product.startswith('0x'): product = product[2:] linux_kernel_version = platform.uname()[2].split('.')[:2] linux_kernel_version = tuple( int(part) for part in linux_kernel_version) if linux_kernel_version < (3, 12): proc_args = [ 'modprobe', 'ftdi-sio', "vendor=0x{0}".format(vendor), "product=0x{0}".format(product) ] else: proc_args = ['modprobe', 'ftdi-sio'] proc_h = subprocess.Popen(proc_args, stdin=subprocess.PIPE, stdout=subprocess.PIPE, close_fds=True, shell=False) if proc_h.wait(): self.print_error('modprobe exited with a non-zero status code') return if linux_kernel_version >= (3, 12) and os.path.isfile( '/sys/bus/usb-serial/drivers/ftdi_sio/new_id'): with open('/sys/bus/usb-serial/drivers/ftdi_sio/new_id', 'w') as file_h: file_h.write("{0} {1}".format(vendor, product)) self.print_status('Finished driver preparation') def do_previous(self, args): """Use the last specified module.""" if self.last_module is None: self.frmwk.print_error('no module has been previously selected') return self.frmwk.current_module, self.last_module = self.last_module, self.frmwk.current_module def do_reload(self, args): """Reload a module in to the framework""" args = shlex.split(args) if len(args) == 0: if self.frmwk.current_module: module_path = self.frmwk.current_module.path else: self.print_error('Must \'use\' module first') return elif not args[0] in self.frmwk.modules.keys(): self.print_error('Invalid Module Selected.') return else: module_path = args[0] try: self.frmwk.reload_module(module_path) except FrameworkRuntimeError as err: self.print_error('Failed to reload module') return self.print_status('Successfully reloaded module: ' + module_path) def complete_reload(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)] def do_resource(self, args): """Run a resource file""" args = shlex.split(args) for rc_file in args: if not os.path.isfile(rc_file): self.print_error('Invalid resource file: ' + rc_file + ' (not found)') continue if not os.access(rc_file, os.R_OK): self.print_error('Invalid resource file: ' + rc_file + ' (no read permissions)') continue self.print_status('Running commands from resource file: ' + rc_file) self.run_rc_file(rc_file) def complete_resource(self, text, line, begidx, endidx): return complete_path(text, allow_files=True) def do_run(self, args): """Run the currently selected module""" args = shlex.split(args) old_module = None if len(args) and args[0] in self.frmwk.modules.keys(): old_module = self.frmwk.current_module self.frmwk.current_module = self.frmwk.modules[args[0]] if self.frmwk.current_module is None: self.print_error('Must \'use\' module first') return module = self.frmwk.current_module missing_options = module.get_missing_options() if missing_options: self.print_error('The following options must be set: ' + ', '.join(missing_options)) return del missing_options try: self.frmwk.run() except KeyboardInterrupt: self.print_line('') except Exception as error: self.print_exception(error) old_module = None if old_module: self.frmwk.current_module = old_module def complete_run(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)] def do_set(self, args): """Set an option, usage: set [option] [value]""" args = shlex.split(args) if len(args) < 2: self.print_error('set: [option] [value]') return name = args[0].upper() value = ' '.join(args[1:]) if self.frmwk.current_module: options = self.frmwk.current_module.options advanced_options = self.frmwk.current_module.advanced_options else: options = self.frmwk.options advanced_options = self.frmwk.advanced_options if name in options: try: options.set_option(name, value) self.print_line(name + ' => ' + value) except TypeError: self.print_error('Invalid data type') return elif name in advanced_options: try: advanced_options.set_option(name, value) self.print_line(name + ' => ' + value) except TypeError: self.print_error('Invalid data type') return self.print_error('Unknown variable name') def complete_set(self, text, line, begidx, endidx): if self.frmwk.current_module: options = self.frmwk.current_module.options else: options = self.frmwk.options return [i + ' ' for i in options.keys() if i.startswith(text.upper())] def do_show(self, args): """Valid parameters for the "show" command are: modules, options""" args = shlex.split(args) if len(args) == 0: args.append('options') elif not args[0] in ['advanced', 'modules', 'options', '-h']: self.print_error('Invalid parameter "' + args[0] + '", use "show -h" for more information') return if args[0] == 'modules': self.print_line('') self.print_line('Modules' + os.linesep + '=======') self.print_line('') longest_name = 18 for module_name in self.frmwk.modules.keys(): longest_name = max(longest_name, len(module_name)) fmt_string = " {0:" + str(longest_name) + "} {1}" self.print_line(fmt_string.format('Name', 'Description')) self.print_line(fmt_string.format('----', '-----------')) module_names = sorted(list(self.frmwk.modules.keys())) module_names.sort() for module_name in module_names: module_obj = self.frmwk.modules[module_name] self.print_line( fmt_string.format(module_name, module_obj.description)) self.print_line('') return elif args[0] == 'options' or args[0] == 'advanced': self.print_line('') if self.frmwk.current_module and args[0] == 'options': options = self.frmwk.current_module.options self.print_line('Module Options' + os.linesep + '==============') if self.frmwk.current_module and args[0] == 'advanced': options = self.frmwk.current_module.advanced_options self.print_line('Advanced Module Options' + os.linesep + '=======================') elif self.frmwk.current_module is None and args[0] == 'options': options = self.frmwk.options self.print_line('Framework Options' + os.linesep + '=================') elif self.frmwk.current_module is None and args[0] == 'advanced': options = self.frmwk.advanced_options self.print_line('Advanced Framework Options' + os.linesep + '==========================') self.print_line('') longest_name = 16 longest_value = 10 for option_name, option_def in options.items(): longest_name = max(longest_name, len(option_name)) longest_value = max(longest_value, len(str(options[option_name]))) fmt_string = " {0:<" + str(longest_name) + "} {1:<" + str( longest_value) + "} {2}" self.print_line(fmt_string.format('Name', 'Value', 'Description')) self.print_line(fmt_string.format('----', '-----', '-----------')) for option_name in options.keys(): option_value = options[option_name] if option_value is None: option_value = '' option_desc = options.get_option_help(option_name) self.print_line( fmt_string.format(option_name, str(option_value), option_desc)) self.print_line('') elif args[0] == '-h': self.print_status( 'Valid parameters for the "show" command are: modules, options' ) def complete_show(self, text, line, begidx, endidx): return [ i for i in ['advanced', 'modules', 'options'] if i.startswith(text.lower()) ] def do_use(self, args): """Select a module to use""" args = shlex.split(args) if len(args) != 1: self.print_line('Usage:') self.print_line(' use [module name]') return mod_name = args[0] if mod_name in self.frmwk.modules.keys(): self.last_module = self.frmwk.current_module self.frmwk.current_module = self.frmwk.modules[mod_name] else: self.logger.error('failed to load module: ' + mod_name) self.print_error('Failed to load module: ' + mod_name) def complete_use(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)]
class InteractiveInterpreter(OverrideCmd): __doc__ = 'The core interpreter for the program' __name__ = 'termineter' prompt = __name__ + ' > ' ruler = '+' doc_header = 'Type help <command> For Information\nList Of Available Commands:' def __init__(self, check_rc_file=True, stdin=None, stdout=None, log_handler=None): OverrideCmd.__init__(self, stdin=stdin, stdout=stdout) if stdin is not None: self.use_rawinput = False # No 'use_rawinput' will cause problems with the ipy command so disable it for now self.__disabled_commands__.append('ipy') if not its.on_linux: self.__hidden_commands__.append('prep_driver') self.__hidden_commands__.append('cd') self.__hidden_commands__.append('exploit') self.last_module = None self.log_handler = log_handler if self.log_handler is None: self.__disabled_commands__.append('logging') self.logger = logging.getLogger(self.__package__ + '.interpreter') self.frmwk = Framework(stdout=stdout) self.print_exception = self.frmwk.print_exception self.print_error = self.frmwk.print_error self.print_good = self.frmwk.print_good self.print_line = self.frmwk.print_line self.print_status = self.frmwk.print_status if check_rc_file: if check_rc_file: check_rc_file = self.frmwk.directories.user_data + 'console.rc' if os.path.isfile(check_rc_file) and os.access(check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) elif isinstance(check_rc_file, str): if os.path.isfile(check_rc_file) and os.access(check_rc_file, os.R_OK): self.print_status('Running commands from resource file: ' + check_rc_file) self.run_rc_file(check_rc_file) else: self.logger.error('could not access resource file: ' + check_rc_file) self.print_error('Could not access resource file: ' + check_rc_file) try: import readline readline.read_history_file(self.frmwk.directories.user_data + 'history.txt') readline.set_completer_delims(readline.get_completer_delims().replace('/', '')) except (ImportError, IOError): pass @property def intro(self): intro = os.linesep intro += ' ______ _ __ ' + os.linesep intro += ' /_ __/__ ______ _ (_)__ ___ / /____ ____' + os.linesep intro += ' / / / -_) __/ \' \/ / _ \/ -_) __/ -_) __/' + os.linesep intro += ' /_/ \__/_/ /_/_/_/_/_//_/\__/\__/\__/_/ ' + os.linesep intro += os.linesep fmt_string = " <[ {0:<18} {1:>18}" intro += fmt_string.format(self.__name__, 'v' + __version__ + '') + os.linesep intro += fmt_string.format('model:', __codename__) + os.linesep intro += fmt_string.format('loaded modules:', len(self.frmwk.modules)) + os.linesep #if self.frmwk.rfcat_available: # intro += fmt_string.format('rfcat:', 'enabled') + os.linesep #else: # intro += fmt_string.format('rfcat:', 'disabled') + os.linesep return intro @property def prompt(self): if self.frmwk.current_module: if self.frmwk.use_colors: return self.__name__ + ' (\033[1;33m' + self.frmwk.current_module.name + '\033[1;m) > ' else: return self.__name__ + ' (' + self.frmwk.current_module.name + ') > ' else: return self.__name__ + ' > ' def run_rc_file(self, rc_file): if os.path.isfile(rc_file) and os.access(rc_file, os.R_OK): self.logger.info('processing "' + rc_file + '" for commands') for line in open(rc_file, 'r'): line = line.strip() if not len(line) or line[0] == '#': continue if line.startswith('print_'): line = line[6:] print_type, message = line.split(' ', 1) if print_type in ('error', 'good', 'line', 'status'): getattr(self, 'print_' + print_type)(message) continue self.print_line(self.prompt + line.strip()) self.onecmd(line.strip()) else: self.logger.error('invalid rc file: ' + rc_file) return False return True @staticmethod def serve(addr, run_once=False, log_level=None, use_ssl=False, ssl_cert=None): import ssl __package__ = '.'.join(InteractiveInterpreter.__module__.split('.')[:-1]) logger = logging.getLogger(__package__ + '.interpreter.server') srv_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) srv_sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) srv_sock.bind(addr) logger.debug('listening for connections on: ' + addr[0] + ':' + str(addr[1])) srv_sock.listen(1) while True: try: (clt_sock, clt_addr) = srv_sock.accept() except KeyboardInterrupt: break logger.info('received connection from: ' + clt_addr[0] + ':' + str(clt_addr[1])) if use_ssl: ssl_sock = ssl.wrap_socket(clt_sock, server_side=True, certfile=ssl_cert) ins = ssl_sock.makefile('r', 1) outs = ssl_sock.makefile('w', 1) else: ins = clt_sock.makefile('r', 1) outs = clt_sock.makefile('w', 1) log_stream = logging.StreamHandler(outs) if log_level is not None: log_stream.setLevel(log_level) log_stream.setFormatter(logging.Formatter("%(levelname)-8s %(message)s")) logging.getLogger('').addHandler(log_stream) interpreter = InteractiveInterpreter(check_rc_file=False, stdin=ins, stdout=outs) try: interpreter.cmdloop() except socket.error: log_stream.close() logging.getLogger('').removeHandler(log_stream) logger.warning('received a socket error during the main interpreter loop') continue log_stream.flush() log_stream.close() logging.getLogger('').removeHandler(log_stream) outs.close() ins.close() clt_sock.shutdown(socket.SHUT_RDWR) clt_sock.close() del clt_sock if run_once: break srv_sock.shutdown(socket.SHUT_RDWR) srv_sock.close() def do_back(self, args): """Stop using a module""" self.frmwk.current_module = None def do_banner(self, args): """Print the banner""" self.print_line(self.intro) def do_cd(self, args): """Change the current working directory""" path = args.split(' ')[0] if not path: self.print_error('must specify a path') return if not os.path.isdir(path): self.print_error('invalid path') return os.chdir(path) def complete_cd(self, text, line, begidx, endidx): return complete_path(text, allow_files=False) def do_connect(self, args): """Connect the serial interface""" args = shlex.split(args) if self.frmwk.is_serial_connected(): self.print_status('Already connected') return missing_options = self.frmwk.options.get_missing_options() if missing_options: self.print_error('The following options must be set: ' + ', '.join(missing_options)) return try: self.frmwk.serial_connect() except Exception as error: self.print_exception(error) return self.print_good('Successfully connected and the device is responding') if len(args) and args[0] == '-l': if self.frmwk.serial_login(): self.print_good('Successfully authenticated to the device') else: self.print_error('Failed to authenticate to the device') def do_disconnect(self, args): """Disconnect the serial interface""" args = shlex.split(args) if not self.frmwk.is_serial_connected(): self.print_error('Not connected') return result = self.frmwk.serial_disconnect() if result: self.print_good('Successfully disconnected') else: self.print_error('An error occurred while closing the serial interface') if len(args) and args[0] == '-r': missing_options = self.frmwk.options.get_missing_options() if missing_options: self.print_error('The following options must be set: ' + ', '.join(missing_options)) return try: self.frmwk.serial_connect() except Exception as error: self.print_exception(error) return self.print_good('Successfully reconnected and the device is responding') def do_exit(self, args): """Exit The Interpreter""" quotes = ( 'I\'ll be back.', 'Hasta la vista, baby.', 'Come with me if you want to live.', 'Where\'s John Connor?' ) self.logger.info('received exit command, now exiting') self.print_status(random.choice(quotes)) try: import readline readline.write_history_file(self.frmwk.directories.user_data + 'history.txt') except (ImportError, IOError): pass return True def do_exploit(self, args): """Run the currently selected module""" self.do_run(args) def do_help(self, args): super(InteractiveInterpreter, self).do_help(args) self.print_line('') def do_logging(self, args): """Set and show logging options""" args = shlex.split(args) if not args: args.append('show') elif not args[0] in ['show', 'set', '-h']: self.print_error('Invalid parameter "' + args[0] + '", use "logging -h" for more information') return if args[0] == '-h': self.print_status('Valid parameters for the "logging" command are: show, set') return elif self.log_handler is None: self.print_error('No log handler is defined') return if args[0] == 'show': loglvl = self.log_handler.level self.print_status('Effective logging level is: ' + ({10: 'DEBUG', 20: 'INFO', 30: 'WARNING', 40: 'ERROR', 50: 'CRITICAL'}.get(loglvl) or 'UNKNOWN')) elif args[0] == 'set': if len(args) == 1: self.print_error('Missing log level, valid options are: debug, info, warning, error, critical') return new_lvl = args[1].upper() if new_lvl in ['DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL']: self.log_handler.setLevel(getattr(logging, new_lvl)) self.print_status('Successfully changed the logging level to: ' + new_lvl) else: self.print_error('Missing log level, valid options are: debug, info, warning, error, critical') def complete_logging(self, text, line, begidx, endidx): return [i for i in ['set', 'show', 'debug', 'info', 'warning', 'error', 'critical'] if i.startswith(text.lower())] def do_info(self, args): """Show module information""" args = shlex.split(args) if not args and self.frmwk.current_module is None: self.print_error('Must select module to show information') return if len(args) and args[0]: if args[0] in self.frmwk.modules.keys(): module = self.frmwk.modules[args[0]] else: self.print_error('Invalid module name') return else: module = self.frmwk.current_module self.print_line('') self.print_line(' Name: ' + module.name) if len(module.author) == 1: self.print_line(' Author: ' + module.author[0]) elif len(module.author) > 1: self.print_line(' Authors: ' + module.author[0]) for additional_author in module.author[1:]: self.print_line(' ' + additional_author) self.print_line(' Version: ' + str(module.version)) self.print_line('') self.print_line('Basic Options: ') longest_name = 16 longest_value = 10 for option_name, option_def in module.options.items(): longest_name = max(longest_name, len(option_name)) longest_value = max(longest_value, len(str(module.options[option_name]))) fmt_string = " {0:<" + str(longest_name) + "} {1:<" + str(longest_value) + "} {2}" self.print_line(fmt_string.format('Name', 'Value', 'Description')) self.print_line(fmt_string.format('----', '-----', '-----------')) for option_name in module.options.keys(): option_value = module.options[option_name] if option_value is None: option_value = '' option_desc = module.options.get_option_help(option_name) self.print_line(fmt_string.format(option_name, str(option_value), option_desc)) self.print_line('') self.print_line('Description:') for line in textwrap.wrap(textwrap.dedent(module.detailed_description), 78): self.print_line(' ' + line) self.print_line('') def complete_info(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)] def do_ipy(self, args): """Start an interactive Python interpreter""" import c1218.data import c1219.data from c1219.access.general import C1219GeneralAccess from c1219.access.security import C1219SecurityAccess from c1219.access.log import C1219LogAccess from c1219.access.telephone import C1219TelephoneAccess vars = { '__version__': __version__, 'C1218Packet': c1218.data.C1218Packet, 'C1218ReadRequest': c1218.data.C1218ReadRequest, 'C1218WriteRequest': c1218.data.C1218WriteRequest, 'C1219ProcedureInit': c1219.data.C1219ProcedureInit, 'C1219GeneralAccess': C1219GeneralAccess, 'C1219SecurityAccess': C1219SecurityAccess, 'C1219LogAccess': C1219LogAccess, 'C1219TelephoneAccess': C1219TelephoneAccess, 'frmwk': self.frmwk, 'os': os, 'sys': sys } banner = 'Python ' + sys.version + ' on ' + sys.platform + os.linesep banner += os.linesep banner += 'The framework instance is in the \'frmwk\' variable.' if self.frmwk.is_serial_connected(): vars['conn'] = self.frmwk.serial_connection banner += os.linesep banner += 'The connection instance is in the \'conn\' variable.' pyconsole = code.InteractiveConsole(vars) savestdin = os.dup(sys.stdin.fileno()) savestdout = os.dup(sys.stdout.fileno()) savestderr = os.dup(sys.stderr.fileno()) try: pyconsole.interact(banner) except SystemExit: sys.stdin = os.fdopen(savestdin, 'r', 0) sys.stdout = os.fdopen(savestdout, 'w', 0) sys.stderr = os.fdopen(savestderr, 'w', 0) def do_prep_driver(self, args): """Prep the optical probe driver""" args = shlex.split(args) if len(args) != 2: self.print_line('Usage:') self.print_line(' prep_driver VVVV PPPP') self.print_line('') self.print_line('Where VVVV and PPPP are the 4 hex digits of the vendor and product IDs respectively') return if os.getuid(): self.print_error('Must be running as root to prep the driver') return vendor, product = args if vendor.startswith('0x'): vendor = vendor[2:] if product.startswith('0x'): product = product[2:] linux_kernel_version = platform.uname()[2].split('.')[:2] linux_kernel_version = tuple(int(part) for part in linux_kernel_version) if linux_kernel_version < (3, 12): proc_args = ['modprobe', 'ftdi-sio', "vendor=0x{0}".format(vendor), "product=0x{0}".format(product)] else: proc_args = ['modprobe', 'ftdi-sio'] proc_h = subprocess.Popen(proc_args, stdin=subprocess.PIPE, stdout=subprocess.PIPE, close_fds=True, shell=False) if proc_h.wait(): self.print_error('modprobe exited with a non-zero status code') return if linux_kernel_version >= (3, 12) and os.path.isfile('/sys/bus/usb-serial/drivers/ftdi_sio/new_id'): with open('/sys/bus/usb-serial/drivers/ftdi_sio/new_id', 'w') as file_h: file_h.write("{0} {1}".format(vendor, product)) self.print_status('Finished driver preparation') def do_previous(self, args): """Use the last specified module.""" if self.last_module is None: self.frmwk.print_error('no module has been previously selected') return self.frmwk.current_module, self.last_module = self.last_module, self.frmwk.current_module def do_reload(self, args): """Reload a module in to the framework""" args = shlex.split(args) if len(args) == 0: if self.frmwk.current_module: module_path = self.frmwk.current_module.path else: self.print_error('Must \'use\' module first') return elif not args[0] in self.frmwk.modules.keys(): self.print_error('Invalid Module Selected.') return else: module_path = args[0] try: self.frmwk.reload_module(module_path) except FrameworkRuntimeError as err: self.print_error('Failed to reload module') return self.print_status('Successfully reloaded module: ' + module_path) def complete_reload(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)] def do_resource(self, args): """Run a resource file""" args = shlex.split(args) for rc_file in args: if not os.path.isfile(rc_file): self.print_error('Invalid resource file: ' + rc_file + ' (not found)') continue if not os.access(rc_file, os.R_OK): self.print_error('Invalid resource file: ' + rc_file + ' (no read permissions)') continue self.print_status('Running commands from resource file: ' + rc_file) self.run_rc_file(rc_file) def complete_resource(self, text, line, begidx, endidx): return complete_path(text, allow_files=True) def do_run(self, args): """Run the currently selected module""" args = shlex.split(args) old_module = None if len(args) and args[0] in self.frmwk.modules.keys(): old_module = self.frmwk.current_module self.frmwk.current_module = self.frmwk.modules[args[0]] if self.frmwk.current_module is None: self.print_error('Must \'use\' module first') return module = self.frmwk.current_module missing_options = module.get_missing_options() if missing_options: self.print_error('The following options must be set: ' + ', '.join(missing_options)) return del missing_options try: self.frmwk.run() except KeyboardInterrupt: self.print_line('') except Exception as error: self.print_exception(error) old_module = None if old_module: self.frmwk.current_module = old_module def complete_run(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)] def do_set(self, args): """Set an option, usage: set [option] [value]""" args = shlex.split(args) if len(args) < 2: self.print_error('set: [option] [value]') return name = args[0].upper() value = ' '.join(args[1:]) if self.frmwk.current_module: options = self.frmwk.current_module.options advanced_options = self.frmwk.current_module.advanced_options else: options = self.frmwk.options advanced_options = self.frmwk.advanced_options if name in options: try: options.set_option(name, value) self.print_line(name + ' => ' + value) except TypeError: self.print_error('Invalid data type') return elif name in advanced_options: try: advanced_options.set_option(name, value) self.print_line(name + ' => ' + value) except TypeError: self.print_error('Invalid data type') return self.print_error('Unknown variable name') def complete_set(self, text, line, begidx, endidx): if self.frmwk.current_module: options = self.frmwk.current_module.options else: options = self.frmwk.options return [i + ' ' for i in options.keys() if i.startswith(text.upper())] def do_show(self, args): """Valid parameters for the "show" command are: modules, options""" args = shlex.split(args) if len(args) == 0: args.append('options') elif not args[0] in ['advanced', 'modules', 'options', '-h']: self.print_error('Invalid parameter "' + args[0] + '", use "show -h" for more information') return if args[0] == 'modules': self.print_line('') self.print_line('Modules' + os.linesep + '=======') self.print_line('') longest_name = 18 for module_name in self.frmwk.modules.keys(): longest_name = max(longest_name, len(module_name)) fmt_string = " {0:" + str(longest_name) + "} {1}" self.print_line(fmt_string.format('Name', 'Description')) self.print_line(fmt_string.format('----', '-----------')) module_names = sorted(list(self.frmwk.modules.keys())) module_names.sort() for module_name in module_names: module_obj = self.frmwk.modules[module_name] self.print_line(fmt_string.format(module_name, module_obj.description)) self.print_line('') return elif args[0] == 'options' or args[0] == 'advanced': self.print_line('') if self.frmwk.current_module and args[0] == 'options': options = self.frmwk.current_module.options self.print_line('Module Options' + os.linesep + '==============') if self.frmwk.current_module and args[0] == 'advanced': options = self.frmwk.current_module.advanced_options self.print_line('Advanced Module Options' + os.linesep + '=======================') elif self.frmwk.current_module is None and args[0] == 'options': options = self.frmwk.options self.print_line('Framework Options' + os.linesep + '=================') elif self.frmwk.current_module is None and args[0] == 'advanced': options = self.frmwk.advanced_options self.print_line('Advanced Framework Options' + os.linesep + '==========================') self.print_line('') longest_name = 16 longest_value = 10 for option_name, option_def in options.items(): longest_name = max(longest_name, len(option_name)) longest_value = max(longest_value, len(str(options[option_name]))) fmt_string = " {0:<" + str(longest_name) + "} {1:<" + str(longest_value) + "} {2}" self.print_line(fmt_string.format('Name', 'Value', 'Description')) self.print_line(fmt_string.format('----', '-----', '-----------')) for option_name in options.keys(): option_value = options[option_name] if option_value is None: option_value = '' option_desc = options.get_option_help(option_name) self.print_line(fmt_string.format(option_name, str(option_value), option_desc)) self.print_line('') elif args[0] == '-h': self.print_status('Valid parameters for the "show" command are: modules, options') def complete_show(self, text, line, begidx, endidx): return [i for i in ['advanced', 'modules', 'options'] if i.startswith(text.lower())] def do_use(self, args): """Select a module to use""" args = shlex.split(args) if len(args) != 1: self.print_line('Usage:') self.print_line(' use [module name]') return mod_name = args[0] if mod_name in self.frmwk.modules.keys(): self.last_module = self.frmwk.current_module self.frmwk.current_module = self.frmwk.modules[mod_name] else: self.logger.error('failed to load module: ' + mod_name) self.print_error('Failed to load module: ' + mod_name) def complete_use(self, text, line, begidx, endidx): return [i for i in self.frmwk.modules.keys() if i.startswith(text)]