401. Plugin infrastructure
* implements installing, loading and setup for plugins * codifies configuration * has a sample plugin * docs * tests
This commit is contained in:
parent
f10c3bb8e5
commit
29b6f91740
105
docs/source/plugins.rst
Normal file
105
docs/source/plugins.rst
Normal file
@ -0,0 +1,105 @@
|
||||
=========
|
||||
Plugins
|
||||
=========
|
||||
|
||||
GNU MediaGoblin supports plugins that, when installed, allow you to
|
||||
augment MediaGoblin's behavior.
|
||||
|
||||
This chapter covers discovering, installing, configuring and removing
|
||||
plugins.
|
||||
|
||||
|
||||
Discovering plugins
|
||||
===================
|
||||
|
||||
MediaGoblin comes with core plugins. Core plugins are located in the
|
||||
``mediagoblin.plugins`` module of the MediaGoblin code. Because they
|
||||
come with MediaGoblin, you don't have to install them, but you do have
|
||||
to add them to your config file if you're interested in using them.
|
||||
|
||||
You can also write your own plugins and additionally find plugins
|
||||
elsewhere on the Internet. Since these plugins don't come with
|
||||
MediaGoblin, you must first install them, then add them to your
|
||||
configuration.
|
||||
|
||||
|
||||
Installing plugins
|
||||
==================
|
||||
|
||||
MediaGoblin core plugins don't need to be installed. For core plugins,
|
||||
you can skip installation!
|
||||
|
||||
If the plugin is not a core plugin and is packaged and available on
|
||||
the Python Package Index, then you can install the plugin with pip::
|
||||
|
||||
pip install <plugin-name>
|
||||
|
||||
For example, if we wanted to install the plugin named
|
||||
"mediagoblin-restrictfive", we would do::
|
||||
|
||||
pip install mediagoblin-restrictfive
|
||||
|
||||
.. Note::
|
||||
|
||||
If you're using a virtual environment, make sure to activate the
|
||||
virtual environment before installing with pip. Otherwise the
|
||||
plugin may get installed in a different environment.
|
||||
|
||||
Once you've installed the plugin software, you need to tell
|
||||
MediaGoblin that this is a plugin you want MediaGoblin to use. To do
|
||||
that, you edit the ``mediagoblin.ini`` file and add the plugin as a
|
||||
subsection of the plugin section.
|
||||
|
||||
For example, say the "mediagoblin-restrictfive" plugin had the Python
|
||||
package path ``restrictfive``, then you would add ``restrictfive`` to
|
||||
the ``plugins`` section as a subsection::
|
||||
|
||||
[plugins]
|
||||
|
||||
[[restrictfive]]
|
||||
|
||||
|
||||
Configuring plugins
|
||||
===================
|
||||
|
||||
Generally, configuration goes in the ``.ini`` file. Configuration for
|
||||
a specific plugin, goes in a subsection of the ``plugins`` section.
|
||||
|
||||
Example 1: Core MediaGoblin plugin
|
||||
|
||||
If you wanted to use the core MediaGoblin flatpages plugin, the module
|
||||
for that is ``mediagoblin.plugins.flatpages`` and you would add that
|
||||
to your ``.ini`` file like this::
|
||||
|
||||
[plugins]
|
||||
|
||||
[[mediagoblin.plugins.flatpages]]
|
||||
# configuration for flatpages plugin here!
|
||||
|
||||
Example 2: Plugin that is not a core MediaGoblin plugin
|
||||
|
||||
If you installed a hypothetical restrictfive plugin which is in the
|
||||
module ``restrictfive``, your ``.ini`` file might look like this (with
|
||||
comments making the bits clearer)::
|
||||
|
||||
[plugins]
|
||||
|
||||
[[restrictfive]]
|
||||
# configuration for restrictfive here!
|
||||
|
||||
Check the plugin's documentation for what configuration options are
|
||||
available.
|
||||
|
||||
|
||||
Removing plugins
|
||||
================
|
||||
|
||||
To remove a plugin, use ``pip uninstall``. For example::
|
||||
|
||||
pip uninstall mediagoblin-restrictfive
|
||||
|
||||
.. Note::
|
||||
|
||||
If you're using a virtual environment, make sure to activate the
|
||||
virtual environment before uninstalling with pip. Otherwise the
|
||||
plugin may get installed in a different environment.
|
@ -30,3 +30,8 @@ base_url = /mgoblin_media/
|
||||
|
||||
[celery]
|
||||
# Put celery stuff here
|
||||
|
||||
# place plugins here---each in their own subsection of [plugins]. see
|
||||
# documentation for details.
|
||||
#[plugins]
|
||||
|
||||
|
@ -27,6 +27,7 @@ from mediagoblin.tools.response import render_404
|
||||
from mediagoblin.tools import request as mg_request
|
||||
from mediagoblin.mg_globals import setup_globals
|
||||
from mediagoblin.init.celery import setup_celery_from_config
|
||||
from mediagoblin.init.plugins import setup_plugins
|
||||
from mediagoblin.init import (get_jinja_loader, get_staticdirector,
|
||||
setup_global_and_app_config, setup_workbench, setup_database,
|
||||
setup_storage, setup_beaker_cache)
|
||||
@ -64,6 +65,11 @@ class MediaGoblinApp(object):
|
||||
# Setup other connections / useful objects
|
||||
##########################################
|
||||
|
||||
# Set up plugins -- need to do this early so that plugins can
|
||||
# affect startup.
|
||||
_log.info("Setting up plugins.")
|
||||
setup_plugins()
|
||||
|
||||
# Set up the database
|
||||
self.connection, self.db = setup_database()
|
||||
|
||||
|
59
mediagoblin/init/plugins/__init__.py
Normal file
59
mediagoblin/init/plugins/__init__.py
Normal file
@ -0,0 +1,59 @@
|
||||
# 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/>.
|
||||
|
||||
import logging
|
||||
|
||||
from mediagoblin import mg_globals
|
||||
from mediagoblin.tools import pluginapi
|
||||
|
||||
|
||||
_log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
def setup_plugins():
|
||||
"""This loads, configures and registers plugins
|
||||
|
||||
See plugin documentation for more details.
|
||||
"""
|
||||
|
||||
global_config = mg_globals.global_config
|
||||
plugin_section = global_config.get('plugins', {})
|
||||
|
||||
if not plugin_section:
|
||||
_log.info("No plugins to load")
|
||||
return
|
||||
|
||||
pcache = pluginapi.PluginCache()
|
||||
|
||||
# Go through and import all the modules that are subsections of
|
||||
# the [plugins] section.
|
||||
for plugin_module, config in plugin_section.items():
|
||||
_log.info("Importing plugin module: %s" % plugin_module)
|
||||
# If this throws errors, that's ok--it'll halt mediagoblin
|
||||
# startup.
|
||||
__import__(plugin_module)
|
||||
|
||||
# Note: One side-effect of importing things is that anything that
|
||||
# subclassed pluginapi.Plugin is registered.
|
||||
|
||||
# Go through all the plugin classes, instantiate them, and call
|
||||
# setup_plugin so they can figure things out.
|
||||
for plugin_class in pcache.plugin_classes:
|
||||
name = plugin_class.__module__ + "." + plugin_class.__name__
|
||||
_log.info("Loading plugin: %s" % name)
|
||||
plugin_obj = plugin_class()
|
||||
plugin_obj.setup_plugin()
|
||||
pcache.register_plugin_object(plugin_obj)
|
6
mediagoblin/plugins/README
Normal file
6
mediagoblin/plugins/README
Normal file
@ -0,0 +1,6 @@
|
||||
========
|
||||
README
|
||||
========
|
||||
|
||||
This directory holds the MediaGoblin core plugins. These plugins are not
|
||||
enabled by default. See documentation for enabling plugins.
|
16
mediagoblin/plugins/__init__.py
Normal file
16
mediagoblin/plugins/__init__.py
Normal file
@ -0,0 +1,16 @@
|
||||
# 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/>.
|
||||
|
6
mediagoblin/plugins/sampleplugin/README
Normal file
6
mediagoblin/plugins/sampleplugin/README
Normal file
@ -0,0 +1,6 @@
|
||||
========
|
||||
README
|
||||
========
|
||||
|
||||
This is a sample plugin. It does nothing interesting other than show
|
||||
one way to structure a MediaGoblin plugin.
|
20
mediagoblin/plugins/sampleplugin/__init__.py
Normal file
20
mediagoblin/plugins/sampleplugin/__init__.py
Normal file
@ -0,0 +1,20 @@
|
||||
# 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 imports the module that has the Plugin subclass in it which
|
||||
# causes that module to get imported and that class to get registered.
|
||||
import mediagoblin.plugins.sampleplugin.main
|
42
mediagoblin/plugins/sampleplugin/main.py
Normal file
42
mediagoblin/plugins/sampleplugin/main.py
Normal file
@ -0,0 +1,42 @@
|
||||
# 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/>.
|
||||
|
||||
import logging
|
||||
from mediagoblin.tools.pluginapi import Plugin, get_config
|
||||
|
||||
|
||||
_log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class SamplePlugin(Plugin):
|
||||
"""
|
||||
This is a sample plugin class. It automatically registers itself
|
||||
with mediagoblin when this module is imported.
|
||||
|
||||
The setup_plugin method prints configuration for this plugin if
|
||||
it exists.
|
||||
"""
|
||||
def __init__(self):
|
||||
self._setup_plugin_called = 0
|
||||
|
||||
def setup_plugin(self):
|
||||
_log.info('Sample plugin set up!')
|
||||
config = get_config('mediagoblin.plugins.sampleplugin')
|
||||
if config:
|
||||
_log.info('%r' % config)
|
||||
else:
|
||||
_log.info('There is no configuration set.')
|
||||
self._setup_plugin_called += 1
|
158
mediagoblin/tests/test_pluginapi.py
Normal file
158
mediagoblin/tests/test_pluginapi.py
Normal file
@ -0,0 +1,158 @@
|
||||
# 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/>.
|
||||
|
||||
import sys
|
||||
from configobj import ConfigObj
|
||||
from mediagoblin import mg_globals
|
||||
from mediagoblin.init.plugins import setup_plugins
|
||||
from mediagoblin.tools import pluginapi
|
||||
from nose.tools import eq_
|
||||
|
||||
|
||||
def with_cleanup(*modules_to_delete):
|
||||
def _with_cleanup(fun):
|
||||
"""Wrapper that saves and restores mg_globals"""
|
||||
def _with_cleanup_inner(*args, **kwargs):
|
||||
old_app_config = mg_globals.app_config
|
||||
old_global_config = mg_globals.global_config
|
||||
# Need to delete icky modules before and after so as to make
|
||||
# sure things work correctly.
|
||||
for module in modules_to_delete:
|
||||
try:
|
||||
del sys.modules[module]
|
||||
except KeyError:
|
||||
pass
|
||||
# The plugin cache gets populated as a side-effect of
|
||||
# importing, so it's best to clear it before and after a test.
|
||||
pcache = pluginapi.PluginCache()
|
||||
pcache.clear()
|
||||
try:
|
||||
return fun(*args, **kwargs)
|
||||
finally:
|
||||
mg_globals.app_config = old_app_config
|
||||
mg_globals.global_config = old_global_config
|
||||
# Need to delete icky modules before and after so as to make
|
||||
# sure things work correctly.
|
||||
for module in modules_to_delete:
|
||||
try:
|
||||
del sys.modules[module]
|
||||
except KeyError:
|
||||
pass
|
||||
pcache.clear()
|
||||
|
||||
_with_cleanup_inner.__name__ = fun.__name__
|
||||
return _with_cleanup_inner
|
||||
return _with_cleanup
|
||||
|
||||
|
||||
def build_config(sections):
|
||||
"""Builds a ConfigObj object with specified data
|
||||
|
||||
:arg sections: list of ``(section_name, section_data,
|
||||
subsection_list)`` tuples where section_data is a dict and
|
||||
subsection_list is a list of ``(section_name, section_data,
|
||||
subsection_list)``, ...
|
||||
|
||||
For example:
|
||||
|
||||
>>> build_config([
|
||||
... ('mediagoblin', {'key1': 'val1'}, []),
|
||||
... ('section2', {}, [
|
||||
... ('subsection1', {}, [])
|
||||
... ])
|
||||
... ])
|
||||
"""
|
||||
cfg = ConfigObj()
|
||||
cfg.filename = 'foo'
|
||||
def _iter_section(cfg, section_list):
|
||||
for section_name, data, subsection_list in section_list:
|
||||
cfg[section_name] = data
|
||||
_iter_section(cfg[section_name], subsection_list)
|
||||
|
||||
_iter_section(cfg, sections)
|
||||
return cfg
|
||||
|
||||
|
||||
@with_cleanup()
|
||||
def test_no_plugins():
|
||||
"""Run setup_plugins with no plugins in config"""
|
||||
cfg = build_config([('mediagoblin', {}, [])])
|
||||
mg_globals.app_config = cfg['mediagoblin']
|
||||
mg_globals.global_config = cfg
|
||||
|
||||
pcache = pluginapi.PluginCache()
|
||||
setup_plugins()
|
||||
|
||||
# Make sure we didn't load anything.
|
||||
eq_(len(pcache.plugin_classes), 0)
|
||||
eq_(len(pcache.plugin_objects), 0)
|
||||
|
||||
|
||||
@with_cleanup('mediagoblin.plugins.sampleplugin',
|
||||
'mediagoblin.plugins.sampleplugin.main')
|
||||
def test_one_plugin():
|
||||
"""Run setup_plugins with a single working plugin"""
|
||||
cfg = build_config([
|
||||
('mediagoblin', {}, []),
|
||||
('plugins', {}, [
|
||||
('mediagoblin.plugins.sampleplugin', {}, [])
|
||||
])
|
||||
])
|
||||
|
||||
mg_globals.app_config = cfg['mediagoblin']
|
||||
mg_globals.global_config = cfg
|
||||
|
||||
pcache = pluginapi.PluginCache()
|
||||
setup_plugins()
|
||||
|
||||
# Make sure we only found one plugin class
|
||||
eq_(len(pcache.plugin_classes), 1)
|
||||
# Make sure the class is the one we think it is.
|
||||
eq_(pcache.plugin_classes[0].__name__, 'SamplePlugin')
|
||||
|
||||
# Make sure there was one plugin created
|
||||
eq_(len(pcache.plugin_objects), 1)
|
||||
# Make sure we called setup_plugin on SamplePlugin
|
||||
eq_(pcache.plugin_objects[0]._setup_plugin_called, 1)
|
||||
|
||||
|
||||
@with_cleanup('mediagoblin.plugins.sampleplugin',
|
||||
'mediagoblin.plugins.sampleplugin.main')
|
||||
def test_same_plugin_twice():
|
||||
"""Run setup_plugins with a single working plugin twice"""
|
||||
cfg = build_config([
|
||||
('mediagoblin', {}, []),
|
||||
('plugins', {}, [
|
||||
('mediagoblin.plugins.sampleplugin', {}, []),
|
||||
('mediagoblin.plugins.sampleplugin', {}, []),
|
||||
])
|
||||
])
|
||||
|
||||
mg_globals.app_config = cfg['mediagoblin']
|
||||
mg_globals.global_config = cfg
|
||||
|
||||
pcache = pluginapi.PluginCache()
|
||||
setup_plugins()
|
||||
|
||||
# Make sure we only found one plugin class
|
||||
eq_(len(pcache.plugin_classes), 1)
|
||||
# Make sure the class is the one we think it is.
|
||||
eq_(pcache.plugin_classes[0].__name__, 'SamplePlugin')
|
||||
|
||||
# Make sure there was one plugin created
|
||||
eq_(len(pcache.plugin_objects), 1)
|
||||
# Make sure we called setup_plugin on SamplePlugin
|
||||
eq_(pcache.plugin_objects[0]._setup_plugin_called, 1)
|
118
mediagoblin/tools/pluginapi.py
Normal file
118
mediagoblin/tools/pluginapi.py
Normal file
@ -0,0 +1,118 @@
|
||||
# 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 and provides the plugin
|
||||
base.
|
||||
|
||||
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?
|
||||
====================
|
||||
|
||||
You create a Python package. In that package, you define a high-level
|
||||
``__init__.py`` that either defines or imports modules that define
|
||||
classes that inherit from the ``Plugin`` class.
|
||||
|
||||
|
||||
Lifecycle
|
||||
=========
|
||||
|
||||
1. All the modules listed as subsections of the ``plugins`` section in
|
||||
the config file are imported and any ``Plugin`` subclasses are
|
||||
loaded causing it to be registered with the ``PluginCache``.
|
||||
|
||||
2. After all plugin modules are imported, registered plugins are
|
||||
instantiated and ``setup_plugin`` is called with the configuration.
|
||||
|
||||
|
||||
How to build a plugin
|
||||
=====================
|
||||
|
||||
See the documentation on building plugins.
|
||||
"""
|
||||
|
||||
import logging
|
||||
|
||||
from mediagoblin import mg_globals
|
||||
|
||||
|
||||
_log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PluginCache(object):
|
||||
"""Cache of plugin things"""
|
||||
__state = {
|
||||
# list of plugin classes
|
||||
"plugin_classes": [],
|
||||
|
||||
# list of plugin objects
|
||||
"plugin_objects": []
|
||||
}
|
||||
|
||||
def clear(self):
|
||||
"""This is only useful for testing."""
|
||||
del self.plugin_classes[:]
|
||||
del self.plugin_objects[:]
|
||||
|
||||
def __init__(self):
|
||||
self.__dict__ = self.__state
|
||||
|
||||
def register_plugin_class(self, plugin_class):
|
||||
"""Registers a plugin class"""
|
||||
self.plugin_classes.append(plugin_class)
|
||||
|
||||
def register_plugin_object(self, plugin_obj):
|
||||
"""Registers a plugin object"""
|
||||
self.plugin_objects.append(plugin_obj)
|
||||
|
||||
|
||||
class MetaPluginClass(type):
|
||||
"""Metaclass for PluginBase derivatives"""
|
||||
def __new__(cls, name, bases, attrs):
|
||||
new_class = super(MetaPluginClass, cls).__new__(cls, name, bases, attrs)
|
||||
parents = [b for b in bases if isinstance(b, MetaPluginClass)]
|
||||
if not parents:
|
||||
return new_class
|
||||
PluginCache().register_plugin_class(new_class)
|
||||
return new_class
|
||||
|
||||
|
||||
class Plugin(object):
|
||||
__metaclass__ = MetaPluginClass
|
||||
|
||||
def setup_plugin(self):
|
||||
pass
|
||||
|
||||
|
||||
def get_config(key):
|
||||
"""Retrieves the configuration for a specified plugin by key
|
||||
|
||||
Example:
|
||||
|
||||
>>> get_config('mediagoblin.plugins.sampleplugin')
|
||||
{'foo': 'bar'}
|
||||
"""
|
||||
|
||||
global_config = mg_globals.global_config
|
||||
plugin_section = global_config.get('plugins', {})
|
||||
return plugin_section.get(key, {})
|
Loading…
x
Reference in New Issue
Block a user