Skip to content
GitLab
Menu
Projects
Groups
Snippets
Loading...
Help
Help
Support
Community forum
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Menu
Open sidebar
chenpangpang
open-webui
Commits
43e08c6a
Commit
43e08c6a
authored
Jun 20, 2024
by
Timothy J. Baek
Browse files
refac
parent
27f8afeb
Changes
6
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
308 additions
and
29 deletions
+308
-29
backend/apps/webui/models/functions.py
backend/apps/webui/models/functions.py
+2
-2
backend/apps/webui/routers/functions.py
backend/apps/webui/routers/functions.py
+4
-4
backend/apps/webui/utils.py
backend/apps/webui/utils.py
+2
-2
src/lib/components/workspace/Functions.svelte
src/lib/components/workspace/Functions.svelte
+1
-1
src/lib/components/workspace/Functions/FunctionEditor.svelte
src/lib/components/workspace/Functions/FunctionEditor.svelte
+281
-0
src/routes/(app)/workspace/functions/create/+page.svelte
src/routes/(app)/workspace/functions/create/+page.svelte
+18
-20
No files found.
backend/apps/webui/models/functions.py
View file @
43e08c6a
...
@@ -64,7 +64,6 @@ class FunctionResponse(BaseModel):
...
@@ -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:
...
@@ -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
()),
}
}
...
...
backend/apps/webui/routers/functions.py
View file @
43e08c6a
...
@@ -69,12 +69,12 @@ async def create_new_function(
...
@@ -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(
...
@@ -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
)
...
...
backend/apps/webui/utils.py
View file @
43e08c6a
...
@@ -33,9 +33,9 @@ def load_function_module_by_id(function_id):
...
@@ -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
:
...
...
src/lib/components/workspace/Functions.svelte
View file @
43e08c6a
...
@@ -74,7 +74,7 @@
...
@@ -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/
tool
s/create"
href="/workspace/
function
s/create"
>
>
<svg
<svg
xmlns="http://www.w3.org/2000/svg"
xmlns="http://www.w3.org/2000/svg"
...
...
src/lib/components/workspace/Functions/FunctionEditor.svelte
View file @
43e08c6a
<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>
src/routes/(app)/workspace/functions/create/+page.svelte
View file @
43e08c6a
<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 createNew
Tool
(localStorage.token, {
const res = await createNew
Function
(localStorage.token, {
id: data.id,
id: data.id,
name: data.name,
name: data.name,
meta: data.meta,
meta: data.meta,
...
@@ -23,19 +23,17 @@
...
@@ -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 @@
...
@@ -44,11 +42,11 @@
</script>
</script>
{#if mounted}
{#if mounted}
<
Toolkit
Editor
<
Function
Editor
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);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
.
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment