summaryrefslogtreecommitdiff
path: root/plugin_loader/loader.py
blob: 0ed58b39e172abe6d758bda9277aa5de526e214d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
from aiohttp import web
from aiohttp_jinja2 import template

from os import path, listdir
from importlib.util import spec_from_file_location, module_from_spec
from logging import getLogger

import injector

class Loader:
    def __init__(self, server_instance, plugin_path) -> None:
        self.logger = getLogger("Loader")
        self.plugin_path = plugin_path
        self.plugins = self.import_plugins()

        server_instance.add_routes([
            web.get("/plugins/iframe", self.plugin_iframe_route),
            web.get("/plugins/reload", self.reload_plugins),
            web.post("/plugins/method_call", self.handle_plugin_method_call),
            web.get("/plugins/load/{name}", self.load_plugin),
            web.get("/steam_resource/{path:.+}", self.get_steam_resource)
        ])

    def import_plugins(self):
        files = [i for i in listdir(self.plugin_path) if i.endswith(".py")]
        dc = {}
        for file in files:
            try:
                spec = spec_from_file_location("_", path.join(self.plugin_path, file))
                module = module_from_spec(spec)
                spec.loader.exec_module(module)
                dc[module.Plugin.name] = module.Plugin
                self.logger.info("Loaded {}".format(module.Plugin.name))
            except Exception as e:
                self.logger.error("Could not load {}. {}".format(file, e))
        return dc

    async def reload_plugins(self, request=None):
        self.logger.info("Re-importing all plugins.")
        self.plugins = self.import_plugins()

    async def handle_plugin_method_call(self, plugin_name, method_name, **kwargs):
        return await getattr(self.plugins[plugin_name], method_name)(**kwargs)

    async def get_steam_resource(self, request):
        tab = (await injector.get_tabs())[0]
        return web.Response(text=await tab.get_steam_resource(f"https://steamloopback.host/{request.match_info['path']}"), content_type="text/html")

    async def load_plugin(self, request):
        plugin = self.plugins[request.match_info["name"]]
        ret = """
        <script src="/static/library.js"></script>
        <script>const plugin_name = '{}' </script>
        {}
        """.format(plugin.name, plugin.main_view_html)
        return web.Response(text=ret, content_type="text/html")

    @template('plugin_view.html')
    async def plugin_iframe_route(self, request):
        return {"plugins": self.plugins.values()}