2024-08-27 22:10:27 +00:00
|
|
|
import os
|
|
|
|
from pathlib import Path
|
|
|
|
from typing import Optional
|
2024-06-20 07:37:02 +00:00
|
|
|
|
2024-12-10 08:54:13 +00:00
|
|
|
from open_webui.models.functions import (
|
2024-06-20 07:37:02 +00:00
|
|
|
FunctionForm,
|
|
|
|
FunctionModel,
|
|
|
|
FunctionResponse,
|
2024-08-27 22:10:27 +00:00
|
|
|
Functions,
|
2024-06-20 07:37:02 +00:00
|
|
|
)
|
2024-12-12 02:36:59 +00:00
|
|
|
from open_webui.utils.plugin import load_function_module_by_id, replace_imports
|
2024-10-21 04:34:36 +00:00
|
|
|
from open_webui.config import CACHE_DIR
|
2024-09-04 14:54:48 +00:00
|
|
|
from open_webui.constants import ERROR_MESSAGES
|
2024-08-27 22:10:27 +00:00
|
|
|
from fastapi import APIRouter, Depends, HTTPException, Request, status
|
2024-12-09 00:01:56 +00:00
|
|
|
from open_webui.utils.auth import get_admin_user, get_verified_user
|
2024-06-20 07:37:02 +00:00
|
|
|
|
|
|
|
router = APIRouter()
|
|
|
|
|
|
|
|
############################
|
|
|
|
# GetFunctions
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
2024-08-14 12:46:31 +00:00
|
|
|
@router.get("/", response_model=list[FunctionResponse])
|
2024-06-20 07:37:02 +00:00
|
|
|
async def get_functions(user=Depends(get_verified_user)):
|
|
|
|
return Functions.get_functions()
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# ExportFunctions
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
2024-08-14 12:46:31 +00:00
|
|
|
@router.get("/export", response_model=list[FunctionModel])
|
2024-06-20 07:37:02 +00:00
|
|
|
async def get_functions(user=Depends(get_admin_user)):
|
|
|
|
return Functions.get_functions()
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# CreateNewFunction
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/create", response_model=Optional[FunctionResponse])
|
|
|
|
async def create_new_function(
|
|
|
|
request: Request, form_data: FunctionForm, user=Depends(get_admin_user)
|
|
|
|
):
|
|
|
|
if not form_data.id.isidentifier():
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail="Only alphanumeric characters and underscores are allowed in the id",
|
|
|
|
)
|
|
|
|
|
|
|
|
form_data.id = form_data.id.lower()
|
|
|
|
|
|
|
|
function = Functions.get_function_by_id(form_data.id)
|
2024-08-14 12:39:53 +00:00
|
|
|
if function is None:
|
2024-06-20 07:37:02 +00:00
|
|
|
try:
|
2024-09-04 17:57:41 +00:00
|
|
|
form_data.content = replace_imports(form_data.content)
|
2024-06-24 03:31:40 +00:00
|
|
|
function_module, function_type, frontmatter = load_function_module_by_id(
|
2024-09-04 17:55:20 +00:00
|
|
|
form_data.id,
|
|
|
|
content=form_data.content,
|
2024-06-24 03:31:40 +00:00
|
|
|
)
|
|
|
|
form_data.meta.manifest = frontmatter
|
2024-06-20 07:37:02 +00:00
|
|
|
|
|
|
|
FUNCTIONS = request.app.state.FUNCTIONS
|
|
|
|
FUNCTIONS[form_data.id] = function_module
|
|
|
|
|
2024-06-20 07:54:58 +00:00
|
|
|
function = Functions.insert_new_function(user.id, function_type, form_data)
|
2024-06-20 07:37:02 +00:00
|
|
|
|
|
|
|
function_cache_dir = Path(CACHE_DIR) / "functions" / form_data.id
|
|
|
|
function_cache_dir.mkdir(parents=True, exist_ok=True)
|
|
|
|
|
|
|
|
if function:
|
|
|
|
return function
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT("Error creating function"),
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.ID_TAKEN,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# GetFunctionById
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/id/{id}", response_model=Optional[FunctionModel])
|
|
|
|
async def get_function_by_id(id: str, user=Depends(get_admin_user)):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
|
|
|
|
if function:
|
|
|
|
return function
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-06-24 03:31:40 +00:00
|
|
|
############################
|
|
|
|
# ToggleFunctionById
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/id/{id}/toggle", response_model=Optional[FunctionModel])
|
|
|
|
async def toggle_function_by_id(id: str, user=Depends(get_admin_user)):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
if function:
|
|
|
|
function = Functions.update_function_by_id(
|
|
|
|
id, {"is_active": not function.is_active}
|
|
|
|
)
|
|
|
|
|
|
|
|
if function:
|
|
|
|
return function
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT("Error updating function"),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
2024-06-27 20:04:12 +00:00
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# ToggleGlobalById
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/id/{id}/toggle/global", response_model=Optional[FunctionModel])
|
|
|
|
async def toggle_global_by_id(id: str, user=Depends(get_admin_user)):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
if function:
|
|
|
|
function = Functions.update_function_by_id(
|
|
|
|
id, {"is_global": not function.is_global}
|
|
|
|
)
|
|
|
|
|
|
|
|
if function:
|
|
|
|
return function
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT("Error updating function"),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
2024-06-24 03:31:40 +00:00
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# UpdateFunctionById
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/id/{id}/update", response_model=Optional[FunctionModel])
|
|
|
|
async def update_function_by_id(
|
|
|
|
request: Request, id: str, form_data: FunctionForm, user=Depends(get_admin_user)
|
|
|
|
):
|
|
|
|
try:
|
2024-09-04 17:57:41 +00:00
|
|
|
form_data.content = replace_imports(form_data.content)
|
2024-09-04 17:55:20 +00:00
|
|
|
function_module, function_type, frontmatter = load_function_module_by_id(
|
|
|
|
id, content=form_data.content
|
|
|
|
)
|
2024-06-24 03:31:40 +00:00
|
|
|
form_data.meta.manifest = frontmatter
|
|
|
|
|
|
|
|
FUNCTIONS = request.app.state.FUNCTIONS
|
|
|
|
FUNCTIONS[id] = function_module
|
|
|
|
|
|
|
|
updated = {**form_data.model_dump(exclude={"id"}), "type": function_type}
|
|
|
|
print(updated)
|
|
|
|
|
|
|
|
function = Functions.update_function_by_id(id, updated)
|
|
|
|
|
|
|
|
if function:
|
|
|
|
return function
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT("Error updating function"),
|
|
|
|
)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# DeleteFunctionById
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.delete("/id/{id}/delete", response_model=bool)
|
|
|
|
async def delete_function_by_id(
|
|
|
|
request: Request, id: str, user=Depends(get_admin_user)
|
|
|
|
):
|
|
|
|
result = Functions.delete_function_by_id(id)
|
|
|
|
|
|
|
|
if result:
|
|
|
|
FUNCTIONS = request.app.state.FUNCTIONS
|
|
|
|
if id in FUNCTIONS:
|
|
|
|
del FUNCTIONS[id]
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2024-06-24 01:34:42 +00:00
|
|
|
############################
|
|
|
|
# GetFunctionValves
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/id/{id}/valves", response_model=Optional[dict])
|
|
|
|
async def get_function_valves_by_id(id: str, user=Depends(get_admin_user)):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
if function:
|
|
|
|
try:
|
2024-06-24 02:18:13 +00:00
|
|
|
valves = Functions.get_function_valves_by_id(id)
|
|
|
|
return valves
|
2024-06-24 01:34:42 +00:00
|
|
|
except Exception as e:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
2024-06-24 02:02:27 +00:00
|
|
|
# GetFunctionValvesSpec
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/id/{id}/valves/spec", response_model=Optional[dict])
|
|
|
|
async def get_function_valves_spec_by_id(
|
|
|
|
request: Request, id: str, user=Depends(get_admin_user)
|
|
|
|
):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
if function:
|
|
|
|
if id in request.app.state.FUNCTIONS:
|
|
|
|
function_module = request.app.state.FUNCTIONS[id]
|
|
|
|
else:
|
2024-06-24 03:31:40 +00:00
|
|
|
function_module, function_type, frontmatter = load_function_module_by_id(id)
|
2024-06-24 02:02:27 +00:00
|
|
|
request.app.state.FUNCTIONS[id] = function_module
|
|
|
|
|
|
|
|
if hasattr(function_module, "Valves"):
|
|
|
|
Valves = function_module.Valves
|
|
|
|
return Valves.schema()
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# UpdateFunctionValves
|
2024-06-24 01:34:42 +00:00
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/id/{id}/valves/update", response_model=Optional[dict])
|
2024-06-24 02:02:27 +00:00
|
|
|
async def update_function_valves_by_id(
|
|
|
|
request: Request, id: str, form_data: dict, user=Depends(get_admin_user)
|
2024-06-24 01:34:42 +00:00
|
|
|
):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
if function:
|
2024-06-24 02:02:27 +00:00
|
|
|
if id in request.app.state.FUNCTIONS:
|
|
|
|
function_module = request.app.state.FUNCTIONS[id]
|
|
|
|
else:
|
2024-06-24 03:31:40 +00:00
|
|
|
function_module, function_type, frontmatter = load_function_module_by_id(id)
|
2024-06-24 02:02:27 +00:00
|
|
|
request.app.state.FUNCTIONS[id] = function_module
|
|
|
|
|
|
|
|
if hasattr(function_module, "Valves"):
|
|
|
|
Valves = function_module.Valves
|
|
|
|
|
|
|
|
try:
|
2024-06-24 02:05:56 +00:00
|
|
|
form_data = {k: v for k, v in form_data.items() if v is not None}
|
2024-06-24 02:02:27 +00:00
|
|
|
valves = Valves(**form_data)
|
|
|
|
Functions.update_function_valves_by_id(id, valves.model_dump())
|
|
|
|
return valves.model_dump()
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|
|
|
|
else:
|
2024-06-24 01:34:42 +00:00
|
|
|
raise HTTPException(
|
2024-06-24 02:02:27 +00:00
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
2024-06-24 01:34:42 +00:00
|
|
|
)
|
2024-06-24 02:02:27 +00:00
|
|
|
|
2024-06-24 01:34:42 +00:00
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-06-22 18:26:33 +00:00
|
|
|
############################
|
|
|
|
# FunctionUserValves
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/id/{id}/valves/user", response_model=Optional[dict])
|
|
|
|
async def get_function_user_valves_by_id(id: str, user=Depends(get_verified_user)):
|
|
|
|
function = Functions.get_function_by_id(id)
|
|
|
|
if function:
|
|
|
|
try:
|
|
|
|
user_valves = Functions.get_user_valves_by_id_and_user_id(id, user.id)
|
|
|
|
return user_valves
|
|
|
|
except Exception as e:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("/id/{id}/valves/user/spec", response_model=Optional[dict])
|
|
|
|
async def get_function_user_valves_spec_by_id(
|
|
|
|
request: Request, id: str, user=Depends(get_verified_user)
|
|
|
|
):
|
2024-06-22 19:08:32 +00:00
|
|
|
function = Functions.get_function_by_id(id)
|
2024-06-22 18:26:33 +00:00
|
|
|
if function:
|
|
|
|
if id in request.app.state.FUNCTIONS:
|
|
|
|
function_module = request.app.state.FUNCTIONS[id]
|
|
|
|
else:
|
2024-06-24 03:31:40 +00:00
|
|
|
function_module, function_type, frontmatter = load_function_module_by_id(id)
|
2024-06-22 18:26:33 +00:00
|
|
|
request.app.state.FUNCTIONS[id] = function_module
|
|
|
|
|
|
|
|
if hasattr(function_module, "UserValves"):
|
|
|
|
UserValves = function_module.UserValves
|
|
|
|
return UserValves.schema()
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("/id/{id}/valves/user/update", response_model=Optional[dict])
|
|
|
|
async def update_function_user_valves_by_id(
|
|
|
|
request: Request, id: str, form_data: dict, user=Depends(get_verified_user)
|
|
|
|
):
|
2024-06-22 19:08:32 +00:00
|
|
|
function = Functions.get_function_by_id(id)
|
2024-06-22 18:26:33 +00:00
|
|
|
|
|
|
|
if function:
|
|
|
|
if id in request.app.state.FUNCTIONS:
|
|
|
|
function_module = request.app.state.FUNCTIONS[id]
|
|
|
|
else:
|
2024-06-24 03:31:40 +00:00
|
|
|
function_module, function_type, frontmatter = load_function_module_by_id(id)
|
2024-06-22 18:26:33 +00:00
|
|
|
request.app.state.FUNCTIONS[id] = function_module
|
|
|
|
|
|
|
|
if hasattr(function_module, "UserValves"):
|
|
|
|
UserValves = function_module.UserValves
|
|
|
|
|
|
|
|
try:
|
2024-06-24 02:05:56 +00:00
|
|
|
form_data = {k: v for k, v in form_data.items() if v is not None}
|
2024-06-22 18:26:33 +00:00
|
|
|
user_valves = UserValves(**form_data)
|
|
|
|
Functions.update_user_valves_by_id_and_user_id(
|
|
|
|
id, user.id, user_valves.model_dump()
|
|
|
|
)
|
|
|
|
return user_valves.model_dump()
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
|
|
detail=ERROR_MESSAGES.NOT_FOUND,
|
|
|
|
)
|