open-webui/backend/open_webui/utils/plugin.py

176 lines
5.7 KiB
Python
Raw Normal View History

2024-06-11 06:40:27 +00:00
import os
2024-06-24 03:31:40 +00:00
import re
2024-08-08 07:46:14 +00:00
import subprocess
2024-08-27 22:10:27 +00:00
import sys
from importlib import util
2024-09-04 17:55:20 +00:00
import types
2024-09-19 22:12:52 +00:00
import tempfile
import logging
2024-06-11 06:40:27 +00:00
from open_webui.env import SRC_LOG_LEVELS
2024-12-10 08:54:13 +00:00
from open_webui.models.functions import Functions
from open_webui.models.tools import Tools
2024-06-11 06:40:27 +00:00
log = logging.getLogger(__name__)
log.setLevel(SRC_LOG_LEVELS["MAIN"])
2024-06-11 06:40:27 +00:00
2024-09-05 18:35:58 +00:00
def extract_frontmatter(content):
2024-06-24 03:31:40 +00:00
"""
2024-09-05 18:35:58 +00:00
Extract frontmatter as a dictionary from the provided content string.
2024-06-24 03:31:40 +00:00
"""
frontmatter = {}
2024-06-24 03:37:41 +00:00
frontmatter_started = False
frontmatter_ended = False
frontmatter_pattern = re.compile(r"^\s*([a-z_]+):\s*(.*)\s*$", re.IGNORECASE)
try:
2024-09-05 18:35:58 +00:00
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()
2024-06-24 03:37:41 +00:00
except Exception as e:
print(f"An error occurred: {e}")
return {}
2024-06-24 03:31:40 +00:00
return frontmatter
2024-09-04 17:55:20 +00:00
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
2024-09-04 17:55:20 +00:00
2024-11-17 01:54:38 +00:00
def load_tools_module_by_id(toolkit_id, content=None):
2024-09-04 17:55:20 +00:00
if content is None:
tool = Tools.get_tool_by_id(toolkit_id)
2024-09-04 17:55:20 +00:00
if not tool:
raise Exception(f"Toolkit not found: {toolkit_id}")
2024-09-04 17:55:20 +00:00
content = tool.content
2024-09-04 17:57:41 +00:00
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", ""))
2024-09-04 17:55:20 +00:00
2024-09-04 18:00:47 +00:00
module_name = f"tool_{toolkit_id}"
2024-09-04 17:55:20 +00:00
module = types.ModuleType(module_name)
sys.modules[module_name] = module
2024-06-11 06:40:27 +00:00
2024-09-19 22:12:52 +00:00
# Create a temporary file and use it to define `__file__` so
# that it works as expected from the module's perspective.
2024-09-19 22:12:52 +00:00
temp_file = tempfile.NamedTemporaryFile(delete=False)
temp_file.close()
2024-06-11 06:40:27 +00:00
try:
2024-09-19 22:12:52 +00:00
with open(temp_file.name, "w", encoding="utf-8") as f:
f.write(content)
module.__dict__["__file__"] = temp_file.name
2024-09-04 17:55:20 +00:00
# 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__}")
2024-09-04 17:55:20 +00:00
# Create and return the object if the class 'Tools' is found in the module
2024-06-11 06:40:27 +00:00
if hasattr(module, "Tools"):
2024-06-24 03:31:40 +00:00
return module.Tools(), frontmatter
2024-06-11 06:40:27 +00:00
else:
2024-09-04 17:55:20 +00:00
raise Exception("No Tools class found in the module")
2024-06-11 06:40:27 +00:00
except Exception as e:
log.error(f"Error loading module: {toolkit_id}: {e}")
2024-09-04 17:55:20 +00:00
del sys.modules[module_name] # Clean up
2024-06-11 06:40:27 +00:00
raise e
finally:
2024-09-19 22:12:52 +00:00
os.unlink(temp_file.name)
2024-06-20 07:37:02 +00:00
2024-09-04 17:55:20 +00:00
def load_function_module_by_id(function_id, content=None):
if content is None:
function = Functions.get_function_by_id(function_id)
2024-09-04 17:55:20 +00:00
if not function:
raise Exception(f"Function not found: {function_id}")
2024-09-04 17:55:20 +00:00
content = function.content
2024-09-04 17:57:41 +00:00
content = replace_imports(content)
Functions.update_function_by_id(function_id, {"content": content})
else:
frontmatter = extract_frontmatter(content)
install_frontmatter_requirements(frontmatter.get("requirements", ""))
2024-09-04 18:00:47 +00:00
module_name = f"function_{function_id}"
2024-09-04 17:55:20 +00:00
module = types.ModuleType(module_name)
sys.modules[module_name] = module
2024-06-20 07:37:02 +00:00
2024-09-19 22:12:52 +00:00
# Create a temporary file and use it to define `__file__` so
# that it works as expected from the module's perspective.
2024-09-19 22:12:52 +00:00
temp_file = tempfile.NamedTemporaryFile(delete=False)
temp_file.close()
2024-06-20 07:37:02 +00:00
try:
2024-09-19 22:12:52 +00:00
with open(temp_file.name, "w", encoding="utf-8") as f:
f.write(content)
module.__dict__["__file__"] = temp_file.name
2024-09-04 17:55:20 +00:00
# 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__}")
2024-09-04 17:55:20 +00:00
# Create appropriate object based on available class type in the module
2024-06-20 07:37:02 +00:00
if hasattr(module, "Pipe"):
2024-06-24 03:31:40 +00:00
return module.Pipe(), "pipe", frontmatter
2024-06-20 07:37:02 +00:00
elif hasattr(module, "Filter"):
2024-06-24 03:31:40 +00:00
return module.Filter(), "filter", frontmatter
2024-07-12 01:41:00 +00:00
elif hasattr(module, "Action"):
return module.Action(), "action", frontmatter
2024-06-20 07:37:02 +00:00
else:
2024-09-04 17:55:20 +00:00
raise Exception("No Function class found in the module")
2024-06-20 07:37:02 +00:00
except Exception as e:
log.error(f"Error loading module: {function_id}: {e}")
2024-09-04 17:55:20 +00:00
del sys.modules[module_name] # Cleanup by removing the module in case of error
Functions.update_function_by_id(function_id, {"is_active": False})
2024-06-20 07:37:02 +00:00
raise e
finally:
2024-09-19 22:12:52 +00:00
os.unlink(temp_file.name)
2024-08-08 07:46:14 +00:00
2024-08-14 12:49:18 +00:00
2024-08-08 07:46:14 +00:00
def install_frontmatter_requirements(requirements):
if requirements:
2024-08-14 12:49:18 +00:00
req_list = [req.strip() for req in requirements.split(",")]
2024-08-08 07:46:14 +00:00
for req in req_list:
log.info(f"Installing requirement: {req}")
2024-08-08 07:46:14 +00:00
subprocess.check_call([sys.executable, "-m", "pip", "install", req])
else:
log.info("No requirements found in frontmatter.")