From 1118e93e15173c1ba7d31f0c93dd86e0797bcc90 Mon Sep 17 00:00:00 2001 From: Steve Nyemba Date: Fri, 7 Feb 2025 22:56:28 -0600 Subject: [PATCH] loader missing --- plugins/loader.py | 100 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 100 insertions(+) create mode 100644 plugins/loader.py diff --git a/plugins/loader.py b/plugins/loader.py new file mode 100644 index 0000000..6830ea7 --- /dev/null +++ b/plugins/loader.py @@ -0,0 +1,100 @@ +""" + implementing a plugin loader i.e can load a function from a file given parameters +""" +import importlib as IL +import importlib.util +import os +import json +import shutil +import pandas as pd + +class Loader : + """ + This class is intended to load a plugin and make it available and assess the quality of the developed plugin + """ + + def __init__(self,**_args): + """ + """ + # _names = _args['names'] if 'names' in _args else None + # path = _args['path'] if 'path' in _args else None + # self._names = _names if type(_names) == list else [_names] + self._modules = {} + self._names = [] + if 'file' in _args : + self.load(**_args) + # self._registry = _args['registry'] + + def load (self,**_args): + """ + This function loads a plugin from a given location + :file location of the file + """ + self._modules = {} + self._names = [] + path = _args ['file'] + _decoratorName = None if 'decorator' not in _args else _args['decorator'] + + if os.path.exists(path) : + _alias = path.split(os.sep)[-1] + spec = importlib.util.spec_from_file_location(_alias, path) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) #--loads it into sys.modules + for _name in dir(module) : + if self.isplugin(module,_name,_decoratorName) : + self._modules[_name] = getattr(module,_name) + return self._modules is not None + # self._names [_name] + # def format (self,**_args): + # uri = _args['alias'],_args['name'] + # def set(self,_pointer) : + def set(self,_key) : + """ + This function will set a pointer to the list of modules to be called + This should be used within the context of using the framework as a library + """ + if type(_key).__name__ == 'function': + # + # The pointer is in the code provided by the user and loaded in memory + # + _pointer = _key + _key = 'inline@'+_key.__name__ + # self._names.append(_key.__name__) + else: + _pointer = self._registry.get(key=_key) + + if _pointer : + self._modules[_key] = _pointer + self._names.append(_key) + + def isplugin(self,module,name,attr=None): + """ + This function determines if a module is a recognized plugin + :module module object loaded from importlib + :name name of the functiion of interest + :attr decorator attribute name (if any) + """ + + p = type(getattr(module,name)).__name__ =='function' + q = True if not attr else hasattr(getattr(module,name),attr) + # + # @TODO: add a generated key, and more indepth validation + return p and q + + def has(self,_name): + """ + This will determine if the module name is loaded or not + """ + return _name in self._modules + def names (self): + return list(self._modules.keys()) + def get (self,_name=None): + """ + This functiion determines how many modules loaded vs unloaded given the list of names + """ + return self._modules.get(_name,None) if _name else self._modules + def apply(self,_name,**_args): + _pointer = self.get(_name) + if _pointer : + return _pointer (**_args) if _args else _pointer() +