mirror of
https://github.com/open-webui/open-webui
synced 2024-11-24 21:13:59 +00:00
refac
This commit is contained in:
parent
27f8afebab
commit
43e08c6afa
@ -64,7 +64,6 @@ class FunctionResponse(BaseModel):
|
|||||||
class FunctionForm(BaseModel):
|
class FunctionForm(BaseModel):
|
||||||
id: str
|
id: str
|
||||||
name: str
|
name: str
|
||||||
type: str
|
|
||||||
content: str
|
content: str
|
||||||
meta: FunctionMeta
|
meta: FunctionMeta
|
||||||
|
|
||||||
@ -75,12 +74,13 @@ class FunctionsTable:
|
|||||||
self.db.create_tables([Function])
|
self.db.create_tables([Function])
|
||||||
|
|
||||||
def insert_new_function(
|
def insert_new_function(
|
||||||
self, user_id: str, form_data: FunctionForm
|
self, user_id: str, type: str, form_data: FunctionForm
|
||||||
) -> Optional[FunctionModel]:
|
) -> Optional[FunctionModel]:
|
||||||
function = FunctionModel(
|
function = FunctionModel(
|
||||||
**{
|
**{
|
||||||
**form_data.model_dump(),
|
**form_data.model_dump(),
|
||||||
"user_id": user_id,
|
"user_id": user_id,
|
||||||
|
"type": type,
|
||||||
"updated_at": int(time.time()),
|
"updated_at": int(time.time()),
|
||||||
"created_at": int(time.time()),
|
"created_at": int(time.time()),
|
||||||
}
|
}
|
||||||
|
@ -69,12 +69,12 @@ async def create_new_function(
|
|||||||
with open(function_path, "w") as function_file:
|
with open(function_path, "w") as function_file:
|
||||||
function_file.write(form_data.content)
|
function_file.write(form_data.content)
|
||||||
|
|
||||||
function_module = load_function_module_by_id(form_data.id)
|
function_module, function_type = load_function_module_by_id(form_data.id)
|
||||||
|
|
||||||
FUNCTIONS = request.app.state.FUNCTIONS
|
FUNCTIONS = request.app.state.FUNCTIONS
|
||||||
FUNCTIONS[form_data.id] = function_module
|
FUNCTIONS[form_data.id] = function_module
|
||||||
|
|
||||||
function = Functions.insert_new_function(user.id, form_data)
|
function = Functions.insert_new_function(user.id, function_type, form_data)
|
||||||
|
|
||||||
function_cache_dir = Path(CACHE_DIR) / "functions" / form_data.id
|
function_cache_dir = Path(CACHE_DIR) / "functions" / form_data.id
|
||||||
function_cache_dir.mkdir(parents=True, exist_ok=True)
|
function_cache_dir.mkdir(parents=True, exist_ok=True)
|
||||||
@ -132,12 +132,12 @@ async def update_toolkit_by_id(
|
|||||||
with open(function_path, "w") as function_file:
|
with open(function_path, "w") as function_file:
|
||||||
function_file.write(form_data.content)
|
function_file.write(form_data.content)
|
||||||
|
|
||||||
function_module = load_function_module_by_id(id)
|
function_module, function_type = load_function_module_by_id(id)
|
||||||
|
|
||||||
FUNCTIONS = request.app.state.FUNCTIONS
|
FUNCTIONS = request.app.state.FUNCTIONS
|
||||||
FUNCTIONS[id] = function_module
|
FUNCTIONS[id] = function_module
|
||||||
|
|
||||||
updated = {**form_data.model_dump(exclude={"id"})}
|
updated = {**form_data.model_dump(exclude={"id"}), "type": function_type}
|
||||||
print(updated)
|
print(updated)
|
||||||
|
|
||||||
function = Functions.update_function_by_id(id, updated)
|
function = Functions.update_function_by_id(id, updated)
|
||||||
|
@ -33,9 +33,9 @@ def load_function_module_by_id(function_id):
|
|||||||
spec.loader.exec_module(module)
|
spec.loader.exec_module(module)
|
||||||
print(f"Loaded module: {module.__name__}")
|
print(f"Loaded module: {module.__name__}")
|
||||||
if hasattr(module, "Pipe"):
|
if hasattr(module, "Pipe"):
|
||||||
return module.Pipe()
|
return module.Pipe(), "pipe"
|
||||||
elif hasattr(module, "Filter"):
|
elif hasattr(module, "Filter"):
|
||||||
return module.Filter()
|
return module.Filter(), "filter"
|
||||||
else:
|
else:
|
||||||
raise Exception("No Function class found")
|
raise Exception("No Function class found")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
@ -74,7 +74,7 @@
|
|||||||
<div>
|
<div>
|
||||||
<a
|
<a
|
||||||
class=" px-2 py-2 rounded-xl border border-gray-200 dark:border-gray-600 dark:border-0 hover:bg-gray-100 dark:bg-gray-800 dark:hover:bg-gray-700 transition font-medium text-sm flex items-center space-x-1"
|
class=" px-2 py-2 rounded-xl border border-gray-200 dark:border-gray-600 dark:border-0 hover:bg-gray-100 dark:bg-gray-800 dark:hover:bg-gray-700 transition font-medium text-sm flex items-center space-x-1"
|
||||||
href="/workspace/tools/create"
|
href="/workspace/functions/create"
|
||||||
>
|
>
|
||||||
<svg
|
<svg
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
@ -0,0 +1,281 @@
|
|||||||
|
<script>
|
||||||
|
import { getContext, createEventDispatcher, onMount } from 'svelte';
|
||||||
|
|
||||||
|
const i18n = getContext('i18n');
|
||||||
|
|
||||||
|
import CodeEditor from '$lib/components/common/CodeEditor.svelte';
|
||||||
|
import { goto } from '$app/navigation';
|
||||||
|
import ConfirmDialog from '$lib/components/common/ConfirmDialog.svelte';
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
let formElement = null;
|
||||||
|
let loading = false;
|
||||||
|
let showConfirm = false;
|
||||||
|
|
||||||
|
export let edit = false;
|
||||||
|
export let clone = false;
|
||||||
|
|
||||||
|
export let id = '';
|
||||||
|
export let name = '';
|
||||||
|
export let meta = {
|
||||||
|
description: ''
|
||||||
|
};
|
||||||
|
export let content = '';
|
||||||
|
|
||||||
|
$: if (name && !edit && !clone) {
|
||||||
|
id = name.replace(/\s+/g, '_').toLowerCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
let codeEditor;
|
||||||
|
let boilerplate = `import os
|
||||||
|
import requests
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
|
||||||
|
class Tools:
|
||||||
|
def __init__(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
# Add your custom tools using pure Python code here, make sure to add type hints
|
||||||
|
# Use Sphinx-style docstrings to document your tools, they will be used for generating tools specifications
|
||||||
|
# Please refer to function_calling_filter_pipeline.py file from pipelines project for an example
|
||||||
|
|
||||||
|
def get_user_name_and_email_and_id(self, __user__: dict = {}) -> str:
|
||||||
|
"""
|
||||||
|
Get the user name, Email and ID from the user object.
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Do not include :param for __user__ in the docstring as it should not be shown in the tool's specification
|
||||||
|
# The session user object will be passed as a parameter when the function is called
|
||||||
|
|
||||||
|
print(__user__)
|
||||||
|
result = ""
|
||||||
|
|
||||||
|
if "name" in __user__:
|
||||||
|
result += f"User: {__user__['name']}"
|
||||||
|
if "id" in __user__:
|
||||||
|
result += f" (ID: {__user__['id']})"
|
||||||
|
if "email" in __user__:
|
||||||
|
result += f" (Email: {__user__['email']})"
|
||||||
|
|
||||||
|
if result == "":
|
||||||
|
result = "User: Unknown"
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
def get_current_time(self) -> str:
|
||||||
|
"""
|
||||||
|
Get the current time in a more human-readable format.
|
||||||
|
:return: The current time.
|
||||||
|
"""
|
||||||
|
|
||||||
|
now = datetime.now()
|
||||||
|
current_time = now.strftime("%I:%M:%S %p") # Using 12-hour format with AM/PM
|
||||||
|
current_date = now.strftime(
|
||||||
|
"%A, %B %d, %Y"
|
||||||
|
) # Full weekday, month name, day, and year
|
||||||
|
|
||||||
|
return f"Current Date and Time = {current_date}, {current_time}"
|
||||||
|
|
||||||
|
def calculator(self, equation: str) -> str:
|
||||||
|
"""
|
||||||
|
Calculate the result of an equation.
|
||||||
|
:param equation: The equation to calculate.
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Avoid using eval in production code
|
||||||
|
# https://nedbatchelder.com/blog/201206/eval_really_is_dangerous.html
|
||||||
|
try:
|
||||||
|
result = eval(equation)
|
||||||
|
return f"{equation} = {result}"
|
||||||
|
except Exception as e:
|
||||||
|
print(e)
|
||||||
|
return "Invalid equation"
|
||||||
|
|
||||||
|
def get_current_weather(self, city: str) -> str:
|
||||||
|
"""
|
||||||
|
Get the current weather for a given city.
|
||||||
|
:param city: The name of the city to get the weather for.
|
||||||
|
:return: The current weather information or an error message.
|
||||||
|
"""
|
||||||
|
api_key = os.getenv("OPENWEATHER_API_KEY")
|
||||||
|
if not api_key:
|
||||||
|
return (
|
||||||
|
"API key is not set in the environment variable 'OPENWEATHER_API_KEY'."
|
||||||
|
)
|
||||||
|
|
||||||
|
base_url = "http://api.openweathermap.org/data/2.5/weather"
|
||||||
|
params = {
|
||||||
|
"q": city,
|
||||||
|
"appid": api_key,
|
||||||
|
"units": "metric", # Optional: Use 'imperial' for Fahrenheit
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = requests.get(base_url, params=params)
|
||||||
|
response.raise_for_status() # Raise HTTPError for bad responses (4xx and 5xx)
|
||||||
|
data = response.json()
|
||||||
|
|
||||||
|
if data.get("cod") != 200:
|
||||||
|
return f"Error fetching weather data: {data.get('message')}"
|
||||||
|
|
||||||
|
weather_description = data["weather"][0]["description"]
|
||||||
|
temperature = data["main"]["temp"]
|
||||||
|
humidity = data["main"]["humidity"]
|
||||||
|
wind_speed = data["wind"]["speed"]
|
||||||
|
|
||||||
|
return f"Weather in {city}: {temperature}°C"
|
||||||
|
except requests.RequestException as e:
|
||||||
|
return f"Error fetching weather data: {str(e)}"
|
||||||
|
`;
|
||||||
|
|
||||||
|
const saveHandler = async () => {
|
||||||
|
loading = true;
|
||||||
|
dispatch('save', {
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
meta,
|
||||||
|
content
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const submitHandler = async () => {
|
||||||
|
if (codeEditor) {
|
||||||
|
const res = await codeEditor.formatPythonCodeHandler();
|
||||||
|
|
||||||
|
if (res) {
|
||||||
|
console.log('Code formatted successfully');
|
||||||
|
saveHandler();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class=" flex flex-col justify-between w-full overflow-y-auto h-full">
|
||||||
|
<div class="mx-auto w-full md:px-0 h-full">
|
||||||
|
<form
|
||||||
|
bind:this={formElement}
|
||||||
|
class=" flex flex-col max-h-[100dvh] h-full"
|
||||||
|
on:submit|preventDefault={() => {
|
||||||
|
if (edit) {
|
||||||
|
submitHandler();
|
||||||
|
} else {
|
||||||
|
showConfirm = true;
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="mb-2.5">
|
||||||
|
<button
|
||||||
|
class="flex space-x-1"
|
||||||
|
on:click={() => {
|
||||||
|
goto('/workspace/tools');
|
||||||
|
}}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div class=" self-center">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 20 20"
|
||||||
|
fill="currentColor"
|
||||||
|
class="w-4 h-4"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
fill-rule="evenodd"
|
||||||
|
d="M17 10a.75.75 0 01-.75.75H5.612l4.158 3.96a.75.75 0 11-1.04 1.08l-5.5-5.25a.75.75 0 010-1.08l5.5-5.25a.75.75 0 111.04 1.08L5.612 9.25H16.25A.75.75 0 0117 10z"
|
||||||
|
clip-rule="evenodd"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div class=" self-center font-medium text-sm">{$i18n.t('Back')}</div>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex flex-col flex-1 overflow-auto h-0 rounded-lg">
|
||||||
|
<div class="w-full mb-2 flex flex-col gap-1.5">
|
||||||
|
<div class="flex gap-2 w-full">
|
||||||
|
<input
|
||||||
|
class="w-full px-3 py-2 text-sm font-medium bg-gray-50 dark:bg-gray-850 dark:text-gray-200 rounded-lg outline-none"
|
||||||
|
type="text"
|
||||||
|
placeholder="Toolkit Name (e.g. My ToolKit)"
|
||||||
|
bind:value={name}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
|
||||||
|
<input
|
||||||
|
class="w-full px-3 py-2 text-sm font-medium disabled:text-gray-300 dark:disabled:text-gray-700 bg-gray-50 dark:bg-gray-850 dark:text-gray-200 rounded-lg outline-none"
|
||||||
|
type="text"
|
||||||
|
placeholder="Toolkit ID (e.g. my_toolkit)"
|
||||||
|
bind:value={id}
|
||||||
|
required
|
||||||
|
disabled={edit}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<input
|
||||||
|
class="w-full px-3 py-2 text-sm font-medium bg-gray-50 dark:bg-gray-850 dark:text-gray-200 rounded-lg outline-none"
|
||||||
|
type="text"
|
||||||
|
placeholder="Toolkit Description (e.g. A toolkit for performing various operations)"
|
||||||
|
bind:value={meta.description}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-2 flex-1 overflow-auto h-0 rounded-lg">
|
||||||
|
<CodeEditor
|
||||||
|
bind:value={content}
|
||||||
|
bind:this={codeEditor}
|
||||||
|
{boilerplate}
|
||||||
|
on:save={() => {
|
||||||
|
if (formElement) {
|
||||||
|
formElement.requestSubmit();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pb-3 flex justify-between">
|
||||||
|
<div class="flex-1 pr-3">
|
||||||
|
<div class="text-xs text-gray-500 line-clamp-2">
|
||||||
|
<span class=" font-semibold dark:text-gray-200">Warning:</span> Tools are a function
|
||||||
|
calling system with arbitrary code execution <br />—
|
||||||
|
<span class=" font-medium dark:text-gray-400"
|
||||||
|
>don't install random tools from sources you don't trust.</span
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="px-3 py-1.5 text-sm font-medium bg-emerald-600 hover:bg-emerald-700 text-gray-50 transition rounded-lg"
|
||||||
|
type="submit"
|
||||||
|
>
|
||||||
|
{$i18n.t('Save')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ConfirmDialog
|
||||||
|
bind:show={showConfirm}
|
||||||
|
on:confirm={() => {
|
||||||
|
submitHandler();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div class="text-sm text-gray-500">
|
||||||
|
<div class=" bg-yellow-500/20 text-yellow-700 dark:text-yellow-200 rounded-lg px-4 py-3">
|
||||||
|
<div>Please carefully review the following warnings:</div>
|
||||||
|
|
||||||
|
<ul class=" mt-1 list-disc pl-4 text-xs">
|
||||||
|
<li>Tools have a function calling system that allows arbitrary code execution.</li>
|
||||||
|
<li>Do not install tools from sources you do not fully trust.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="my-3">
|
||||||
|
I acknowledge that I have read and I understand the implications of my action. I am aware of
|
||||||
|
the risks associated with executing arbitrary code and I have verified the trustworthiness of
|
||||||
|
the source.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</ConfirmDialog>
|
@ -1,18 +1,18 @@
|
|||||||
<script>
|
<script>
|
||||||
import { goto } from '$app/navigation';
|
|
||||||
import { createNewTool, getTools } from '$lib/apis/tools';
|
|
||||||
import ToolkitEditor from '$lib/components/workspace/Tools/ToolkitEditor.svelte';
|
|
||||||
import { tools } from '$lib/stores';
|
|
||||||
import { onMount } from 'svelte';
|
|
||||||
import { toast } from 'svelte-sonner';
|
import { toast } from 'svelte-sonner';
|
||||||
|
import { onMount } from 'svelte';
|
||||||
|
import { goto } from '$app/navigation';
|
||||||
|
|
||||||
|
import { createNewFunction, getFunctions } from '$lib/apis/functions';
|
||||||
|
import FunctionEditor from '$lib/components/workspace/Functions/FunctionEditor.svelte';
|
||||||
|
|
||||||
let mounted = false;
|
let mounted = false;
|
||||||
let clone = false;
|
let clone = false;
|
||||||
let tool = null;
|
let func = null;
|
||||||
|
|
||||||
const saveHandler = async (data) => {
|
const saveHandler = async (data) => {
|
||||||
console.log(data);
|
console.log(data);
|
||||||
const res = await createNewTool(localStorage.token, {
|
const res = await createNewFunction(localStorage.token, {
|
||||||
id: data.id,
|
id: data.id,
|
||||||
name: data.name,
|
name: data.name,
|
||||||
meta: data.meta,
|
meta: data.meta,
|
||||||
@ -23,19 +23,17 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (res) {
|
if (res) {
|
||||||
toast.success('Tool created successfully');
|
toast.success('Function created successfully');
|
||||||
tools.set(await getTools(localStorage.token));
|
await goto('/workspace/functions');
|
||||||
|
|
||||||
await goto('/workspace/tools');
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
if (sessionStorage.tool) {
|
if (sessionStorage.function) {
|
||||||
tool = JSON.parse(sessionStorage.tool);
|
func = JSON.parse(sessionStorage.function);
|
||||||
sessionStorage.removeItem('tool');
|
sessionStorage.removeItem('function');
|
||||||
|
|
||||||
console.log(tool);
|
console.log(func);
|
||||||
clone = true;
|
clone = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -44,11 +42,11 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if mounted}
|
{#if mounted}
|
||||||
<ToolkitEditor
|
<FunctionEditor
|
||||||
id={tool?.id ?? ''}
|
id={func?.id ?? ''}
|
||||||
name={tool?.name ?? ''}
|
name={func?.name ?? ''}
|
||||||
meta={tool?.meta ?? { description: '' }}
|
meta={func?.meta ?? { description: '' }}
|
||||||
content={tool?.content ?? ''}
|
content={func?.content ?? ''}
|
||||||
{clone}
|
{clone}
|
||||||
on:save={(e) => {
|
on:save={(e) => {
|
||||||
saveHandler(e.detail);
|
saveHandler(e.detail);
|
||||||
|
Loading…
Reference in New Issue
Block a user