""" This module contains all the api calls written for use in plugins. If you want to do anything that is't allowed through these function calls or through the functions provided for macros, contact me and I'll see what I can do to add some more functionality into the next version. """ import glob import imp import os import pappyproxy import stat import crochet from twisted.internet import defer from .colors import Colors from .util import PappyException class Plugin(object): def __init__(self, cmd, fname=None): self.cmd = cmd self.filename = '' self.source = None self.module_name = '' if fname: self.filename = fname self.load_file(fname) def load_file(self, fname): module_name = os.path.basename(os.path.splitext(fname)[0]) if os.path.basename(fname) == '__init__.py': return st = os.stat(fname) if (st.st_mode & stat.S_IWOTH): raise PappyException("Refusing to load world-writable plugin: %s" % fname) self.source = imp.load_source('%s'%module_name, fname) if hasattr(self.source, 'load_cmds'): self.source.load_cmds(self.cmd) else: print ('WARNING: %s does not define load_cmds. It will not be ' 'possible to interact with the plugin through the console.' % fname) self.module_name = module_name class PluginLoader(object): def __init__(self, cmd): self.cmd = cmd self.loaded_plugins = [] self.plugins_by_name = {} def load_plugin(self, fname): p = Plugin(self.cmd, fname) self.loaded_plugins.append(p) self.plugins_by_name[p.module_name] = p def load_directory(self, directory): fnames = glob.glob(os.path.join(directory, '*.py')) for fname in fnames: try: self.load_plugin(fname) except PappyException as e: print str(e) ########################## ## Plugin helper functions def plugin_by_name(name): """ Returns an interface to access the methods of a plugin from its name. For example, to call the ``foo`` function from the ``bar`` plugin you would call ``plugin_by_name('bar').foo()``. """ import pappyproxy.pappy if name in pappyproxy.pappy.session.plugin_loader.plugins_by_name: return pappyproxy.pappy.session.plugin_loader.plugins_by_name[name].source else: raise PappyException('No plugin with name %s is loaded' % name) def add_intercepting_macro(name, macro): """ Adds an intercepting macro to the proxy. You can either use a :class:`pappyproxy.macros.FileInterceptMacro` to load an intercepting macro from the disk, or you can create your own using an :class:`pappyproxy.macros.InterceptMacro` for a base class. You must give a unique name that will be used in :func:`pappyproxy.plugin.remove_intercepting_macro` to deactivate it. Remember that activating an intercepting macro will disable request streaming and will affect performance. So please try and only use this if you may need to modify messages before they are passed along. """ for factory in pappyproxy.pappy.session.server_factories: factory.add_intercepting_macro(macro, name=name) def remove_intercepting_macro(name): """ Stops an active intercepting macro. You must pass in the name that you used when calling :func:`pappyproxy.plugin.add_intercepting_macro` to identify which macro you would like to stop. """ for factory in pappyproxy.pappy.session.server_factories: factory.remove_intercepting_macro(name=name) def active_intercepting_macros(): """ Returns a dict of the active intercepting macro objects. Modifying this list will not affect which macros are active. """ # every factory should have the same int macros so screw it we'll # just use the macros from the first one ret = [] if len(pappyproxy.pappy.session.server_factories) > 0: ret = pappyproxy.pappy.session.server_factories[0].get_macro_list() return ret def in_memory_reqs(): """ Returns a list containing the ids of the requests which exist in memory only (requests with an m## style id). You can call either :func:`pappyproxy.http.Request.save` or :func:`pappyproxy.http.Request.async_deep_save` to save the request to the data file. """ return list(pappyproxy.http.Request.cache.inmem_reqs) def req_history(num=-1, ids=None, include_unmangled=False): """ Returns an a generator that generates deferreds which resolve to requests in history, ignoring the current context. If ``n`` is given, it will stop after ``n`` requests have been generated. If ``ids`` is given, it will only include those IDs. If ``include_unmangled`` is True, then the iterator will include requests which are the unmangled version of other requests. An example of using the iterator to print the 10 most recent requests:: @defer.inlineCallbacks def find_food(): for req_d in req_history(10): req = yield req_d print '-'*10 print req.full_message_pretty """ return pappyproxy.Request.cache.req_it(num=num, ids=ids, include_unmangled=include_unmangled) def async_main_context_ids(n=-1): """ Returns a deferred that resolves into a list of up to ``n`` of the most recent requests in the main context. You can then use :func:`pappyproxy.http.Request.load_request` to load the requests in the current context. If no value is passed for ``n``, this will return all of the IDs in the context. """ return pappyproxy.pappy.main_context.get_reqs(n) @crochet.wait_for(timeout=None) @defer.inlineCallbacks def main_context_ids(*args, **kwargs): """ Same as :func:`pappyproxy.plugin.async_main_context_ids` but can be called from macros and other non-async only functions. Cannot be called in async functions. """ ret = yield async_main_context_ids(*args, **kwargs) defer.returnValue(ret) def add_to_history(req): """ Save a request to history without saving it to the data file. The request will only be saved in memory, so when the program is exited or `clrmem` is run, the request will be deleted. :param req: The request to add to history :type req: :class:`pappyproxy.http.Request` """ pappyproxy.http.Request.cache.add(req) pappyproxy.context.reset_context_caches() def get_active_filter_strings(): """ Returns a list of filter strings representing the currently active filters """ filts = pappyproxy.pappy.main_context.active_filters strs = [] for f in filts: strs.append(f.filter_string) return strs def run_cmd(cmd): """ Run a command as if you typed it into the console. Try and use existing APIs to do what you want before using this. """ pappyproxy.pappy.cons.onecmd(cmd) def require_modules(*largs): """ A wrapper to make sure that plugin dependencies are installed. For example, if a command requires the ``psutil`` and ``objgraph`` package, you should format your command like:: @require_modules('psutil', 'objgraph') def my_command(line): import objgraph import psutil # ... rest of command ... If you try to run the command without being able to import all of the required modules, the command will print an error and not run the command. """ def wr(func): def wr2(*args, **kwargs): missing = [] for l in largs: try: imp.find_module(l) except ImportError: missing.append(l) if missing: print 'Command requires %s module(s)' % (', '.join([Colors.RED+m+Colors.ENDC for m in missing])) else: return func(*args, **kwargs) return wr2 return wr def set_context_to_saved(name): """ Sets the current context to the context saved under the given name. Raises PappyException if name does not exist """ @crochet.wait_for(timeout=None) @defer.inlineCallbacks def delete_saved_context(name): """ Deletes the saved context with the given name. Raises PappyException if name does not exist """ def save_current_context(name): """ Saves the current context under the given name. """ def save_context(name, filter_strs): """ Takes a list of filter strings and saves it as a context under the given name. :param name: The name to save the context under :type name: string :param filter_strs: The in-order list of filter strings of the context to save. :type filter_strs: List of strings """