|
|
|
|
@ -3,6 +3,7 @@ from collections import OrderedDict
|
|
|
|
|
from collections.abc import Callable
|
|
|
|
|
from typing import Any
|
|
|
|
|
|
|
|
|
|
from configs import dify_config
|
|
|
|
|
from core.tools.utils.yaml_utils import load_yaml_file
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -19,6 +20,87 @@ def get_position_map(folder_path: str, *, file_name: str = "_position.yaml") ->
|
|
|
|
|
return {name: index for index, name in enumerate(positions)}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_tool_position_map(folder_path: str, file_name: str = "_position.yaml") -> dict[str, int]:
|
|
|
|
|
"""
|
|
|
|
|
Get the mapping for tools from name to index from a YAML file.
|
|
|
|
|
:param folder_path:
|
|
|
|
|
:param file_name: the YAML file name, default to '_position.yaml'
|
|
|
|
|
:return: a dict with name as key and index as value
|
|
|
|
|
"""
|
|
|
|
|
position_map = get_position_map(folder_path, file_name=file_name)
|
|
|
|
|
|
|
|
|
|
return pin_position_map(
|
|
|
|
|
position_map,
|
|
|
|
|
pin_list=dify_config.POSITION_TOOL_PINS_LIST,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_provider_position_map(folder_path: str, file_name: str = "_position.yaml") -> dict[str, int]:
|
|
|
|
|
"""
|
|
|
|
|
Get the mapping for providers from name to index from a YAML file.
|
|
|
|
|
:param folder_path:
|
|
|
|
|
:param file_name: the YAML file name, default to '_position.yaml'
|
|
|
|
|
:return: a dict with name as key and index as value
|
|
|
|
|
"""
|
|
|
|
|
position_map = get_position_map(folder_path, file_name=file_name)
|
|
|
|
|
return pin_position_map(
|
|
|
|
|
position_map,
|
|
|
|
|
pin_list=dify_config.POSITION_PROVIDER_PINS_LIST,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def pin_position_map(original_position_map: dict[str, int], pin_list: list[str]) -> dict[str, int]:
|
|
|
|
|
"""
|
|
|
|
|
Pin the items in the pin list to the beginning of the position map.
|
|
|
|
|
Overall logic: exclude > include > pin
|
|
|
|
|
:param position_map: the position map to be sorted and filtered
|
|
|
|
|
:param pin_list: the list of pins to be put at the beginning
|
|
|
|
|
:return: the sorted position map
|
|
|
|
|
"""
|
|
|
|
|
positions = sorted(original_position_map.keys(), key=lambda x: original_position_map[x])
|
|
|
|
|
|
|
|
|
|
# Add pins to position map
|
|
|
|
|
position_map = {name: idx for idx, name in enumerate(pin_list)}
|
|
|
|
|
|
|
|
|
|
# Add remaining positions to position map
|
|
|
|
|
start_idx = len(position_map)
|
|
|
|
|
for name in positions:
|
|
|
|
|
if name not in position_map:
|
|
|
|
|
position_map[name] = start_idx
|
|
|
|
|
start_idx += 1
|
|
|
|
|
|
|
|
|
|
return position_map
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def is_filtered(
|
|
|
|
|
include_set: set[str],
|
|
|
|
|
exclude_set: set[str],
|
|
|
|
|
data: Any,
|
|
|
|
|
name_func: Callable[[Any], str],
|
|
|
|
|
) -> bool:
|
|
|
|
|
"""
|
|
|
|
|
Chcek if the object should be filtered out.
|
|
|
|
|
Overall logic: exclude > include > pin
|
|
|
|
|
:param include_set: the set of names to be included
|
|
|
|
|
:param exclude_set: the set of names to be excluded
|
|
|
|
|
:param name_func: the function to get the name of the object
|
|
|
|
|
:param data: the data to be filtered
|
|
|
|
|
:return: True if the object should be filtered out, False otherwise
|
|
|
|
|
"""
|
|
|
|
|
if not data:
|
|
|
|
|
return False
|
|
|
|
|
if not include_set and not exclude_set:
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
name = name_func(data)
|
|
|
|
|
|
|
|
|
|
if name in exclude_set: # exclude_set is prioritized
|
|
|
|
|
return True
|
|
|
|
|
if include_set and name not in include_set: # filter out only if include_set is not empty
|
|
|
|
|
return True
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def sort_by_position_map(
|
|
|
|
|
position_map: dict[str, int],
|
|
|
|
|
data: list[Any],
|
|
|
|
|
|