Even more plugin API stuff
This commit is contained in:
parent
0148f74d90
commit
d7f072aeff
3 changed files with 104 additions and 56 deletions
|
@ -30,6 +30,18 @@ class MaubotZipImportError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class MaubotZipMetaError(MaubotZipImportError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class MaubotZipPreLoadError(MaubotZipImportError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class MaubotZipLoadError(MaubotZipImportError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class ZippedPluginLoader(PluginLoader):
|
class ZippedPluginLoader(PluginLoader):
|
||||||
path_cache: Dict[str, 'ZippedPluginLoader'] = {}
|
path_cache: Dict[str, 'ZippedPluginLoader'] = {}
|
||||||
log: logging.Logger = logging.getLogger("maubot.loader.zip")
|
log: logging.Logger = logging.getLogger("maubot.loader.zip")
|
||||||
|
@ -96,16 +108,16 @@ class ZippedPluginLoader(PluginLoader):
|
||||||
file = ZipFile(source)
|
file = ZipFile(source)
|
||||||
data = file.read("maubot.ini")
|
data = file.read("maubot.ini")
|
||||||
except FileNotFoundError as e:
|
except FileNotFoundError as e:
|
||||||
raise MaubotZipImportError("Maubot plugin not found") from e
|
raise MaubotZipMetaError("Maubot plugin not found") from e
|
||||||
except BadZipFile as e:
|
except BadZipFile as e:
|
||||||
raise MaubotZipImportError("File is not a maubot plugin") from e
|
raise MaubotZipMetaError("File is not a maubot plugin") from e
|
||||||
except KeyError as e:
|
except KeyError as e:
|
||||||
raise MaubotZipImportError("File does not contain a maubot plugin definition") from e
|
raise MaubotZipMetaError("File does not contain a maubot plugin definition") from e
|
||||||
config = configparser.ConfigParser()
|
config = configparser.ConfigParser()
|
||||||
try:
|
try:
|
||||||
config.read_string(data.decode("utf-8"))
|
config.read_string(data.decode("utf-8"))
|
||||||
except (configparser.Error, KeyError, IndexError, ValueError) as e:
|
except (configparser.Error, KeyError, IndexError, ValueError) as e:
|
||||||
raise MaubotZipImportError("Maubot plugin definition in file is invalid") from e
|
raise MaubotZipMetaError("Maubot plugin definition in file is invalid") from e
|
||||||
return file, config
|
return file, config
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -120,7 +132,7 @@ class ZippedPluginLoader(PluginLoader):
|
||||||
if "/" in main_class:
|
if "/" in main_class:
|
||||||
main_module, main_class = main_class.split("/")[:2]
|
main_module, main_class = main_class.split("/")[:2]
|
||||||
except (configparser.Error, KeyError, IndexError, ValueError) as e:
|
except (configparser.Error, KeyError, IndexError, ValueError) as e:
|
||||||
raise MaubotZipImportError("Maubot plugin definition in file is invalid") from e
|
raise MaubotZipMetaError("Maubot plugin definition in file is invalid") from e
|
||||||
return meta_id, version, modules, main_class, main_module
|
return meta_id, version, modules, main_class, main_module
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -133,7 +145,7 @@ class ZippedPluginLoader(PluginLoader):
|
||||||
file, config = self._open_meta(self.path)
|
file, config = self._open_meta(self.path)
|
||||||
meta = self._read_meta(config)
|
meta = self._read_meta(config)
|
||||||
if self.id and meta[0] != self.id:
|
if self.id and meta[0] != self.id:
|
||||||
raise MaubotZipImportError("Maubot plugin ID changed during reload")
|
raise MaubotZipMetaError("Maubot plugin ID changed during reload")
|
||||||
self.id, self.version, self.modules, self.main_class, self.main_module = meta
|
self.id, self.version, self.modules, self.main_class, self.main_module = meta
|
||||||
self._file = file
|
self._file = file
|
||||||
|
|
||||||
|
@ -145,22 +157,22 @@ class ZippedPluginLoader(PluginLoader):
|
||||||
self._importer.reset_cache()
|
self._importer.reset_cache()
|
||||||
return self._importer
|
return self._importer
|
||||||
except ZipImportError as e:
|
except ZipImportError as e:
|
||||||
raise MaubotZipImportError("File not found or not a maubot plugin") from e
|
raise MaubotZipMetaError("File not found or not a maubot plugin") from e
|
||||||
|
|
||||||
def _run_preload_checks(self, importer: zipimporter) -> None:
|
def _run_preload_checks(self, importer: zipimporter) -> None:
|
||||||
try:
|
try:
|
||||||
code = importer.get_code(self.main_module.replace(".", "/"))
|
code = importer.get_code(self.main_module.replace(".", "/"))
|
||||||
if self.main_class not in code.co_names:
|
if self.main_class not in code.co_names:
|
||||||
raise MaubotZipImportError(
|
raise MaubotZipPreLoadError(
|
||||||
f"Main class {self.main_class} not in {self.main_module}")
|
f"Main class {self.main_class} not in {self.main_module}")
|
||||||
except ZipImportError as e:
|
except ZipImportError as e:
|
||||||
raise MaubotZipImportError(
|
raise MaubotZipPreLoadError(
|
||||||
f"Main module {self.main_module} not found in file") from e
|
f"Main module {self.main_module} not found in file") from e
|
||||||
for module in self.modules:
|
for module in self.modules:
|
||||||
try:
|
try:
|
||||||
importer.find_module(module)
|
importer.find_module(module)
|
||||||
except ZipImportError as e:
|
except ZipImportError as e:
|
||||||
raise MaubotZipImportError(f"Module {module} not found in file") from e
|
raise MaubotZipPreLoadError(f"Module {module} not found in file") from e
|
||||||
|
|
||||||
async def load(self, reset_cache: bool = False) -> Type[PluginClass]:
|
async def load(self, reset_cache: bool = False) -> Type[PluginClass]:
|
||||||
try:
|
try:
|
||||||
|
@ -175,13 +187,22 @@ class ZippedPluginLoader(PluginLoader):
|
||||||
importer = self._get_importer(reset_cache=reset_cache)
|
importer = self._get_importer(reset_cache=reset_cache)
|
||||||
self._run_preload_checks(importer)
|
self._run_preload_checks(importer)
|
||||||
if reset_cache:
|
if reset_cache:
|
||||||
self.log.debug(f"Preloaded plugin {self.id} from {self.path}")
|
self.log.debug(f"Re-preloaded plugin {self.id} from {self.path}")
|
||||||
for module in self.modules:
|
for module in self.modules:
|
||||||
importer.load_module(module)
|
try:
|
||||||
main_mod = sys.modules[self.main_module]
|
importer.load_module(module)
|
||||||
plugin = getattr(main_mod, self.main_class)
|
except ZipImportError as e:
|
||||||
|
raise MaubotZipLoadError(f"Module {module} not found in file")
|
||||||
|
try:
|
||||||
|
main_mod = sys.modules[self.main_module]
|
||||||
|
except KeyError as e:
|
||||||
|
raise MaubotZipLoadError(f"Main module {self.main_module} of plugin not found") from e
|
||||||
|
try:
|
||||||
|
plugin = getattr(main_mod, self.main_class)
|
||||||
|
except AttributeError as e:
|
||||||
|
raise MaubotZipLoadError(f"Main class {self.main_class} of plugin not found") from e
|
||||||
if not issubclass(plugin, Plugin):
|
if not issubclass(plugin, Plugin):
|
||||||
raise MaubotZipImportError("Main class of plugin does not extend maubot.Plugin")
|
raise MaubotZipLoadError("Main class of plugin does not extend maubot.Plugin")
|
||||||
self._loaded = plugin
|
self._loaded = plugin
|
||||||
self.log.debug(f"Loaded and imported plugin {self.id} from {self.path}")
|
self.log.debug(f"Loaded and imported plugin {self.id} from {self.path}")
|
||||||
return plugin
|
return plugin
|
||||||
|
|
|
@ -15,11 +15,12 @@
|
||||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
from aiohttp import web
|
from aiohttp import web
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
|
import traceback
|
||||||
import os.path
|
import os.path
|
||||||
|
|
||||||
from ...loader import PluginLoader, ZippedPluginLoader, MaubotZipImportError
|
from ...loader import PluginLoader, ZippedPluginLoader, MaubotZipImportError
|
||||||
from .responses import (ErrPluginNotFound, ErrPluginInUse, ErrInputPluginInvalid,
|
from .responses import (ErrPluginNotFound, ErrPluginInUse, plugin_import_error,
|
||||||
ErrPluginReloadFailed, RespDeleted, RespOK)
|
plugin_reload_error, RespDeleted, RespOK, ErrUnsupportedPluginLoader)
|
||||||
from . import routes, config
|
from . import routes, config
|
||||||
|
|
||||||
|
|
||||||
|
@ -62,7 +63,55 @@ async def reload_plugin(request: web.Request) -> web.Response:
|
||||||
plugin = PluginLoader.id_cache.get(plugin_id, None)
|
plugin = PluginLoader.id_cache.get(plugin_id, None)
|
||||||
if not plugin:
|
if not plugin:
|
||||||
return ErrPluginNotFound
|
return ErrPluginNotFound
|
||||||
return await reload(plugin)
|
|
||||||
|
await plugin.stop_instances()
|
||||||
|
try:
|
||||||
|
await plugin.reload()
|
||||||
|
except MaubotZipImportError as e:
|
||||||
|
return plugin_reload_error(str(e), traceback.format_exc())
|
||||||
|
await plugin.start_instances()
|
||||||
|
return RespOK
|
||||||
|
|
||||||
|
|
||||||
|
async def upload_new_plugin(content: bytes, pid: str, version: str) -> web.Response:
|
||||||
|
path = os.path.join(config["plugin_directories.upload"], f"{pid}-v{version}.mbp")
|
||||||
|
with open(path, "wb") as p:
|
||||||
|
p.write(content)
|
||||||
|
try:
|
||||||
|
ZippedPluginLoader.get(path)
|
||||||
|
except MaubotZipImportError as e:
|
||||||
|
ZippedPluginLoader.trash(path)
|
||||||
|
return plugin_import_error(str(e), traceback.format_exc())
|
||||||
|
return RespOK
|
||||||
|
|
||||||
|
|
||||||
|
async def upload_replacement_plugin(plugin: ZippedPluginLoader, content: bytes, new_version: str
|
||||||
|
) -> web.Response:
|
||||||
|
dirname = os.path.dirname(plugin.path)
|
||||||
|
filename = os.path.basename(plugin.path)
|
||||||
|
if plugin.version in filename:
|
||||||
|
filename = filename.replace(plugin.version, new_version)
|
||||||
|
else:
|
||||||
|
filename = filename.rstrip(".mbp") + new_version + ".mbp"
|
||||||
|
path = os.path.join(dirname, filename)
|
||||||
|
with open(path, "wb") as p:
|
||||||
|
p.write(content)
|
||||||
|
old_path = plugin.path
|
||||||
|
plugin.path = path
|
||||||
|
await plugin.stop_instances()
|
||||||
|
try:
|
||||||
|
await plugin.reload()
|
||||||
|
except MaubotZipImportError as e:
|
||||||
|
plugin.path = old_path
|
||||||
|
try:
|
||||||
|
await plugin.reload()
|
||||||
|
except MaubotZipImportError:
|
||||||
|
pass
|
||||||
|
await plugin.start_instances()
|
||||||
|
return plugin_import_error(str(e), traceback.format_exc())
|
||||||
|
await plugin.start_instances()
|
||||||
|
ZippedPluginLoader.trash(plugin.path, reason="update")
|
||||||
|
return RespOK
|
||||||
|
|
||||||
|
|
||||||
@routes.post("/plugins/upload")
|
@routes.post("/plugins/upload")
|
||||||
|
@ -72,40 +121,11 @@ async def upload_plugin(request: web.Request) -> web.Response:
|
||||||
try:
|
try:
|
||||||
pid, version = ZippedPluginLoader.verify_meta(file)
|
pid, version = ZippedPluginLoader.verify_meta(file)
|
||||||
except MaubotZipImportError as e:
|
except MaubotZipImportError as e:
|
||||||
return ErrInputPluginInvalid(e)
|
return plugin_import_error(str(e), traceback.format_exc())
|
||||||
plugin = PluginLoader.id_cache.get(pid, None)
|
plugin = PluginLoader.id_cache.get(pid, None)
|
||||||
if not plugin:
|
if not plugin:
|
||||||
path = os.path.join(config["plugin_directories.upload"], f"{pid}-v{version}.mbp")
|
return await upload_new_plugin(content, pid, version)
|
||||||
with open(path, "wb") as p:
|
|
||||||
p.write(content)
|
|
||||||
try:
|
|
||||||
ZippedPluginLoader.get(path)
|
|
||||||
except MaubotZipImportError as e:
|
|
||||||
ZippedPluginLoader.trash(path)
|
|
||||||
# TODO log error?
|
|
||||||
return ErrInputPluginInvalid(e)
|
|
||||||
elif isinstance(plugin, ZippedPluginLoader):
|
elif isinstance(plugin, ZippedPluginLoader):
|
||||||
dirname = os.path.dirname(plugin.path)
|
return await upload_replacement_plugin(plugin, content, version)
|
||||||
filename = os.path.basename(plugin.path)
|
|
||||||
if plugin.version in filename:
|
|
||||||
filename = filename.replace(plugin.version, version)
|
|
||||||
else:
|
|
||||||
filename = filename.rstrip(".mbp") + version + ".mbp"
|
|
||||||
path = os.path.join(dirname, filename)
|
|
||||||
with open(path, "wb") as p:
|
|
||||||
p.write(content)
|
|
||||||
ZippedPluginLoader.trash(plugin.path, reason="update")
|
|
||||||
plugin.path = path
|
|
||||||
return await reload(plugin)
|
|
||||||
else:
|
else:
|
||||||
return web.json_response({})
|
return ErrUnsupportedPluginLoader
|
||||||
|
|
||||||
|
|
||||||
async def reload(plugin: PluginLoader) -> web.Response:
|
|
||||||
await plugin.stop_instances()
|
|
||||||
try:
|
|
||||||
await plugin.reload()
|
|
||||||
except MaubotZipImportError as e:
|
|
||||||
return ErrPluginReloadFailed(e)
|
|
||||||
await plugin.start_instances()
|
|
||||||
return RespOK
|
|
||||||
|
|
|
@ -36,20 +36,27 @@ ErrPluginInUse = web.json_response({
|
||||||
}, status=web.HTTPPreconditionFailed)
|
}, status=web.HTTPPreconditionFailed)
|
||||||
|
|
||||||
|
|
||||||
def ErrInputPluginInvalid(error) -> web.Response:
|
def plugin_import_error(error: str, stacktrace: str) -> web.Response:
|
||||||
return web.json_response({
|
return web.json_response({
|
||||||
"error": str(error),
|
"error": error,
|
||||||
|
"stacktrace": stacktrace,
|
||||||
"errcode": "plugin_invalid",
|
"errcode": "plugin_invalid",
|
||||||
}, status=web.HTTPBadRequest)
|
}, status=web.HTTPBadRequest)
|
||||||
|
|
||||||
|
|
||||||
def ErrPluginReloadFailed(error) -> web.Response:
|
def plugin_reload_error(error: str, stacktrace: str) -> web.Response:
|
||||||
return web.json_response({
|
return web.json_response({
|
||||||
"error": str(error),
|
"error": error,
|
||||||
"errcode": "plugin_invalid",
|
"stacktrace": stacktrace,
|
||||||
|
"errcode": "plugin_reload_fail",
|
||||||
}, status=web.HTTPInternalServerError)
|
}, status=web.HTTPInternalServerError)
|
||||||
|
|
||||||
|
|
||||||
|
ErrUnsupportedPluginLoader = web.json_response({
|
||||||
|
"error": "Existing plugin with same ID uses unsupported plugin loader",
|
||||||
|
"errcode": "unsupported_plugin_loader",
|
||||||
|
}, status=web.HTTPBadRequest)
|
||||||
|
|
||||||
ErrNotImplemented = web.json_response({
|
ErrNotImplemented = web.json_response({
|
||||||
"error": "Not implemented",
|
"error": "Not implemented",
|
||||||
"errcode": "not_implemented",
|
"errcode": "not_implemented",
|
||||||
|
|
Loading…
Reference in a new issue