-
Notifications
You must be signed in to change notification settings - Fork 3.4k
/
Copy pathsequential_planner_extensions.py
124 lines (102 loc) · 5.04 KB
/
sequential_planner_extensions.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
# Copyright (c) Microsoft. All rights reserved.
import logging
from semantic_kernel.functions.kernel_arguments import KernelArguments
from semantic_kernel.functions.kernel_function_metadata import KernelFunctionMetadata
from semantic_kernel.kernel import Kernel
from semantic_kernel.memory.memory_query_result import MemoryQueryResult
from semantic_kernel.planners.sequential_planner.sequential_planner_config import SequentialPlannerConfig
logger: logging.Logger = logging.getLogger(__name__)
class SequentialPlannerFunctionExtension:
"""Function extension for the sequential planner."""
@staticmethod
def to_manual_string(function: KernelFunctionMetadata):
"""Convert the function to a manual string."""
inputs = [
f" - {parameter.name}: {parameter.description}"
+ (f" (default value: {parameter.default_value})" if parameter.default_value else "")
for parameter in function.parameters
]
inputs = "\n".join(inputs)
return f"{function.fully_qualified_name}:\n description: {function.description}\n inputs:\n {inputs}"
@staticmethod
def to_embedding_string(function: KernelFunctionMetadata):
"""Convert the function to an embedding string."""
inputs = "\n".join([f" - {parameter.name}: {parameter.description}" for parameter in function.parameters])
return f"{function.name}:\n description: {function.description}\n inputs:\n{inputs}"
class SequentialPlannerKernelExtension:
"""Kernel extension for the sequential planner."""
PLANNER_MEMORY_COLLECTION_NAME = " Planning.KernelFunctionManual"
PLAN_KERNEL_FUNCTIONS_ARE_REMEMBERED = "Planning.KernelFunctionsAreRemembered"
@staticmethod
async def get_functions_manual(
kernel: "Kernel",
arguments: KernelArguments,
semantic_query: str | None = None,
config: SequentialPlannerConfig = None,
) -> str:
"""Get the functions manual."""
config = config or SequentialPlannerConfig()
if config.get_available_functions is None:
functions = await SequentialPlannerKernelExtension.get_available_functions(
kernel, arguments, config, semantic_query
)
else:
functions = await config.get_available_functions(config, semantic_query)
return "\n\n".join([SequentialPlannerFunctionExtension.to_manual_string(func) for func in functions])
@staticmethod
async def get_available_functions(
kernel: Kernel,
arguments: KernelArguments,
config: SequentialPlannerConfig,
semantic_query: str | None = None,
):
"""Get the available functions based on the semantic query."""
excluded_plugins = config.excluded_plugins or []
excluded_functions = config.excluded_functions or []
included_functions = config.included_functions or []
available_functions = [
func
for func in kernel.get_list_of_function_metadata({"excluded_plugins": excluded_plugins})
if func.name not in excluded_functions
]
if semantic_query is None or config.relevancy_threshold is None:
# If no semantic query is provided, return all available functions.
# If a Memory provider has not been registered, return all available functions.
return available_functions
# Add functions that were found in the search results.
relevant_functions = await SequentialPlannerKernelExtension.get_relevant_functions(
kernel,
available_functions,
)
# Add any missing functions that were included but not found in the search results.
missing_functions = [
func for func in included_functions if func not in [func.name for func in relevant_functions]
]
relevant_functions += [func for func in available_functions if func.name in missing_functions]
return sorted(relevant_functions, key=lambda x: (x.plugin_name, x.name))
@staticmethod
async def get_relevant_functions(
kernel: Kernel,
available_functions: list[KernelFunctionMetadata],
memories: list[MemoryQueryResult] | None = None,
) -> list[KernelFunctionMetadata]:
"""Get relevant functions from the memories."""
relevant_functions = []
# TODO (eavanvalkenburg): cancellation
# https://github.com/microsoft/semantic-kernel/issues/5668
if memories is None:
return relevant_functions
for memory_entry in memories:
function = next(
(func for func in available_functions if func.fully_qualified_name == memory_entry.id),
None,
)
if function is not None:
logger.debug(
"Found relevant function. Relevance Score: {}, Function: {}".format(
memory_entry.relevance,
function.fully_qualified_name,
)
)
relevant_functions.append(function)
return relevant_functions