123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368 |
- # GNU MediaGoblin -- federated, autonomous media hosting
- # Copyright (C) 2011, 2012 MediaGoblin contributors. See AUTHORS.
- #
- # This program is free software: you can redistribute it and/or modify
- # it under the terms of the GNU Affero General Public License as published by
- # the Free Software Foundation, either version 3 of the License, or
- # (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU Affero General Public License for more details.
- #
- # You should have received a copy of the GNU Affero General Public License
- # along with this program. If not, see <http://www.gnu.org/licenses/>.
- """
- This module implements the plugin api bits.
- Two things about things in this module:
- 1. they should be excessively well documented because we should pull
- from this file for the docs
- 2. they should be well tested
- How do plugins work?
- ====================
- Plugins are structured like any Python project. You create a Python package.
- In that package, you define a high-level ``__init__.py`` module that has a
- ``hooks`` dict that maps hooks to callables that implement those hooks.
- Additionally, you want a LICENSE file that specifies the license and a
- ``setup.py`` that specifies the metadata for packaging your plugin. A rough
- file structure could look like this::
- myplugin/
- |- setup.py # plugin project packaging metadata
- |- README # holds plugin project information
- |- LICENSE # holds license information
- |- myplugin/ # plugin package directory
- |- __init__.py # has hooks dict and code
- Lifecycle
- =========
- 1. All the modules listed as subsections of the ``plugins`` section in
- the config file are imported. MediaGoblin registers any hooks in
- the ``hooks`` dict of those modules.
- 2. After all plugin modules are imported, the ``setup`` hook is called
- allowing plugins to do any set up they need to do.
- """
- import logging
- from functools import wraps
- from mediagoblin import mg_globals
- _log = logging.getLogger(__name__)
- class PluginManager(object):
- """Manager for plugin things
- .. Note::
- This is a Borg class--there is one and only one of this class.
- """
- __state = {
- # list of plugin classes
- "plugins": [],
- # map of hook names -> list of callables for that hook
- "hooks": {},
- # list of registered template paths
- "template_paths": set(),
- # list of template hooks
- "template_hooks": {},
- # list of registered routes
- "routes": [],
- }
- def clear(self):
- """This is only useful for testing."""
- # Why lists don't have a clear is not clear.
- del self.plugins[:]
- del self.routes[:]
- self.hooks.clear()
- self.template_paths.clear()
- def __init__(self):
- self.__dict__ = self.__state
- def register_plugin(self, plugin):
- """Registers a plugin class"""
- self.plugins.append(plugin)
- def register_hooks(self, hook_mapping):
- """Takes a hook_mapping and registers all the hooks"""
- for hook, callables in hook_mapping.items():
- if isinstance(callables, (list, tuple)):
- self.hooks.setdefault(hook, []).extend(list(callables))
- else:
- # In this case, it's actually a single callable---not a
- # list of callables.
- self.hooks.setdefault(hook, []).append(callables)
- def get_hook_callables(self, hook_name):
- return self.hooks.get(hook_name, [])
- def register_template_path(self, path):
- """Registers a template path"""
- self.template_paths.add(path)
- def get_template_paths(self):
- """Returns a tuple of registered template paths"""
- return tuple(self.template_paths)
- def register_route(self, route):
- """Registers a single route"""
- _log.debug('registering route: {0}'.format(route))
- self.routes.append(route)
- def get_routes(self):
- return tuple(self.routes)
- def register_template_hooks(self, template_hooks):
- for hook, templates in template_hooks.items():
- if isinstance(templates, (list, tuple)):
- self.template_hooks.setdefault(hook, []).extend(list(templates))
- else:
- # In this case, it's actually a single callable---not a
- # list of callables.
- self.template_hooks.setdefault(hook, []).append(templates)
- def get_template_hooks(self, hook_name):
- return self.template_hooks.get(hook_name, [])
- def register_routes(routes):
- """Registers one or more routes
- If your plugin handles requests, then you need to call this with
- the routes your plugin handles.
- A "route" is a `routes.Route` object. See `the routes.Route
- documentation
- <http://routes.readthedocs.org/en/latest/modules/route.html>`_ for
- more details.
- Example passing in a single route:
- >>> register_routes(('about-view', '/about',
- ... 'mediagoblin.views:about_view_handler'))
- Example passing in a list of routes:
- >>> register_routes([
- ... ('contact-view', '/contact', 'mediagoblin.views:contact_handler'),
- ... ('about-view', '/about', 'mediagoblin.views:about_handler')
- ... ])
- .. Note::
- Be careful when designing your route urls. If they clash with
- core urls, then it could result in DISASTER!
- """
- if isinstance(routes, list):
- for route in routes:
- PluginManager().register_route(route)
- else:
- PluginManager().register_route(routes)
- def register_template_path(path):
- """Registers a path for template loading
- If your plugin has templates, then you need to call this with
- the absolute path of the root of templates directory.
- Example:
- >>> my_plugin_dir = os.path.dirname(__file__)
- >>> template_dir = os.path.join(my_plugin_dir, 'templates')
- >>> register_template_path(template_dir)
- .. Note::
- You can only do this in `setup_plugins()`. Doing this after
- that will have no effect on template loading.
- """
- PluginManager().register_template_path(path)
- def get_config(key):
- """Retrieves the configuration for a specified plugin by key
- Example:
- >>> get_config('mediagoblin.plugins.sampleplugin')
- {'foo': 'bar'}
- >>> get_config('myplugin')
- {}
- >>> get_config('flatpages')
- {'directory': '/srv/mediagoblin/pages', 'nesting': 1}}
- """
- global_config = mg_globals.global_config
- plugin_section = global_config.get('plugins', {})
- return plugin_section.get(key, {})
- def register_template_hooks(template_hooks):
- """
- Register a dict of template hooks.
- Takes template_hooks as an argument, which is a dictionary of
- template hook names/keys to the templates they should provide.
- (The value can either be a single template path or an iterable
- of paths.)
- Example:
- .. code-block:: python
- {"media_sidebar": "/plugin/sidemess/mess_up_the_side.html",
- "media_descriptionbox": ["/plugin/sidemess/even_more_mess.html",
- "/plugin/sidemess/so_much_mess.html"]}
- """
- PluginManager().register_template_hooks(template_hooks)
- def get_hook_templates(hook_name):
- """
- Get a list of hook templates for this hook_name.
- Note: for the most part, you access this via a template tag, not
- this method directly, like so:
- .. code-block:: html+jinja
- {% template_hook("media_sidebar") %}
- ... which will include all templates for you, partly using this
- method.
- However, this method is exposed to templates, and if you wish, you
- can iterate over templates in a template hook manually like so:
- .. code-block:: html+jinja
- {% for template_path in get_hook_templates("media_sidebar") %}
- <div class="extra_structure">
- {% include template_path %}
- </div>
- {% endfor %}
- Returns:
- A list of strings representing template paths.
- """
- return PluginManager().get_template_hooks(hook_name)
- #############################
- ## Hooks: The Next Generation
- #############################
- def hook_handle(hook_name, *args, **kwargs):
- """
- Run through hooks attempting to find one that handle this hook.
- All callables called with the same arguments until one handles
- things and returns a non-None value.
- (If you are writing a handler and you don't have a particularly
- useful value to return even though you've handled this, returning
- True is a good solution.)
- Note that there is a special keyword argument:
- if "default_handler" is passed in as a keyword argument, this will
- be used if no handler is found.
- Some examples of using this:
- - You need an interface implemented, but only one fit for it
- - You need to *do* something, but only one thing needs to do it.
- """
- default_handler = kwargs.pop('default_handler', None)
- callables = PluginManager().get_hook_callables(hook_name)
- result = None
- for callable in callables:
- result = callable(*args, **kwargs)
- if result is not None:
- break
- if result is None and default_handler is not None:
- result = default_handler(*args, **kwargs)
- return result
- def hook_runall(hook_name, *args, **kwargs):
- """
- Run through all callable hooks and pass in arguments.
- All non-None results are accrued in a list and returned from this.
- (Other "false-like" values like False and friends are still
- accrued, however.)
- Some examples of using this:
- - You have an interface call where actually multiple things can
- and should implement it
- - You need to get a list of things from various plugins that
- handle them and do something with them
- - You need to *do* something, and actually multiple plugins need
- to do it separately
- """
- callables = PluginManager().get_hook_callables(hook_name)
- results = []
- for callable in callables:
- result = callable(*args, **kwargs)
- if result is not None:
- results.append(result)
- return results
- def hook_transform(hook_name, arg):
- """
- Run through a bunch of hook callables and transform some input.
- Note that unlike the other hook tools, this one only takes ONE
- argument. This argument is passed to each function, which in turn
- returns something that becomes the input of the next callable.
- Some examples of using this:
- - You have an object, say a form, but you want plugins to each be
- able to modify it.
- """
- result = arg
- callables = PluginManager().get_hook_callables(hook_name)
- for callable in callables:
- result = callable(result)
- return result
|