bitbot-3.11-fork/src/ModuleManager.py

168 lines
6 KiB
Python
Raw Normal View History

import gc, glob, imp, io, inspect, os, sys, uuid
from . import utils
2016-03-29 11:56:58 +00:00
BITBOT_HOOKS_MAGIC = "__bitbot_hooks"
BITBOT_EXPORTS_MAGIC = "__bitbot_exports"
class ModuleException(Exception):
pass
class ModuleWarning(Exception):
pass
class ModuleNotFoundException(ModuleException):
pass
class ModuleNameCollisionException(ModuleException):
pass
class ModuleLoadException(ModuleException):
pass
class ModuleUnloadException(ModuleException):
pass
class ModuleNotLoadedWarning(ModuleWarning):
pass
class BaseModule(object):
def __init__(self, bot, events, exports, timers):
self.bot = bot
self.events = events
self.exports = exports
self.timers = timers
self.on_load()
def on_load(self):
pass
2016-03-29 11:56:58 +00:00
class ModuleManager(object):
def __init__(self, events, exports, timers, config, log, directory):
self.events = events
self.exports = exports
2018-09-28 15:51:36 +00:00
self.config = config
self.timers = timers
2018-09-28 15:51:36 +00:00
self.log = log
2016-03-29 11:56:58 +00:00
self.directory = directory
2018-09-28 15:51:36 +00:00
2016-03-29 11:56:58 +00:00
self.modules = {}
self.waiting_requirement = {}
2018-09-28 15:51:36 +00:00
2016-03-29 11:56:58 +00:00
def list_modules(self):
return sorted(glob.glob(os.path.join(self.directory, "*.py")))
def _module_name(self, path):
return os.path.basename(path).rsplit(".py", 1)[0].lower()
def _module_path(self, name):
return os.path.join(self.directory, "%s.py" % name)
def _import_name(self, name):
return "bitbot_%s" % name
def _get_magic(self, obj, magic, default):
return getattr(obj, magic) if hasattr(obj, magic) else default
2018-09-28 15:51:36 +00:00
def _load_module(self, bot, name):
path = self._module_path(name)
for hashflag, value in utils.get_hashflags(path):
if hashflag == "ignore":
# nope, ignore this module.
raise ModuleNotLoadedWarning("module ignored")
elif hashflag == "require-config" and value:
if not self.config.get(value.lower(), None):
# nope, required config option not present.
raise ModuleNotLoadedWarning("required config not present")
elif hashflag == "require-module" and value:
requirement = value.lower()
if not requirement in self.modules:
if not requirement in self.waiting_requirement:
self.waiting_requirement[requirement] = set([])
self.waiting_requirement[requirement].add(path)
raise ModuleNotLoadedWarning("waiting for requirement")
module = imp.load_source(self._import_name(name), path)
if not hasattr(module, "Module"):
raise ModuleLoadException("module '%s' doesn't have a "
"'Module' class." % name)
if not inspect.isclass(module.Module):
raise ModuleLoadException("module '%s' has a 'Module' attribute "
"but it is not a class." % name)
context = str(uuid.uuid4())
context_events = self.events.new_context(context)
context_exports = self.exports.new_context(context)
context_timers = self.timers.new_context(context)
module_object = module.Module(bot, context_events, context_exports,
context_timers)
2016-03-29 11:56:58 +00:00
if not hasattr(module_object, "_name"):
module_object._name = name.title()
for attribute_name in dir(module_object):
attribute = getattr(module_object, attribute_name)
for hook in self._get_magic(attribute, BITBOT_HOOKS_MAGIC, []):
context_events.on(hook["event"]).hook(attribute,
**hook["kwargs"])
for export in self._get_magic(module_object, BITBOT_EXPORTS_MAGIC, []):
context_exports.add(export["setting"], export["value"])
module_object._context = context
module_object._import_name = name
if name in self.modules:
raise ModuleNameCollisionException("Module name '%s' "
"attempted to be used twice")
2016-03-29 11:56:58 +00:00
return module_object
2018-09-28 15:51:36 +00:00
def load_module(self, bot, name):
2017-09-05 09:03:38 +00:00
try:
2018-09-28 15:51:36 +00:00
module = self._load_module(bot, name)
except ModuleWarning as warning:
2018-09-28 15:51:36 +00:00
self.log.error("Module '%s' not loaded", [name])
raise
except Exception as e:
2018-09-28 15:51:36 +00:00
self.log.error("Failed to load module \"%s\": %s",
[name, str(e)])
raise
self.modules[module._import_name] = module
if name in self.waiting_requirement:
for requirement_name in self.waiting_requirement:
2018-09-28 15:51:36 +00:00
self.load_module(bot, requirement_name)
self.log.info("Module '%s' loaded", [name])
2017-09-05 09:03:38 +00:00
2018-09-28 15:51:36 +00:00
def load_modules(self, bot, whitelist=[], blacklist=[]):
for path in self.list_modules():
name = self._module_name(path)
if name in whitelist or (not whitelist and not name in blacklist):
try:
2018-09-28 15:51:36 +00:00
self.load_module(bot, name)
except ModuleWarning:
pass
def unload_module(self, name):
if not name in self.modules:
raise ModuleNotFoundException()
module = self.modules[name]
if hasattr(module, "unload"):
try:
module.unload()
except:
pass
del self.modules[name]
context = module._context
self.events.purge_context(context)
self.exports.purge_context(context)
self.timers.purge_context(context)
del sys.modules[self._import_name(name)]
references = sys.getrefcount(module)
referrers = gc.get_referrers(module)
del module
references -= 1 # 'del module' removes one reference
references -= 1 # one of the refs is from getrefcount
2018-09-28 15:51:36 +00:00
self.log.info("Module '%s' unloaded (%d reference%s)",
[name, references, "" if references == 1 else "s"])
if references > 0:
self.log.info("References left for '%s': %s",
[name, ", ".join([str(referrer) for referrer in referrers])])