mirror of
https://github.com/open-webui/open-webui
synced 2025-04-03 12:31:32 +00:00
Merge 078f025830
into 5ce6c8ced3
This commit is contained in:
commit
0ff981e1a7
@ -397,7 +397,12 @@ async def chat_action(request: Request, action_id: str, form_data: dict, user: A
|
|||||||
|
|
||||||
if hasattr(function_module, "valves") and hasattr(function_module, "Valves"):
|
if hasattr(function_module, "valves") and hasattr(function_module, "Valves"):
|
||||||
valves = Functions.get_function_valves_by_id(action_id)
|
valves = Functions.get_function_valves_by_id(action_id)
|
||||||
function_module.valves = function_module.Valves(**(valves if valves else {}))
|
model_valves = (
|
||||||
|
model.get("info", {}).get("meta", {}).get("valves", {}).get(action_id, {})
|
||||||
|
)
|
||||||
|
function_module.valves = function_module.Valves(
|
||||||
|
**(valves if valves else {}), **model_valves
|
||||||
|
)
|
||||||
|
|
||||||
if hasattr(function_module, "action"):
|
if hasattr(function_module, "action"):
|
||||||
try:
|
try:
|
||||||
|
@ -61,6 +61,17 @@ async def process_filter_functions(
|
|||||||
# Apply valves to the function
|
# Apply valves to the function
|
||||||
if hasattr(function_module, "valves") and hasattr(function_module, "Valves"):
|
if hasattr(function_module, "valves") and hasattr(function_module, "Valves"):
|
||||||
valves = Functions.get_function_valves_by_id(filter_id)
|
valves = Functions.get_function_valves_by_id(filter_id)
|
||||||
|
# overwrite global valves with model valves
|
||||||
|
model_valves = (
|
||||||
|
extra_params.get("__model__", {})
|
||||||
|
.get("info", {})
|
||||||
|
.get("meta", {})
|
||||||
|
.get("valves", {})
|
||||||
|
.get("functions", {})
|
||||||
|
.get(filter_id, {})
|
||||||
|
)
|
||||||
|
valves = {**valves, **model_valves}
|
||||||
|
|
||||||
function_module.valves = function_module.Valves(
|
function_module.valves = function_module.Valves(
|
||||||
**(valves if valves else {})
|
**(valves if valves else {})
|
||||||
)
|
)
|
||||||
|
@ -56,6 +56,18 @@ def get_tools(
|
|||||||
extra_params["__id__"] = tool_id
|
extra_params["__id__"] = tool_id
|
||||||
if hasattr(module, "valves") and hasattr(module, "Valves"):
|
if hasattr(module, "valves") and hasattr(module, "Valves"):
|
||||||
valves = Tools.get_tool_valves_by_id(tool_id) or {}
|
valves = Tools.get_tool_valves_by_id(tool_id) or {}
|
||||||
|
# overwrite global valves with model valves
|
||||||
|
valves = {
|
||||||
|
**valves,
|
||||||
|
**(
|
||||||
|
extra_params.get("__model__", {})
|
||||||
|
.get("info", {})
|
||||||
|
.get("meta", {})
|
||||||
|
.get("valves", {})
|
||||||
|
.get("tools", {})
|
||||||
|
.get(tool_id, {})
|
||||||
|
),
|
||||||
|
}
|
||||||
module.valves = module.Valves(**valves)
|
module.valves = module.Valves(**valves)
|
||||||
|
|
||||||
if hasattr(module, "UserValves"):
|
if hasattr(module, "UserValves"):
|
||||||
|
@ -106,12 +106,12 @@
|
|||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
|
||||||
|
|
||||||
{#if (valvesSpec.properties[property]?.description ?? null) !== null}
|
{#if (valvesSpec.properties[property]?.description ?? null) !== null}
|
||||||
<div class="text-xs text-gray-500">
|
<div class="text-xs text-gray-500">
|
||||||
{valvesSpec.properties[property].description}
|
{valvesSpec.properties[property].description}
|
||||||
</div>
|
</div>
|
||||||
|
{/if}
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
{/each}
|
{/each}
|
||||||
|
@ -2,12 +2,17 @@
|
|||||||
import { getContext, onMount } from 'svelte';
|
import { getContext, onMount } from 'svelte';
|
||||||
import Checkbox from '$lib/components/common/Checkbox.svelte';
|
import Checkbox from '$lib/components/common/Checkbox.svelte';
|
||||||
import Tooltip from '$lib/components/common/Tooltip.svelte';
|
import Tooltip from '$lib/components/common/Tooltip.svelte';
|
||||||
|
import Collapsible from '$lib/components/common/Collapsible.svelte';
|
||||||
|
import Valves from '$lib/components/common/Valves.svelte';
|
||||||
|
|
||||||
const i18n = getContext('i18n');
|
const i18n = getContext('i18n');
|
||||||
|
|
||||||
export let actions = [];
|
export let actions = [];
|
||||||
export let selectedActionIds = [];
|
export let selectedActionIds = [];
|
||||||
|
|
||||||
|
export let valvesSpecs = {};
|
||||||
|
export let valves = {};
|
||||||
|
|
||||||
let _actions = {};
|
let _actions = {};
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
@ -33,7 +38,7 @@
|
|||||||
|
|
||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
{#if actions.length > 0}
|
{#if actions.length > 0}
|
||||||
<div class=" flex items-center mt-2 flex-wrap">
|
<div class=" flex flex-col items-left mt-2 flex-wrap">
|
||||||
{#each Object.keys(_actions) as action, actionIdx}
|
{#each Object.keys(_actions) as action, actionIdx}
|
||||||
<div class=" flex items-center gap-2 mr-3">
|
<div class=" flex items-center gap-2 mr-3">
|
||||||
<div class="self-center flex items-center">
|
<div class="self-center flex items-center">
|
||||||
@ -52,6 +57,18 @@
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{#if _actions[action].selected}
|
||||||
|
<Collapsible
|
||||||
|
title={$i18n.t('Valves')}
|
||||||
|
open={false}
|
||||||
|
buttonClassName="w-full"
|
||||||
|
className="mt-2"
|
||||||
|
>
|
||||||
|
<div slot="content">
|
||||||
|
<Valves valvesSpec={valvesSpecs[action]} bind:valves={valves[action]} />
|
||||||
|
</div>
|
||||||
|
</Collapsible>
|
||||||
|
{/if}
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
@ -2,15 +2,22 @@
|
|||||||
import { getContext, onMount } from 'svelte';
|
import { getContext, onMount } from 'svelte';
|
||||||
import Checkbox from '$lib/components/common/Checkbox.svelte';
|
import Checkbox from '$lib/components/common/Checkbox.svelte';
|
||||||
import Tooltip from '$lib/components/common/Tooltip.svelte';
|
import Tooltip from '$lib/components/common/Tooltip.svelte';
|
||||||
|
import Collapsible from '$lib/components/common/Collapsible.svelte';
|
||||||
|
import Valves from '$lib/components/common/Valves.svelte';
|
||||||
|
|
||||||
const i18n = getContext('i18n');
|
const i18n = getContext('i18n');
|
||||||
|
|
||||||
export let filters = [];
|
export let filters = [];
|
||||||
export let selectedFilterIds = [];
|
export let selectedFilterIds = [];
|
||||||
|
|
||||||
|
export let valvesSpecs = {};
|
||||||
|
export let valves = {};
|
||||||
|
|
||||||
let _filters = {};
|
let _filters = {};
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
|
console.log(`valvesSpecs: ${JSON.stringify(valvesSpecs)}`);
|
||||||
|
console.log(`valves: ${JSON.stringify(valves)}`);
|
||||||
_filters = filters.reduce((acc, filter) => {
|
_filters = filters.reduce((acc, filter) => {
|
||||||
acc[filter.id] = {
|
acc[filter.id] = {
|
||||||
...filter,
|
...filter,
|
||||||
@ -34,7 +41,7 @@
|
|||||||
<!-- TODO: Filer order matters -->
|
<!-- TODO: Filer order matters -->
|
||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
{#if filters.length > 0}
|
{#if filters.length > 0}
|
||||||
<div class=" flex items-center mt-2 flex-wrap">
|
<div class=" flex flex-col items-left mt-2 flex-wrap">
|
||||||
{#each Object.keys(_filters) as filter, filterIdx}
|
{#each Object.keys(_filters) as filter, filterIdx}
|
||||||
<div class=" flex items-center gap-2 mr-3">
|
<div class=" flex items-center gap-2 mr-3">
|
||||||
<div class="self-center flex items-center">
|
<div class="self-center flex items-center">
|
||||||
@ -60,6 +67,18 @@
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{#if _filters[filter].selected}
|
||||||
|
<Collapsible
|
||||||
|
title={$i18n.t('Valves')}
|
||||||
|
open={false}
|
||||||
|
buttonClassName="w-full"
|
||||||
|
className="mt-2"
|
||||||
|
>
|
||||||
|
<div slot="content">
|
||||||
|
<Valves valvesSpec={valvesSpecs[filter]} bind:valves={valves[filter]} />
|
||||||
|
</div>
|
||||||
|
</Collapsible>
|
||||||
|
{/if}
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { onMount, getContext, tick } from 'svelte';
|
import { onMount, getContext, tick } from 'svelte';
|
||||||
import { models, tools, functions, knowledge as knowledgeCollections, user } from '$lib/stores';
|
import { models, tools, functions, knowledge as knowledgeCollections, user } from '$lib/stores';
|
||||||
|
import { get } from 'svelte/store';
|
||||||
|
|
||||||
import AdvancedParams from '$lib/components/chat/Settings/Advanced/AdvancedParams.svelte';
|
import AdvancedParams from '$lib/components/chat/Settings/Advanced/AdvancedParams.svelte';
|
||||||
import Tags from '$lib/components/common/Tags.svelte';
|
import Tags from '$lib/components/common/Tags.svelte';
|
||||||
@ -13,6 +14,8 @@
|
|||||||
import { getTools } from '$lib/apis/tools';
|
import { getTools } from '$lib/apis/tools';
|
||||||
import { getFunctions } from '$lib/apis/functions';
|
import { getFunctions } from '$lib/apis/functions';
|
||||||
import { getKnowledgeBases } from '$lib/apis/knowledge';
|
import { getKnowledgeBases } from '$lib/apis/knowledge';
|
||||||
|
import { getToolValvesSpecById } from '$lib/apis/tools';
|
||||||
|
import { getFunctionValvesSpecById } from '$lib/apis/functions';
|
||||||
import AccessControl from '../common/AccessControl.svelte';
|
import AccessControl from '../common/AccessControl.svelte';
|
||||||
import { stringify } from 'postcss';
|
import { stringify } from 'postcss';
|
||||||
import { toast } from 'svelte-sonner';
|
import { toast } from 'svelte-sonner';
|
||||||
@ -64,7 +67,11 @@
|
|||||||
profile_image_url: '/static/favicon.png',
|
profile_image_url: '/static/favicon.png',
|
||||||
description: '',
|
description: '',
|
||||||
suggestion_prompts: null,
|
suggestion_prompts: null,
|
||||||
tags: []
|
tags: [],
|
||||||
|
valves: {
|
||||||
|
functions: {},
|
||||||
|
tools: {}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
params: {
|
params: {
|
||||||
system: ''
|
system: ''
|
||||||
@ -84,6 +91,10 @@
|
|||||||
let toolIds = [];
|
let toolIds = [];
|
||||||
let filterIds = [];
|
let filterIds = [];
|
||||||
let actionIds = [];
|
let actionIds = [];
|
||||||
|
let toolValvesSpecs = {};
|
||||||
|
let functionValvesSpecs = {};
|
||||||
|
let functionValves = {};
|
||||||
|
let toolValves = {};
|
||||||
|
|
||||||
let accessControl = {};
|
let accessControl = {};
|
||||||
|
|
||||||
@ -105,6 +116,10 @@
|
|||||||
|
|
||||||
info.id = id;
|
info.id = id;
|
||||||
info.name = name;
|
info.name = name;
|
||||||
|
info.meta.valves = {};
|
||||||
|
|
||||||
|
console.log(`submitting with new toolValves: ${JSON.stringify(toolValves)}`);
|
||||||
|
console.log(`submitting with new functionValves: ${JSON.stringify(functionValves)}`);
|
||||||
|
|
||||||
if (id === '') {
|
if (id === '') {
|
||||||
toast.error('Model ID is required.');
|
toast.error('Model ID is required.');
|
||||||
@ -131,6 +146,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
info.meta.valves.tools = toolValves;
|
||||||
if (toolIds.length > 0) {
|
if (toolIds.length > 0) {
|
||||||
info.meta.toolIds = toolIds;
|
info.meta.toolIds = toolIds;
|
||||||
} else {
|
} else {
|
||||||
@ -139,6 +155,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
info.meta.valves.functions = functionValves;
|
||||||
if (filterIds.length > 0) {
|
if (filterIds.length > 0) {
|
||||||
info.meta.filterIds = filterIds;
|
info.meta.filterIds = filterIds;
|
||||||
} else {
|
} else {
|
||||||
@ -161,6 +178,16 @@
|
|||||||
delete info.params[key];
|
delete info.params[key];
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
// iterate through all valve settings for both functions and tools, and remove empty settings (aka things that got set back to default)
|
||||||
|
Object.keys(info.meta.valves).forEach((valveType) => {
|
||||||
|
Object.keys(info.meta.valves[valveType]).forEach((functionId) => {
|
||||||
|
Object.keys(info.meta.valves[valveType][functionId]).forEach((key) => {
|
||||||
|
if (info.meta.valves[valveType][functionId][key] === null) {
|
||||||
|
delete info.meta.valves.functions[functionId][key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
await onSubmit(info);
|
await onSubmit(info);
|
||||||
|
|
||||||
@ -173,6 +200,57 @@
|
|||||||
await functions.set(await getFunctions(localStorage.token));
|
await functions.set(await getFunctions(localStorage.token));
|
||||||
await knowledgeCollections.set(await getKnowledgeBases(localStorage.token));
|
await knowledgeCollections.set(await getKnowledgeBases(localStorage.token));
|
||||||
|
|
||||||
|
// Populate the tool valve specs
|
||||||
|
try {
|
||||||
|
const currentTools = get(tools) ?? [];
|
||||||
|
const toolValvePromises = currentTools.map((tool) =>
|
||||||
|
getToolValvesSpecById(localStorage.token, tool.id)
|
||||||
|
);
|
||||||
|
const resolvedToolValves = await Promise.all(toolValvePromises);
|
||||||
|
console.log(`resolvedToolValves: ${resolvedToolValves}`);
|
||||||
|
|
||||||
|
// Convert array of results to an object with tool IDs as keys
|
||||||
|
toolValvesSpecs = resolvedToolValves.reduce((acc, spec, index) => {
|
||||||
|
if (spec) {
|
||||||
|
console.log(
|
||||||
|
`setting index ${index} to ${spec} from ${JSON.stringify(currentTools[index])}`
|
||||||
|
);
|
||||||
|
acc[currentTools[index].id] = spec;
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
console.log(`toolValvesSpecs: ${JSON.stringify(toolValvesSpecs)}`);
|
||||||
|
|
||||||
|
// Initialize tool valves with default values from specs for the selected tools
|
||||||
|
toolValves = model?.meta?.valves?.tools ?? {};
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error loading tool valve specs:', error);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Populate the function valve specs
|
||||||
|
try {
|
||||||
|
// Use get() to access the store's current value in non-reactive context
|
||||||
|
const currentFunctions = get(functions) ?? [];
|
||||||
|
console.log(JSON.stringify(currentFunctions));
|
||||||
|
const functionValvePromises = currentFunctions.map((func) =>
|
||||||
|
getFunctionValvesSpecById(localStorage.token, func.id)
|
||||||
|
);
|
||||||
|
const resolvedFunctionValves = await Promise.all(functionValvePromises);
|
||||||
|
console.log(resolvedFunctionValves);
|
||||||
|
|
||||||
|
// Convert array of results to an object with function IDs as keys
|
||||||
|
functionValvesSpecs = resolvedFunctionValves.reduce((acc, spec, index) => {
|
||||||
|
if (spec) {
|
||||||
|
acc[currentFunctions[index].id] = spec;
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
console.log(`functionValvesSpecs: ${JSON.stringify(functionValvesSpecs)}`);
|
||||||
|
functionValves = model?.meta?.valves?.functions ?? {};
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error loading function valve specs:', error);
|
||||||
|
}
|
||||||
|
|
||||||
// Scroll to top 'workspace-container' element
|
// Scroll to top 'workspace-container' element
|
||||||
const workspaceContainer = document.getElementById('workspace-container');
|
const workspaceContainer = document.getElementById('workspace-container');
|
||||||
if (workspaceContainer) {
|
if (workspaceContainer) {
|
||||||
@ -692,11 +770,18 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="my-2">
|
<div class="my-2">
|
||||||
<ToolsSelector bind:selectedToolIds={toolIds} tools={$tools} />
|
<ToolsSelector
|
||||||
|
valvesSpecs={toolValvesSpecs}
|
||||||
|
bind:valves={toolValves}
|
||||||
|
bind:selectedToolIds={toolIds}
|
||||||
|
tools={$tools}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="my-2">
|
<div class="my-2">
|
||||||
<FiltersSelector
|
<FiltersSelector
|
||||||
|
valvesSpecs={functionValvesSpecs}
|
||||||
|
bind:valves={functionValves}
|
||||||
bind:selectedFilterIds={filterIds}
|
bind:selectedFilterIds={filterIds}
|
||||||
filters={$functions.filter((func) => func.type === 'filter')}
|
filters={$functions.filter((func) => func.type === 'filter')}
|
||||||
/>
|
/>
|
||||||
@ -704,6 +789,8 @@
|
|||||||
|
|
||||||
<div class="my-2">
|
<div class="my-2">
|
||||||
<ActionsSelector
|
<ActionsSelector
|
||||||
|
valvesSpecs={functionValvesSpecs}
|
||||||
|
bind:valves={functionValves}
|
||||||
bind:selectedActionIds={actionIds}
|
bind:selectedActionIds={actionIds}
|
||||||
actions={$functions.filter((func) => func.type === 'action')}
|
actions={$functions.filter((func) => func.type === 'action')}
|
||||||
/>
|
/>
|
||||||
|
@ -1,9 +1,14 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import Checkbox from '$lib/components/common/Checkbox.svelte';
|
import Checkbox from '$lib/components/common/Checkbox.svelte';
|
||||||
import { getContext, onMount } from 'svelte';
|
import { getContext, onMount } from 'svelte';
|
||||||
|
import Collapsible from '$lib/components/common/Collapsible.svelte';
|
||||||
|
import Valves from '$lib/components/common/Valves.svelte';
|
||||||
|
|
||||||
export let tools = [];
|
export let tools = [];
|
||||||
|
|
||||||
|
export let valvesSpecs = {};
|
||||||
|
export let valves = {};
|
||||||
|
|
||||||
let _tools = {};
|
let _tools = {};
|
||||||
|
|
||||||
export let selectedToolIds = [];
|
export let selectedToolIds = [];
|
||||||
@ -33,7 +38,7 @@
|
|||||||
|
|
||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
{#if tools.length > 0}
|
{#if tools.length > 0}
|
||||||
<div class=" flex items-center mt-2 flex-wrap">
|
<div class=" flex flex-col items-left mt-2 flex-wrap">
|
||||||
{#each Object.keys(_tools) as tool, toolIdx}
|
{#each Object.keys(_tools) as tool, toolIdx}
|
||||||
<div class=" flex items-center gap-2 mr-3">
|
<div class=" flex items-center gap-2 mr-3">
|
||||||
<div class="self-center flex items-center">
|
<div class="self-center flex items-center">
|
||||||
@ -50,6 +55,18 @@
|
|||||||
{_tools[tool].name}
|
{_tools[tool].name}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{#if _tools[tool].selected}
|
||||||
|
<Collapsible
|
||||||
|
title={$i18n.t('Valves')}
|
||||||
|
open={false}
|
||||||
|
buttonClassName="w-full"
|
||||||
|
className="mt-2"
|
||||||
|
>
|
||||||
|
<div slot="content">
|
||||||
|
<Valves valvesSpec={valvesSpecs[tool]} bind:valves={valves[tool]} />
|
||||||
|
</div>
|
||||||
|
</Collapsible>
|
||||||
|
{/if}
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
Loading…
Reference in New Issue
Block a user