import os import re import subprocess import sys from importlib import util import types import tempfile import logging from open_webui.env import SRC_LOG_LEVELS from open_webui.models.functions import Functions from open_webui.models.tools import Tools log = logging.getLogger(__name__) log.setLevel(SRC_LOG_LEVELS["MAIN"]) def extract_frontmatter(content): """ Extract frontmatter as a dictionary from the provided content string. """ frontmatter = {} frontmatter_started = False frontmatter_ended = False frontmatter_pattern = re.compile(r"^\s*([a-z_]+):\s*(.*)\s*$", re.IGNORECASE) try: lines = content.splitlines() if len(lines) < 1 or lines[0].strip() != '"""': # The content doesn't start with triple quotes return {} frontmatter_started = True for line in lines[1:]: if '"""' in line: if frontmatter_started: frontmatter_ended = True break if frontmatter_started and not frontmatter_ended: match = frontmatter_pattern.match(line) if match: key, value = match.groups() frontmatter[key.strip()] = value.strip() except Exception as e: print(f"An error occurred: {e}") return {} return frontmatter def replace_imports(content): """ Replace the import paths in the content. """ replacements = { "from utils": "from open_webui.utils", "from apps": "from open_webui.apps", "from main": "from open_webui.main", "from config": "from open_webui.config", } for old, new in replacements.items(): content = content.replace(old, new) return content def load_tools_module_by_id(toolkit_id, content=None): if content is None: tool = Tools.get_tool_by_id(toolkit_id) if not tool: raise Exception(f"Toolkit not found: {toolkit_id}") content = tool.content content = replace_imports(content) Tools.update_tool_by_id(toolkit_id, {"content": content}) else: frontmatter = extract_frontmatter(content) # Install required packages found within the frontmatter install_frontmatter_requirements(frontmatter.get("requirements", "")) module_name = f"tool_{toolkit_id}" module = types.ModuleType(module_name) sys.modules[module_name] = module # Create a temporary file and use it to define `__file__` so # that it works as expected from the module's perspective. temp_file = tempfile.NamedTemporaryFile(delete=False) temp_file.close() try: with open(temp_file.name, "w", encoding="utf-8") as f: f.write(content) module.__dict__["__file__"] = temp_file.name # Executing the modified content in the created module's namespace exec(content, module.__dict__) frontmatter = extract_frontmatter(content) log.info(f"Loaded module: {module.__name__}") # Create and return the object if the class 'Tools' is found in the module if hasattr(module, "Tools"): return module.Tools(), frontmatter else: raise Exception("No Tools class found in the module") except Exception as e: log.error(f"Error loading module: {toolkit_id}: {e}") del sys.modules[module_name] # Clean up raise e finally: os.unlink(temp_file.name) def load_function_module_by_id(function_id, content=None): if content is None: function = Functions.get_function_by_id(function_id) if not function: raise Exception(f"Function not found: {function_id}") content = function.content content = replace_imports(content) Functions.update_function_by_id(function_id, {"content": content}) else: frontmatter = extract_frontmatter(content) install_frontmatter_requirements(frontmatter.get("requirements", "")) module_name = f"function_{function_id}" module = types.ModuleType(module_name) sys.modules[module_name] = module # Create a temporary file and use it to define `__file__` so # that it works as expected from the module's perspective. temp_file = tempfile.NamedTemporaryFile(delete=False) temp_file.close() try: with open(temp_file.name, "w", encoding="utf-8") as f: f.write(content) module.__dict__["__file__"] = temp_file.name # Execute the modified content in the created module's namespace exec(content, module.__dict__) frontmatter = extract_frontmatter(content) log.info(f"Loaded module: {module.__name__}") # Create appropriate object based on available class type in the module if hasattr(module, "Pipe"): return module.Pipe(), "pipe", frontmatter elif hasattr(module, "Filter"): return module.Filter(), "filter", frontmatter elif hasattr(module, "Action"): return module.Action(), "action", frontmatter else: raise Exception("No Function class found in the module") except Exception as e: log.error(f"Error loading module: {function_id}: {e}") del sys.modules[module_name] # Cleanup by removing the module in case of error Functions.update_function_by_id(function_id, {"is_active": False}) raise e finally: os.unlink(temp_file.name) def install_frontmatter_requirements(requirements): if requirements: req_list = [req.strip() for req in requirements.split(",")] for req in req_list: log.info(f"Installing requirement: {req}") subprocess.check_call([sys.executable, "-m", "pip", "install", req]) else: log.info("No requirements found in frontmatter.")