I have written a simple system to load plugins from files in Python 2.7, and I'd like to see if any of it can be improved upon.
vedo.py
from __future__ import absolute_import
import os.path
import inspect
from vedo.event_emitter import EventEmitter
from vedo import VedoPlugin
def _get_module_name(directory, filename):
return '{0}.{1}'.format(os.path.basename(directory), filename)
def _is_plugin_class(plugin_class):
return (inspect.isclass(plugin_class) and
issubclass(plugin_class, VedoPlugin) and
plugin_class != VedoPlugin)
class Vedo(EventEmitter):
def __init__(self, plugin_directory):
self._plugin_directory = plugin_directory
self._plugins = []
@property
def plugin_directory(self):
return self._plugin_directory
@property
def plugins(self):
return self._plugins
def load_module(self, module):
plugin_classes = [c for c in module.__dict__.values() if
_is_plugin_class(c)]
for plugin_class in plugin_classes:
self.plugins.append(plugin_class(self))
def load_file(self, name):
try:
plugin_module = _get_module_name(self._plugin_directory, name)
namespace = __import__(plugin_module)
except ImportError:
return
self.load_module(getattr(namespace, name))
Each plugin must inherit from VedoPlugin, and is loaded by the Vedo class.
Here's an example plugin:
plugins/first.py
class MyFirstPlugin(VedoPlugin):
def __init__(self, vedo):
super(MyFirstPlugin, self).__init__(vedo)
And a quick example of how to use Vedo:
vedo = Vedo("./plugins")
vedo.load_file("first")
vedo.plugins[0] # This is an instance of `MyFirstPlugin'
Here's some things I'm not too fond of, but I'm not sure what I can do about them:
- Should the private functions in the top of
vedo.pybe moved inside theVedoclass? - Should
load_modulebe private? It's only really going to be used when loading files. - Is there a way to get around the ugliness that is the plugin
__init__function? Every single plugin will need to have that constructor, which I'm not too happy about.
-
\$\begingroup\$ What is the purpose of this class, and how do you use it? Optionally, why do you use it? \$\endgroup\$holroy– holroy2015年11月02日 15:49:48 +00:00Commented Nov 2, 2015 at 15:49
-
\$\begingroup\$ The purpose of the class is to load plugins into an array in a (relatively safe) way. I've added an example of how I use it (normally the plugins listen for events on Vedo passed in their constructor) when loading plugins. I use it as it allows other people to develop plugins and drop them into the project. \$\endgroup\$Jack Wilsdon– Jack Wilsdon2015年11月02日 19:37:57 +00:00Commented Nov 2, 2015 at 19:37
1 Answer 1
Regarding your questions:
- No, I think the functions as they are are fine, not everything needs to be moved into a class.
- If it's an implementation detail, sure.
- AFAIK no, but if it's otherwise empty the method doesn't have to be added.
Also:
In
load_modulethe loop can be more compact by using a combination oflist.extendandfilter.def load_module(self, module): self.plugins.extend(filter(_is_plugin_class, module.__dict__.values()))- The numbered arguments for
str.formatcould be left out.
Looks good otherwise. If you're really concerned with keeping the
internals safe you could also not expose the plugins list directly,
but provide a separate accessor instead.
-
1\$\begingroup\$ By "provide a separate accessor", what do you mean?
@propertymakes the value itself read-only, however you could still modify the array using it's methods. Would it be better to return a copy of the array? \$\endgroup\$Jack Wilsdon– Jack Wilsdon2015年12月05日 00:14:51 +00:00Commented Dec 5, 2015 at 0:14 -
\$\begingroup\$ Depends on whether your users could get hold of the list. In that case maybe have a
get_pluginmethod, implement array access on the class, or yes, return a new list copy every time. \$\endgroup\$ferada– ferada2015年12月05日 00:18:44 +00:00Commented Dec 5, 2015 at 0:18 -
\$\begingroup\$ The issue is that I'm not sure if I should "trust" the installed plugins or try to sandbox them as much as possible. I guess there is no proper way to sandbox them, and plugins can only be installed by the user anyway, so hopefully they know what they are doing. \$\endgroup\$Jack Wilsdon– Jack Wilsdon2015年12月05日 00:19:59 +00:00Commented Dec 5, 2015 at 0:19
You must log in to answer this question.
Explore related questions
See similar questions with these tags.