aiqtoolkit 1.1.0a20250429__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of aiqtoolkit might be problematic. Click here for more details.

Files changed (309) hide show
  1. aiq/agent/__init__.py +0 -0
  2. aiq/agent/base.py +76 -0
  3. aiq/agent/dual_node.py +67 -0
  4. aiq/agent/react_agent/__init__.py +0 -0
  5. aiq/agent/react_agent/agent.py +322 -0
  6. aiq/agent/react_agent/output_parser.py +104 -0
  7. aiq/agent/react_agent/prompt.py +46 -0
  8. aiq/agent/react_agent/register.py +148 -0
  9. aiq/agent/reasoning_agent/__init__.py +0 -0
  10. aiq/agent/reasoning_agent/reasoning_agent.py +224 -0
  11. aiq/agent/register.py +23 -0
  12. aiq/agent/rewoo_agent/__init__.py +0 -0
  13. aiq/agent/rewoo_agent/agent.py +410 -0
  14. aiq/agent/rewoo_agent/prompt.py +108 -0
  15. aiq/agent/rewoo_agent/register.py +158 -0
  16. aiq/agent/tool_calling_agent/__init__.py +0 -0
  17. aiq/agent/tool_calling_agent/agent.py +123 -0
  18. aiq/agent/tool_calling_agent/register.py +105 -0
  19. aiq/builder/__init__.py +0 -0
  20. aiq/builder/builder.py +223 -0
  21. aiq/builder/component_utils.py +303 -0
  22. aiq/builder/context.py +198 -0
  23. aiq/builder/embedder.py +24 -0
  24. aiq/builder/eval_builder.py +116 -0
  25. aiq/builder/evaluator.py +29 -0
  26. aiq/builder/framework_enum.py +24 -0
  27. aiq/builder/front_end.py +73 -0
  28. aiq/builder/function.py +297 -0
  29. aiq/builder/function_base.py +372 -0
  30. aiq/builder/function_info.py +627 -0
  31. aiq/builder/intermediate_step_manager.py +125 -0
  32. aiq/builder/llm.py +25 -0
  33. aiq/builder/retriever.py +25 -0
  34. aiq/builder/user_interaction_manager.py +71 -0
  35. aiq/builder/workflow.py +134 -0
  36. aiq/builder/workflow_builder.py +733 -0
  37. aiq/cli/__init__.py +14 -0
  38. aiq/cli/cli_utils/__init__.py +0 -0
  39. aiq/cli/cli_utils/config_override.py +233 -0
  40. aiq/cli/cli_utils/validation.py +37 -0
  41. aiq/cli/commands/__init__.py +0 -0
  42. aiq/cli/commands/configure/__init__.py +0 -0
  43. aiq/cli/commands/configure/channel/__init__.py +0 -0
  44. aiq/cli/commands/configure/channel/add.py +28 -0
  45. aiq/cli/commands/configure/channel/channel.py +34 -0
  46. aiq/cli/commands/configure/channel/remove.py +30 -0
  47. aiq/cli/commands/configure/channel/update.py +30 -0
  48. aiq/cli/commands/configure/configure.py +33 -0
  49. aiq/cli/commands/evaluate.py +139 -0
  50. aiq/cli/commands/info/__init__.py +14 -0
  51. aiq/cli/commands/info/info.py +37 -0
  52. aiq/cli/commands/info/list_channels.py +32 -0
  53. aiq/cli/commands/info/list_components.py +129 -0
  54. aiq/cli/commands/registry/__init__.py +14 -0
  55. aiq/cli/commands/registry/publish.py +88 -0
  56. aiq/cli/commands/registry/pull.py +118 -0
  57. aiq/cli/commands/registry/registry.py +36 -0
  58. aiq/cli/commands/registry/remove.py +108 -0
  59. aiq/cli/commands/registry/search.py +155 -0
  60. aiq/cli/commands/start.py +250 -0
  61. aiq/cli/commands/uninstall.py +83 -0
  62. aiq/cli/commands/validate.py +47 -0
  63. aiq/cli/commands/workflow/__init__.py +14 -0
  64. aiq/cli/commands/workflow/templates/__init__.py.j2 +0 -0
  65. aiq/cli/commands/workflow/templates/config.yml.j2 +16 -0
  66. aiq/cli/commands/workflow/templates/pyproject.toml.j2 +22 -0
  67. aiq/cli/commands/workflow/templates/register.py.j2 +5 -0
  68. aiq/cli/commands/workflow/templates/workflow.py.j2 +36 -0
  69. aiq/cli/commands/workflow/workflow.py +37 -0
  70. aiq/cli/commands/workflow/workflow_commands.py +307 -0
  71. aiq/cli/entrypoint.py +133 -0
  72. aiq/cli/main.py +44 -0
  73. aiq/cli/register_workflow.py +408 -0
  74. aiq/cli/type_registry.py +869 -0
  75. aiq/data_models/__init__.py +14 -0
  76. aiq/data_models/api_server.py +550 -0
  77. aiq/data_models/common.py +143 -0
  78. aiq/data_models/component.py +46 -0
  79. aiq/data_models/component_ref.py +135 -0
  80. aiq/data_models/config.py +349 -0
  81. aiq/data_models/dataset_handler.py +122 -0
  82. aiq/data_models/discovery_metadata.py +269 -0
  83. aiq/data_models/embedder.py +26 -0
  84. aiq/data_models/evaluate.py +101 -0
  85. aiq/data_models/evaluator.py +26 -0
  86. aiq/data_models/front_end.py +26 -0
  87. aiq/data_models/function.py +30 -0
  88. aiq/data_models/function_dependencies.py +64 -0
  89. aiq/data_models/interactive.py +237 -0
  90. aiq/data_models/intermediate_step.py +269 -0
  91. aiq/data_models/invocation_node.py +38 -0
  92. aiq/data_models/llm.py +26 -0
  93. aiq/data_models/logging.py +26 -0
  94. aiq/data_models/memory.py +26 -0
  95. aiq/data_models/profiler.py +53 -0
  96. aiq/data_models/registry_handler.py +26 -0
  97. aiq/data_models/retriever.py +30 -0
  98. aiq/data_models/step_adaptor.py +64 -0
  99. aiq/data_models/streaming.py +33 -0
  100. aiq/data_models/swe_bench_model.py +54 -0
  101. aiq/data_models/telemetry_exporter.py +26 -0
  102. aiq/embedder/__init__.py +0 -0
  103. aiq/embedder/langchain_client.py +41 -0
  104. aiq/embedder/nim_embedder.py +58 -0
  105. aiq/embedder/openai_embedder.py +42 -0
  106. aiq/embedder/register.py +24 -0
  107. aiq/eval/__init__.py +14 -0
  108. aiq/eval/config.py +42 -0
  109. aiq/eval/dataset_handler/__init__.py +0 -0
  110. aiq/eval/dataset_handler/dataset_downloader.py +106 -0
  111. aiq/eval/dataset_handler/dataset_filter.py +52 -0
  112. aiq/eval/dataset_handler/dataset_handler.py +164 -0
  113. aiq/eval/evaluate.py +322 -0
  114. aiq/eval/evaluator/__init__.py +14 -0
  115. aiq/eval/evaluator/evaluator_model.py +44 -0
  116. aiq/eval/intermediate_step_adapter.py +93 -0
  117. aiq/eval/rag_evaluator/__init__.py +0 -0
  118. aiq/eval/rag_evaluator/evaluate.py +138 -0
  119. aiq/eval/rag_evaluator/register.py +138 -0
  120. aiq/eval/register.py +22 -0
  121. aiq/eval/remote_workflow.py +128 -0
  122. aiq/eval/runtime_event_subscriber.py +52 -0
  123. aiq/eval/swe_bench_evaluator/__init__.py +0 -0
  124. aiq/eval/swe_bench_evaluator/evaluate.py +215 -0
  125. aiq/eval/swe_bench_evaluator/register.py +36 -0
  126. aiq/eval/trajectory_evaluator/__init__.py +0 -0
  127. aiq/eval/trajectory_evaluator/evaluate.py +118 -0
  128. aiq/eval/trajectory_evaluator/register.py +40 -0
  129. aiq/eval/utils/__init__.py +0 -0
  130. aiq/eval/utils/output_uploader.py +131 -0
  131. aiq/eval/utils/tqdm_position_registry.py +40 -0
  132. aiq/front_ends/__init__.py +14 -0
  133. aiq/front_ends/console/__init__.py +14 -0
  134. aiq/front_ends/console/console_front_end_config.py +32 -0
  135. aiq/front_ends/console/console_front_end_plugin.py +107 -0
  136. aiq/front_ends/console/register.py +25 -0
  137. aiq/front_ends/cron/__init__.py +14 -0
  138. aiq/front_ends/fastapi/__init__.py +14 -0
  139. aiq/front_ends/fastapi/fastapi_front_end_config.py +150 -0
  140. aiq/front_ends/fastapi/fastapi_front_end_plugin.py +103 -0
  141. aiq/front_ends/fastapi/fastapi_front_end_plugin_worker.py +574 -0
  142. aiq/front_ends/fastapi/intermediate_steps_subscriber.py +80 -0
  143. aiq/front_ends/fastapi/job_store.py +161 -0
  144. aiq/front_ends/fastapi/main.py +70 -0
  145. aiq/front_ends/fastapi/message_handler.py +279 -0
  146. aiq/front_ends/fastapi/message_validator.py +345 -0
  147. aiq/front_ends/fastapi/register.py +25 -0
  148. aiq/front_ends/fastapi/response_helpers.py +181 -0
  149. aiq/front_ends/fastapi/step_adaptor.py +315 -0
  150. aiq/front_ends/fastapi/websocket.py +148 -0
  151. aiq/front_ends/mcp/__init__.py +14 -0
  152. aiq/front_ends/mcp/mcp_front_end_config.py +32 -0
  153. aiq/front_ends/mcp/mcp_front_end_plugin.py +93 -0
  154. aiq/front_ends/mcp/register.py +27 -0
  155. aiq/front_ends/mcp/tool_converter.py +242 -0
  156. aiq/front_ends/register.py +22 -0
  157. aiq/front_ends/simple_base/__init__.py +14 -0
  158. aiq/front_ends/simple_base/simple_front_end_plugin_base.py +52 -0
  159. aiq/llm/__init__.py +0 -0
  160. aiq/llm/nim_llm.py +45 -0
  161. aiq/llm/openai_llm.py +45 -0
  162. aiq/llm/register.py +22 -0
  163. aiq/llm/utils/__init__.py +14 -0
  164. aiq/llm/utils/env_config_value.py +94 -0
  165. aiq/llm/utils/error.py +17 -0
  166. aiq/memory/__init__.py +20 -0
  167. aiq/memory/interfaces.py +183 -0
  168. aiq/memory/models.py +102 -0
  169. aiq/meta/module_to_distro.json +3 -0
  170. aiq/meta/pypi.md +59 -0
  171. aiq/observability/__init__.py +0 -0
  172. aiq/observability/async_otel_listener.py +270 -0
  173. aiq/observability/register.py +97 -0
  174. aiq/plugins/.namespace +1 -0
  175. aiq/profiler/__init__.py +0 -0
  176. aiq/profiler/callbacks/__init__.py +0 -0
  177. aiq/profiler/callbacks/agno_callback_handler.py +295 -0
  178. aiq/profiler/callbacks/base_callback_class.py +20 -0
  179. aiq/profiler/callbacks/langchain_callback_handler.py +278 -0
  180. aiq/profiler/callbacks/llama_index_callback_handler.py +205 -0
  181. aiq/profiler/callbacks/semantic_kernel_callback_handler.py +238 -0
  182. aiq/profiler/callbacks/token_usage_base_model.py +27 -0
  183. aiq/profiler/data_frame_row.py +51 -0
  184. aiq/profiler/decorators/__init__.py +0 -0
  185. aiq/profiler/decorators/framework_wrapper.py +131 -0
  186. aiq/profiler/decorators/function_tracking.py +254 -0
  187. aiq/profiler/forecasting/__init__.py +0 -0
  188. aiq/profiler/forecasting/config.py +18 -0
  189. aiq/profiler/forecasting/model_trainer.py +75 -0
  190. aiq/profiler/forecasting/models/__init__.py +22 -0
  191. aiq/profiler/forecasting/models/forecasting_base_model.py +40 -0
  192. aiq/profiler/forecasting/models/linear_model.py +196 -0
  193. aiq/profiler/forecasting/models/random_forest_regressor.py +268 -0
  194. aiq/profiler/inference_metrics_model.py +25 -0
  195. aiq/profiler/inference_optimization/__init__.py +0 -0
  196. aiq/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
  197. aiq/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +452 -0
  198. aiq/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +258 -0
  199. aiq/profiler/inference_optimization/data_models.py +386 -0
  200. aiq/profiler/inference_optimization/experimental/__init__.py +0 -0
  201. aiq/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +468 -0
  202. aiq/profiler/inference_optimization/experimental/prefix_span_analysis.py +405 -0
  203. aiq/profiler/inference_optimization/llm_metrics.py +212 -0
  204. aiq/profiler/inference_optimization/prompt_caching.py +163 -0
  205. aiq/profiler/inference_optimization/token_uniqueness.py +107 -0
  206. aiq/profiler/inference_optimization/workflow_runtimes.py +72 -0
  207. aiq/profiler/intermediate_property_adapter.py +102 -0
  208. aiq/profiler/profile_runner.py +433 -0
  209. aiq/profiler/utils.py +184 -0
  210. aiq/registry_handlers/__init__.py +0 -0
  211. aiq/registry_handlers/local/__init__.py +0 -0
  212. aiq/registry_handlers/local/local_handler.py +176 -0
  213. aiq/registry_handlers/local/register_local.py +37 -0
  214. aiq/registry_handlers/metadata_factory.py +60 -0
  215. aiq/registry_handlers/package_utils.py +198 -0
  216. aiq/registry_handlers/pypi/__init__.py +0 -0
  217. aiq/registry_handlers/pypi/pypi_handler.py +251 -0
  218. aiq/registry_handlers/pypi/register_pypi.py +40 -0
  219. aiq/registry_handlers/register.py +21 -0
  220. aiq/registry_handlers/registry_handler_base.py +157 -0
  221. aiq/registry_handlers/rest/__init__.py +0 -0
  222. aiq/registry_handlers/rest/register_rest.py +56 -0
  223. aiq/registry_handlers/rest/rest_handler.py +237 -0
  224. aiq/registry_handlers/schemas/__init__.py +0 -0
  225. aiq/registry_handlers/schemas/headers.py +42 -0
  226. aiq/registry_handlers/schemas/package.py +68 -0
  227. aiq/registry_handlers/schemas/publish.py +63 -0
  228. aiq/registry_handlers/schemas/pull.py +81 -0
  229. aiq/registry_handlers/schemas/remove.py +36 -0
  230. aiq/registry_handlers/schemas/search.py +91 -0
  231. aiq/registry_handlers/schemas/status.py +47 -0
  232. aiq/retriever/__init__.py +0 -0
  233. aiq/retriever/interface.py +37 -0
  234. aiq/retriever/milvus/__init__.py +14 -0
  235. aiq/retriever/milvus/register.py +81 -0
  236. aiq/retriever/milvus/retriever.py +228 -0
  237. aiq/retriever/models.py +74 -0
  238. aiq/retriever/nemo_retriever/__init__.py +14 -0
  239. aiq/retriever/nemo_retriever/register.py +60 -0
  240. aiq/retriever/nemo_retriever/retriever.py +190 -0
  241. aiq/retriever/register.py +22 -0
  242. aiq/runtime/__init__.py +14 -0
  243. aiq/runtime/loader.py +188 -0
  244. aiq/runtime/runner.py +176 -0
  245. aiq/runtime/session.py +116 -0
  246. aiq/settings/__init__.py +0 -0
  247. aiq/settings/global_settings.py +318 -0
  248. aiq/test/.namespace +1 -0
  249. aiq/tool/__init__.py +0 -0
  250. aiq/tool/code_execution/__init__.py +0 -0
  251. aiq/tool/code_execution/code_sandbox.py +188 -0
  252. aiq/tool/code_execution/local_sandbox/Dockerfile.sandbox +60 -0
  253. aiq/tool/code_execution/local_sandbox/__init__.py +13 -0
  254. aiq/tool/code_execution/local_sandbox/local_sandbox_server.py +79 -0
  255. aiq/tool/code_execution/local_sandbox/sandbox.requirements.txt +4 -0
  256. aiq/tool/code_execution/local_sandbox/start_local_sandbox.sh +25 -0
  257. aiq/tool/code_execution/register.py +70 -0
  258. aiq/tool/code_execution/utils.py +100 -0
  259. aiq/tool/datetime_tools.py +42 -0
  260. aiq/tool/document_search.py +141 -0
  261. aiq/tool/github_tools/__init__.py +0 -0
  262. aiq/tool/github_tools/create_github_commit.py +133 -0
  263. aiq/tool/github_tools/create_github_issue.py +87 -0
  264. aiq/tool/github_tools/create_github_pr.py +106 -0
  265. aiq/tool/github_tools/get_github_file.py +106 -0
  266. aiq/tool/github_tools/get_github_issue.py +166 -0
  267. aiq/tool/github_tools/get_github_pr.py +256 -0
  268. aiq/tool/github_tools/update_github_issue.py +100 -0
  269. aiq/tool/mcp/__init__.py +14 -0
  270. aiq/tool/mcp/mcp_client.py +220 -0
  271. aiq/tool/mcp/mcp_tool.py +75 -0
  272. aiq/tool/memory_tools/__init__.py +0 -0
  273. aiq/tool/memory_tools/add_memory_tool.py +67 -0
  274. aiq/tool/memory_tools/delete_memory_tool.py +67 -0
  275. aiq/tool/memory_tools/get_memory_tool.py +72 -0
  276. aiq/tool/nvidia_rag.py +95 -0
  277. aiq/tool/register.py +36 -0
  278. aiq/tool/retriever.py +89 -0
  279. aiq/utils/__init__.py +0 -0
  280. aiq/utils/data_models/__init__.py +0 -0
  281. aiq/utils/data_models/schema_validator.py +58 -0
  282. aiq/utils/debugging_utils.py +43 -0
  283. aiq/utils/exception_handlers/__init__.py +0 -0
  284. aiq/utils/exception_handlers/schemas.py +114 -0
  285. aiq/utils/io/__init__.py +0 -0
  286. aiq/utils/io/yaml_tools.py +50 -0
  287. aiq/utils/metadata_utils.py +74 -0
  288. aiq/utils/producer_consumer_queue.py +178 -0
  289. aiq/utils/reactive/__init__.py +0 -0
  290. aiq/utils/reactive/base/__init__.py +0 -0
  291. aiq/utils/reactive/base/observable_base.py +65 -0
  292. aiq/utils/reactive/base/observer_base.py +55 -0
  293. aiq/utils/reactive/base/subject_base.py +79 -0
  294. aiq/utils/reactive/observable.py +59 -0
  295. aiq/utils/reactive/observer.py +76 -0
  296. aiq/utils/reactive/subject.py +131 -0
  297. aiq/utils/reactive/subscription.py +49 -0
  298. aiq/utils/settings/__init__.py +0 -0
  299. aiq/utils/settings/global_settings.py +197 -0
  300. aiq/utils/type_converter.py +232 -0
  301. aiq/utils/type_utils.py +397 -0
  302. aiq/utils/url_utils.py +27 -0
  303. aiqtoolkit-1.1.0a20250429.dist-info/METADATA +326 -0
  304. aiqtoolkit-1.1.0a20250429.dist-info/RECORD +309 -0
  305. aiqtoolkit-1.1.0a20250429.dist-info/WHEEL +5 -0
  306. aiqtoolkit-1.1.0a20250429.dist-info/entry_points.txt +17 -0
  307. aiqtoolkit-1.1.0a20250429.dist-info/licenses/LICENSE-3rd-party.txt +3686 -0
  308. aiqtoolkit-1.1.0a20250429.dist-info/licenses/LICENSE.md +201 -0
  309. aiqtoolkit-1.1.0a20250429.dist-info/top_level.txt +1 -0
@@ -0,0 +1,869 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2024-2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ import logging
17
+ import typing
18
+ from collections.abc import AsyncIterator
19
+ from collections.abc import Callable
20
+ from contextlib import AbstractAsyncContextManager
21
+ from contextlib import contextmanager
22
+ from copy import deepcopy
23
+ from functools import cached_property
24
+ from logging import Handler
25
+
26
+ from opentelemetry.sdk.trace.export import SpanExporter
27
+ from pydantic import BaseModel
28
+ from pydantic import ConfigDict
29
+ from pydantic import Field
30
+ from pydantic import Tag
31
+ from pydantic import computed_field
32
+ from pydantic import field_validator
33
+
34
+ from aiq.builder.builder import Builder
35
+ from aiq.builder.builder import EvalBuilder
36
+ from aiq.builder.embedder import EmbedderProviderInfo
37
+ from aiq.builder.evaluator import EvaluatorInfo
38
+ from aiq.builder.front_end import FrontEndBase
39
+ from aiq.builder.function import Function
40
+ from aiq.builder.function_base import FunctionBase
41
+ from aiq.builder.function_info import FunctionInfo
42
+ from aiq.builder.llm import LLMProviderInfo
43
+ from aiq.builder.retriever import RetrieverProviderInfo
44
+ from aiq.data_models.common import TypedBaseModelT
45
+ from aiq.data_models.component import AIQComponentEnum
46
+ from aiq.data_models.config import AIQConfig
47
+ from aiq.data_models.discovery_metadata import DiscoveryMetadata
48
+ from aiq.data_models.embedder import EmbedderBaseConfig
49
+ from aiq.data_models.embedder import EmbedderBaseConfigT
50
+ from aiq.data_models.evaluator import EvaluatorBaseConfig
51
+ from aiq.data_models.evaluator import EvaluatorBaseConfigT
52
+ from aiq.data_models.front_end import FrontEndBaseConfig
53
+ from aiq.data_models.front_end import FrontEndConfigT
54
+ from aiq.data_models.function import FunctionBaseConfig
55
+ from aiq.data_models.function import FunctionConfigT
56
+ from aiq.data_models.llm import LLMBaseConfig
57
+ from aiq.data_models.llm import LLMBaseConfigT
58
+ from aiq.data_models.logging import LoggingBaseConfig
59
+ from aiq.data_models.logging import LoggingMethodConfigT
60
+ from aiq.data_models.memory import MemoryBaseConfig
61
+ from aiq.data_models.memory import MemoryBaseConfigT
62
+ from aiq.data_models.registry_handler import RegistryHandlerBaseConfig
63
+ from aiq.data_models.registry_handler import RegistryHandlerBaseConfigT
64
+ from aiq.data_models.retriever import RetrieverBaseConfig
65
+ from aiq.data_models.retriever import RetrieverBaseConfigT
66
+ from aiq.data_models.telemetry_exporter import TelemetryExporterBaseConfig
67
+ from aiq.data_models.telemetry_exporter import TelemetryExporterConfigT
68
+ from aiq.memory.interfaces import MemoryEditor
69
+ from aiq.registry_handlers.registry_handler_base import AbstractRegistryHandler
70
+
71
+ logger = logging.getLogger(__name__)
72
+
73
+ FrontEndBuildCallableT = Callable[[FrontEndConfigT, AIQConfig], AsyncIterator[FrontEndBase]]
74
+ TelemetryExporterBuildCallableT = Callable[[TelemetryExporterConfigT, Builder], AsyncIterator[SpanExporter]]
75
+ LoggingMethodBuildCallableT = Callable[[LoggingMethodConfigT, Builder], AsyncIterator[Handler]]
76
+ FunctionBuildCallableT = Callable[[FunctionConfigT, Builder], AsyncIterator[FunctionInfo | Callable | FunctionBase]]
77
+ LLMProviderBuildCallableT = Callable[[LLMBaseConfigT, Builder], AsyncIterator[LLMProviderInfo]]
78
+ LLMClientBuildCallableT = Callable[[LLMBaseConfigT, Builder], AsyncIterator[typing.Any]]
79
+ EmbedderProviderBuildCallableT = Callable[[EmbedderBaseConfigT, Builder], AsyncIterator[EmbedderProviderInfo]]
80
+ EmbedderClientBuildCallableT = Callable[[EmbedderBaseConfigT, Builder], AsyncIterator[typing.Any]]
81
+ EvaluatorBuildCallableT = Callable[[EvaluatorBaseConfigT, EvalBuilder], AsyncIterator[EvaluatorInfo]]
82
+ MemoryBuildCallableT = Callable[[MemoryBaseConfigT, Builder], AsyncIterator[MemoryEditor]]
83
+ RetrieverProviderBuildCallableT = Callable[[RetrieverBaseConfigT, Builder], AsyncIterator[RetrieverProviderInfo]]
84
+ RetrieverClientBuildCallableT = Callable[[RetrieverBaseConfigT, Builder], AsyncIterator[typing.Any]]
85
+ RegistryHandlerBuildCallableT = Callable[[RegistryHandlerBaseConfigT], AsyncIterator[AbstractRegistryHandler]]
86
+ ToolWrapperBuildCallableT = Callable[[str, Function, Builder], typing.Any]
87
+
88
+ TeleExporterRegisteredCallableT = Callable[[TelemetryExporterConfigT, Builder], AbstractAsyncContextManager[typing.Any]]
89
+ LoggingMethodRegisteredCallableT = Callable[[LoggingMethodConfigT, Builder], AbstractAsyncContextManager[typing.Any]]
90
+ FrontEndRegisteredCallableT = Callable[[FrontEndConfigT, AIQConfig], AbstractAsyncContextManager[FrontEndBase]]
91
+ FunctionRegisteredCallableT = Callable[[FunctionConfigT, Builder],
92
+ AbstractAsyncContextManager[FunctionInfo | Callable | FunctionBase]]
93
+ LLMProviderRegisteredCallableT = Callable[[LLMBaseConfigT, Builder], AbstractAsyncContextManager[LLMProviderInfo]]
94
+ LLMClientRegisteredCallableT = Callable[[LLMBaseConfigT, Builder], AbstractAsyncContextManager[typing.Any]]
95
+ EmbedderProviderRegisteredCallableT = Callable[[EmbedderBaseConfigT, Builder],
96
+ AbstractAsyncContextManager[EmbedderProviderInfo]]
97
+ EmbedderClientRegisteredCallableT = Callable[[EmbedderBaseConfigT, Builder], AbstractAsyncContextManager[typing.Any]]
98
+ EvaluatorRegisteredCallableT = Callable[[EvaluatorBaseConfigT, EvalBuilder], AbstractAsyncContextManager[EvaluatorInfo]]
99
+ MemoryRegisteredCallableT = Callable[[MemoryBaseConfigT, Builder], AbstractAsyncContextManager[MemoryEditor]]
100
+ RetrieverProviderRegisteredCallableT = Callable[[RetrieverBaseConfigT, Builder],
101
+ AbstractAsyncContextManager[RetrieverProviderInfo]]
102
+ RetrieverClientRegisteredCallableT = Callable[[RetrieverBaseConfigT, Builder], AbstractAsyncContextManager[typing.Any]]
103
+ RegistryHandlerRegisteredCallableT = Callable[[RegistryHandlerBaseConfigT],
104
+ AbstractAsyncContextManager[AbstractRegistryHandler]]
105
+
106
+
107
+ class RegisteredInfo(BaseModel, typing.Generic[TypedBaseModelT]):
108
+
109
+ model_config = ConfigDict(frozen=True)
110
+
111
+ full_type: str
112
+ config_type: type[TypedBaseModelT]
113
+ discovery_metadata: DiscoveryMetadata = DiscoveryMetadata()
114
+
115
+ @computed_field
116
+ @cached_property
117
+ def module_name(self) -> str:
118
+ return self.full_type.split("/")[0]
119
+
120
+ @computed_field
121
+ @cached_property
122
+ def local_name(self) -> str:
123
+ return self.full_type.split("/")[-1]
124
+
125
+ @field_validator("full_type", mode="after")
126
+ @classmethod
127
+ def validate_full_type(cls, full_type: str) -> str:
128
+ parts = full_type.split("/")
129
+
130
+ if (len(parts) != 2):
131
+ raise ValueError(f"Invalid full type: {full_type}. Expected format: `module_name/local_name`")
132
+
133
+ return full_type
134
+
135
+
136
+ class RegisteredTelemetryExporter(RegisteredInfo[TelemetryExporterBaseConfig]):
137
+
138
+ build_fn: TeleExporterRegisteredCallableT = Field(repr=False)
139
+
140
+
141
+ class RegisteredLoggingMethod(RegisteredInfo[LoggingBaseConfig]):
142
+
143
+ build_fn: LoggingMethodRegisteredCallableT = Field(repr=False)
144
+
145
+
146
+ class RegisteredFrontEndInfo(RegisteredInfo[FrontEndBaseConfig]):
147
+ """
148
+ Represents a registered front end. Front ends are the entry points to the workflow and are responsible for
149
+ orchestrating the workflow.
150
+ """
151
+
152
+ build_fn: FrontEndRegisteredCallableT = Field(repr=False)
153
+
154
+
155
+ class RegisteredFunctionInfo(RegisteredInfo[FunctionBaseConfig]):
156
+ """
157
+ Represents a registered function. Functions are the building blocks of the workflow with predefined inputs, outputs,
158
+ and a description.
159
+ """
160
+
161
+ build_fn: FunctionRegisteredCallableT = Field(repr=False)
162
+ framework_wrappers: list[str] = Field(default_factory=list)
163
+
164
+
165
+ class RegisteredLLMProviderInfo(RegisteredInfo[LLMBaseConfig]):
166
+ """
167
+ Represents a registered LLM provider. LLM Providers are the operators of the LLMs. i.e. NIMs, OpenAI, Anthropic,
168
+ etc.
169
+ """
170
+
171
+ build_fn: LLMProviderRegisteredCallableT = Field(repr=False)
172
+
173
+
174
+ class RegisteredLLMClientInfo(RegisteredInfo[LLMBaseConfig]):
175
+ """
176
+ Represents a registered LLM client. LLM Clients are the clients that interact with the LLM providers and are
177
+ specific to a particular LLM framework.
178
+ """
179
+
180
+ llm_framework: str
181
+ build_fn: LLMClientRegisteredCallableT = Field(repr=False)
182
+
183
+
184
+ class RegisteredEmbedderProviderInfo(RegisteredInfo[EmbedderBaseConfig]):
185
+ """
186
+ Represents a registered Embedder provider. Embedder Providers are the operators of the Embedder models. i.e. NIMs,
187
+ OpenAI, Anthropic, etc.
188
+ """
189
+
190
+ build_fn: EmbedderProviderRegisteredCallableT = Field(repr=False)
191
+
192
+
193
+ class RegisteredEmbedderClientInfo(RegisteredInfo[EmbedderBaseConfig]):
194
+ """
195
+ Represents a registered Embedder client. Embedder Clients are the clients that interact with the Embedder providers
196
+ and are specific to a particular LLM framework.
197
+ """
198
+
199
+ llm_framework: str
200
+ build_fn: EmbedderClientRegisteredCallableT = Field(repr=False)
201
+
202
+
203
+ class RegisteredEvaluatorInfo(RegisteredInfo[EvaluatorBaseConfig]):
204
+ """
205
+ Represents a registered Evaluator e.g. RagEvaluator, TrajectoryEvaluator, etc.
206
+ """
207
+
208
+ build_fn: EvaluatorRegisteredCallableT = Field(repr=False)
209
+
210
+
211
+ class RegisteredMemoryInfo(RegisteredInfo[MemoryBaseConfig]):
212
+ """
213
+ Represents a registered Memory object which adheres to the memory interface.
214
+ """
215
+
216
+ build_fn: MemoryRegisteredCallableT = Field(repr=False)
217
+
218
+
219
+ class RegisteredToolWrapper(BaseModel):
220
+ """
221
+ Represents a registered tool wrapper. Tool wrappers are used to wrap the functions in a particular LLM framework.
222
+ They do not have their own configuration, but they are used to wrap the functions in a particular LLM framework.
223
+ """
224
+
225
+ llm_framework: str
226
+ build_fn: ToolWrapperBuildCallableT = Field(repr=False)
227
+ discovery_metadata: DiscoveryMetadata
228
+
229
+
230
+ class RegisteredRetrieverProviderInfo(RegisteredInfo[RetrieverBaseConfig]):
231
+ """
232
+ Represents a registered Retriever object which adheres to the retriever interface.
233
+ """
234
+
235
+ build_fn: RetrieverProviderRegisteredCallableT = Field(repr=False)
236
+
237
+
238
+ class RegisteredRetrieverClientInfo(RegisteredInfo[RetrieverBaseConfig]):
239
+ """
240
+ Represents a registered Retriever Client. Retriever Clients are the LLM Framework-specific clients that expose an
241
+ interface to the Retriever object.
242
+ """
243
+ llm_framework: str | None
244
+ build_fn: RetrieverClientRegisteredCallableT = Field(repr=False)
245
+
246
+
247
+ class RegisteredRegistryHandlerInfo(RegisteredInfo[RegistryHandlerBaseConfig]):
248
+ """
249
+ Represents a registered LLM client. LLM Clients are the clients that interact with the LLM providers and are
250
+ specific to a particular LLM framework.
251
+ """
252
+
253
+ build_fn: RegistryHandlerRegisteredCallableT = Field(repr=False)
254
+
255
+
256
+ class RegisteredPackage(BaseModel):
257
+ package_name: str
258
+ discovery_metadata: DiscoveryMetadata
259
+
260
+
261
+ class TypeRegistry: # pylint: disable=too-many-public-methods
262
+
263
+ def __init__(self) -> None:
264
+ # Telemetry Exporters
265
+ self._registered_telemetry_exporters: dict[type[TelemetryExporterBaseConfig], RegisteredTelemetryExporter] = {}
266
+
267
+ # Logging Methods
268
+ self._registered_logging_methods: dict[type[LoggingBaseConfig], RegisteredLoggingMethod] = {}
269
+
270
+ # Front Ends
271
+ self._registered_front_end_infos: dict[type[FrontEndBaseConfig], RegisteredFrontEndInfo] = {}
272
+
273
+ # Functions
274
+ self._registered_functions: dict[type[FunctionBaseConfig], RegisteredFunctionInfo] = {}
275
+
276
+ # LLMs
277
+ self._registered_llm_provider_infos: dict[type[LLMBaseConfig], RegisteredLLMProviderInfo] = {}
278
+ self._llm_client_provider_to_framework: dict[type[LLMBaseConfig], dict[str, RegisteredLLMClientInfo]] = {}
279
+ self._llm_client_framework_to_provider: dict[str, dict[type[LLMBaseConfig], RegisteredLLMClientInfo]] = {}
280
+
281
+ # Embedders
282
+ self._registered_embedder_provider_infos: dict[type[EmbedderBaseConfig], RegisteredEmbedderProviderInfo] = {}
283
+ self._embedder_client_provider_to_framework: dict[type[EmbedderBaseConfig],
284
+ dict[str, RegisteredEmbedderClientInfo]] = {}
285
+ self._embedder_client_framework_to_provider: dict[str,
286
+ dict[type[EmbedderBaseConfig],
287
+ RegisteredEmbedderClientInfo]] = {}
288
+
289
+ # Evaluators
290
+ self._registered_evaluator_infos: dict[type[EvaluatorBaseConfig], RegisteredEvaluatorInfo] = {}
291
+
292
+ # Memory
293
+ self._registered_memory_infos: dict[type[MemoryBaseConfig], RegisteredMemoryInfo] = {}
294
+
295
+ # Retrievers
296
+ self._registered_retriever_provider_infos: dict[type[RetrieverBaseConfig], RegisteredRetrieverProviderInfo] = {}
297
+ self._retriever_client_provider_to_framework: dict[type[RetrieverBaseConfig],
298
+ dict[str | None, RegisteredRetrieverClientInfo]] = {}
299
+ self._retriever_client_framework_to_provider: dict[str | None,
300
+ dict[type[RetrieverBaseConfig],
301
+ RegisteredRetrieverClientInfo]] = {}
302
+
303
+ # Registry Handlers
304
+ self._registered_registry_handler_infos: dict[type[RegistryHandlerBaseConfig],
305
+ RegisteredRegistryHandlerInfo] = {}
306
+
307
+ # Tool Wrappers
308
+ self._registered_tool_wrappers: dict[str, RegisteredToolWrapper] = {}
309
+
310
+ # Packages
311
+ self._registered_packages: dict[str, RegisteredPackage] = {}
312
+
313
+ self._registration_changed_hooks: list[Callable[[], None]] = []
314
+ self._registration_changed_hooks_active: bool = True
315
+
316
+ self._registered_channel_map = {}
317
+
318
+ def _registration_changed(self):
319
+
320
+ if (not self._registration_changed_hooks_active):
321
+ return
322
+
323
+ logger.debug("Registration changed. Notifying hooks.")
324
+
325
+ for hook in self._registration_changed_hooks:
326
+ hook()
327
+
328
+ def add_registration_changed_hook(self, cb: Callable[[], typing.Any]) -> None:
329
+
330
+ self._registration_changed_hooks.append(cb)
331
+
332
+ @contextmanager
333
+ def pause_registration_changed_hooks(self):
334
+
335
+ self._registration_changed_hooks_active = False
336
+
337
+ try:
338
+ yield
339
+ finally:
340
+ self._registration_changed_hooks_active = True
341
+
342
+ # Ensure that the registration changed hooks are called
343
+ self._registration_changed()
344
+
345
+ def register_telemetry_exporter(self, registration: RegisteredTelemetryExporter):
346
+
347
+ if (registration.config_type in self._registered_telemetry_exporters):
348
+ raise ValueError(f"A telemetry exporter with the same config type `{registration.config_type}` has already "
349
+ "been registered.")
350
+
351
+ self._registered_telemetry_exporters[registration.config_type] = registration
352
+
353
+ self._registration_changed()
354
+
355
+ def get_telemetry_exporter(self, config_type: type[TelemetryExporterBaseConfig]) -> RegisteredTelemetryExporter:
356
+
357
+ try:
358
+ return self._registered_telemetry_exporters[config_type]
359
+ except KeyError as err:
360
+ raise KeyError(f"Could not find a registered telemetry exporter for config `{config_type}`. "
361
+ f"Registered configs: {set(self._registered_telemetry_exporters.keys())}") from err
362
+
363
+ def get_registered_telemetry_exporters(self) -> list[RegisteredInfo[TelemetryExporterBaseConfig]]:
364
+
365
+ return list(self._registered_telemetry_exporters.values())
366
+
367
+ def register_logging_method(self, registration: RegisteredLoggingMethod):
368
+
369
+ if (registration.config_type in self._registered_logging_methods):
370
+ raise ValueError(f"A logging method with the same config type `{registration.config_type}` has already "
371
+ "been registered.")
372
+
373
+ self._registered_logging_methods[registration.config_type] = registration
374
+
375
+ self._registration_changed()
376
+
377
+ def get_logging_method(self, config_type: type[LoggingBaseConfig]) -> RegisteredLoggingMethod:
378
+ try:
379
+ return self._registered_logging_methods[config_type]
380
+ except KeyError as err:
381
+ raise KeyError(f"No logging method found for config `{config_type}`. "
382
+ f"Known: {set(self._registered_logging_methods.keys())}") from err
383
+
384
+ def get_registered_logging_method(self) -> list[RegisteredInfo[LoggingBaseConfig]]:
385
+
386
+ return list(self._registered_logging_methods.values())
387
+
388
+ def register_front_end(self, registration: RegisteredFrontEndInfo):
389
+
390
+ if (registration.config_type in self._registered_front_end_infos):
391
+ raise ValueError(f"A front end with the same config type `{registration.config_type}` has already been "
392
+ "registered.")
393
+
394
+ self._registered_front_end_infos[registration.config_type] = registration
395
+
396
+ self._registration_changed()
397
+
398
+ def get_front_end(self, config_type: type[FrontEndBaseConfig]) -> RegisteredFrontEndInfo:
399
+
400
+ try:
401
+ return self._registered_front_end_infos[config_type]
402
+ except KeyError as err:
403
+ raise KeyError(f"Could not find a registered front end for config `{config_type}`. "
404
+ f"Registered configs: {set(self._registered_front_end_infos.keys())}") from err
405
+
406
+ def get_registered_front_ends(self) -> list[RegisteredInfo[FrontEndBaseConfig]]:
407
+
408
+ return list(self._registered_front_end_infos.values())
409
+
410
+ def register_function(self, registration: RegisteredFunctionInfo):
411
+
412
+ if (registration.config_type in self._registered_functions):
413
+ raise ValueError(f"A function with the same config type `{registration.config_type}` has already been "
414
+ "registered.")
415
+
416
+ self._registered_functions[registration.config_type] = registration
417
+
418
+ self._registration_changed()
419
+
420
+ def get_function(self, config_type: type[FunctionBaseConfig]) -> RegisteredFunctionInfo:
421
+
422
+ try:
423
+ return self._registered_functions[config_type]
424
+ except KeyError as err:
425
+ raise KeyError(f"Could not find a registered function for config `{config_type}`. "
426
+ f"Registered configs: {set(self._registered_functions.keys())}") from err
427
+
428
+ def get_registered_functions(self) -> list[RegisteredInfo[FunctionBaseConfig]]:
429
+
430
+ return list(self._registered_functions.values())
431
+
432
+ def register_llm_provider(self, info: RegisteredLLMProviderInfo):
433
+
434
+ if (info.config_type in self._registered_llm_provider_infos):
435
+ raise ValueError(
436
+ f"An LLM provider with the same config type `{info.config_type}` has already been registered.")
437
+
438
+ self._registered_llm_provider_infos[info.config_type] = info
439
+
440
+ self._registration_changed()
441
+
442
+ def get_llm_provider(self, config_type: type[LLMBaseConfig]) -> RegisteredLLMProviderInfo:
443
+
444
+ try:
445
+ return self._registered_llm_provider_infos[config_type]
446
+ except KeyError as err:
447
+ raise KeyError(f"Could not find a registered LLM provider for config `{config_type}`. "
448
+ f"Registered configs: {set(self._registered_llm_provider_infos.keys())}") from err
449
+
450
+ def get_registered_llm_providers(self) -> list[RegisteredInfo[LLMBaseConfig]]:
451
+
452
+ return list(self._registered_llm_provider_infos.values())
453
+
454
+ def register_llm_client(self, info: RegisteredLLMClientInfo):
455
+
456
+ if (info.config_type in self._llm_client_provider_to_framework
457
+ and info.llm_framework in self._llm_client_provider_to_framework[info.config_type]):
458
+ raise ValueError(f"An LLM client with the same config type `{info.config_type}` "
459
+ f"and LLM framework `{info.llm_framework}` has already been registered.")
460
+
461
+ self._llm_client_provider_to_framework.setdefault(info.config_type, {})[info.llm_framework] = info
462
+ self._llm_client_framework_to_provider.setdefault(info.llm_framework, {})[info.config_type] = info
463
+
464
+ self._registration_changed()
465
+
466
+ def get_llm_client(self, config_type: type[LLMBaseConfig], wrapper_type: str) -> RegisteredLLMClientInfo:
467
+
468
+ try:
469
+ client_info = self._llm_client_provider_to_framework[config_type][wrapper_type]
470
+ except KeyError as err:
471
+ raise KeyError(f"An invalid LLM config and wrapper combination was supplied. Config: `{config_type}`, "
472
+ f"Wrapper: `{wrapper_type}`. The workflow is requesting a {wrapper_type} LLM client but "
473
+ f"there is no registered conversion from that LLM provider to LLM framework: "
474
+ f"{wrapper_type}. "
475
+ f"Please provide an LLM configuration from one of the following providers: "
476
+ f"{set(self._llm_client_provider_to_framework.keys())}") from err
477
+
478
+ return client_info
479
+
480
+ def register_embedder_provider(self, info: RegisteredEmbedderProviderInfo):
481
+
482
+ if (info.config_type in self._registered_embedder_provider_infos):
483
+ raise ValueError(f"An Embedder provider with the same config type `{info.config_type}` has already been "
484
+ "registered.")
485
+
486
+ self._registered_embedder_provider_infos[info.config_type] = info
487
+
488
+ self._registration_changed()
489
+
490
+ def get_embedder_provider(self, config_type: type[EmbedderBaseConfig]) -> RegisteredEmbedderProviderInfo:
491
+
492
+ try:
493
+ return self._registered_embedder_provider_infos[config_type]
494
+ except KeyError as err:
495
+ raise KeyError(f"Could not find a registered Embedder provider for config `{config_type}`. "
496
+ f"Registered configs: {set(self._registered_embedder_provider_infos.keys())}") from err
497
+
498
+ def get_registered_embedder_providers(self) -> list[RegisteredInfo[EmbedderBaseConfig]]:
499
+
500
+ return list(self._registered_embedder_provider_infos.values())
501
+
502
+ def register_embedder_client(self, info: RegisteredEmbedderClientInfo):
503
+
504
+ if (info.config_type in self._embedder_client_provider_to_framework
505
+ and info.llm_framework in self._embedder_client_provider_to_framework[info.config_type]):
506
+ raise ValueError(f"An Embedder client with the same config type `{info.config_type}` has already been "
507
+ "registered.")
508
+
509
+ self._embedder_client_provider_to_framework.setdefault(info.config_type, {})[info.llm_framework] = info
510
+ self._embedder_client_framework_to_provider.setdefault(info.llm_framework, {})[info.config_type] = info
511
+
512
+ self._registration_changed()
513
+
514
+ def get_embedder_client(self, config_type: type[EmbedderBaseConfig],
515
+ wrapper_type: str) -> RegisteredEmbedderClientInfo:
516
+
517
+ try:
518
+ client_info = self._embedder_client_provider_to_framework[config_type][wrapper_type]
519
+ except KeyError as err:
520
+ raise KeyError(
521
+ f"An invalid Embedder config and wrapper combination was supplied. Config: `{config_type}`, "
522
+ "Wrapper: `{wrapper_type}`. The workflow is requesting a {wrapper_type} Embedder client but "
523
+ "there is no registered conversion from that Embedder provider to LLM framework: {wrapper_type}. "
524
+ "Please provide an Embedder configuration from one of the following providers: "
525
+ f"{set(self._embedder_client_provider_to_framework.keys())}") from err
526
+
527
+ return client_info
528
+
529
+ def register_evaluator(self, info: RegisteredEvaluatorInfo):
530
+
531
+ if (info.config_type in self._registered_evaluator_infos):
532
+ raise ValueError(f"An Evaluator with the same config type `{info.config_type}` has already been "
533
+ "registered.")
534
+
535
+ self._registered_evaluator_infos[info.config_type] = info
536
+
537
+ self._registration_changed()
538
+
539
+ def get_evaluator(self, config_type: type[EvaluatorBaseConfig]) -> RegisteredEvaluatorInfo:
540
+
541
+ try:
542
+ return self._registered_evaluator_infos[config_type]
543
+ except KeyError as err:
544
+ raise KeyError(f"Could not find a registered Evaluator for config `{config_type}`. "
545
+ f"Registered configs: {set(self._registered_evaluator_infos.keys())}") from err
546
+
547
+ def get_registered_evaluators(self) -> list[RegisteredInfo[EvaluatorBaseConfig]]:
548
+
549
+ return list(self._registered_evaluator_infos.values())
550
+
551
+ def register_memory(self, info: RegisteredMemoryInfo):
552
+
553
+ if (info.config_type in self._registered_memory_infos):
554
+ raise ValueError(
555
+ f"A Memory client with the same config type `{info.config_type}` has already been registered.")
556
+
557
+ self._registered_memory_infos[info.config_type] = info
558
+
559
+ self._registration_changed()
560
+
561
+ def get_memory(self, config_type: type[MemoryBaseConfig]) -> RegisteredMemoryInfo:
562
+
563
+ try:
564
+ return self._registered_memory_infos[config_type]
565
+ except KeyError as err:
566
+ raise KeyError(f"Could not find a registered Memory client for config `{config_type}`. "
567
+ f"Registered configs: {set(self._registered_memory_infos.keys())}") from err
568
+
569
+ def get_registered_memorys(self) -> list[RegisteredInfo[MemoryBaseConfig]]:
570
+
571
+ return list(self._registered_memory_infos.values())
572
+
573
+ def register_retriever_provider(self, info: RegisteredRetrieverProviderInfo):
574
+
575
+ if (info.config_type in self._registered_retriever_provider_infos):
576
+ raise ValueError(
577
+ f"A Retriever provider with the same config type `{info.config_type}` has already been registered")
578
+
579
+ self._registered_retriever_provider_infos[info.config_type] = info
580
+
581
+ self._registration_changed()
582
+
583
+ def get_retriever_provider(self, config_type: type[RetrieverBaseConfig]) -> RegisteredRetrieverProviderInfo:
584
+
585
+ try:
586
+ return self._registered_retriever_provider_infos[config_type]
587
+ except KeyError as err:
588
+ raise KeyError(f"Could not find a registered Retriever provider for config `{config_type}`. "
589
+ f"Registered configs: {set(self._registered_retriever_provider_infos.keys())}") from err
590
+
591
+ def get_registered_retriever_providers(self) -> list[RegisteredInfo[RetrieverBaseConfig]]:
592
+
593
+ return list(self._registered_retriever_provider_infos.values())
594
+
595
+ def register_retriever_client(self, info: RegisteredRetrieverClientInfo):
596
+
597
+ if (info.config_type in self._retriever_client_provider_to_framework
598
+ and info.llm_framework in self._retriever_client_provider_to_framework[info.config_type]):
599
+ raise ValueError(f"A Retriever client with the same config type `{info.config_type}` "
600
+ " and LLM framework `{info.llm_framework}` has already been registered.")
601
+
602
+ self._retriever_client_provider_to_framework.setdefault(info.config_type, {})[info.llm_framework] = info
603
+ self._retriever_client_framework_to_provider.setdefault(info.llm_framework, {})[info.config_type] = info
604
+
605
+ self._registration_changed()
606
+
607
+ def get_retriever_client(self, config_type: type[RetrieverBaseConfig],
608
+ wrapper_type: str | None) -> RegisteredRetrieverClientInfo:
609
+
610
+ try:
611
+ client_info = self._retriever_client_provider_to_framework[config_type][wrapper_type]
612
+ except KeyError as err:
613
+ raise KeyError(
614
+ f"An invalid Retriever config and wrapper combination was supplied. Config: `{config_type}`, "
615
+ "Wrapper: `{wrapper_type}`. The workflow is requesting a {wrapper_type} Retriever client but "
616
+ "there is no registered conversion from that Retriever provider to LLM framework: {wrapper_type}. "
617
+ "Please provide a Retriever configuration from one of the following providers: "
618
+ f"{set(self._retriever_client_provider_to_framework.keys())}") from err
619
+
620
+ return client_info
621
+
622
+ def register_tool_wrapper(self, registration: RegisteredToolWrapper):
623
+
624
+ if (registration.llm_framework in self._registered_tool_wrappers):
625
+ raise ValueError(f"A tool wrapper for the LLM framework `{registration.llm_framework}` has already been "
626
+ "registered.")
627
+
628
+ self._registered_tool_wrappers[registration.llm_framework] = registration
629
+
630
+ self._registration_changed()
631
+
632
+ def get_tool_wrapper(self, llm_framework: str) -> RegisteredToolWrapper:
633
+
634
+ try:
635
+ return self._registered_tool_wrappers[llm_framework]
636
+ except KeyError as err:
637
+ raise KeyError(f"Could not find a registered tool wrapper for LLM framework `{llm_framework}`. "
638
+ f"Registered LLM frameworks: {set(self._registered_tool_wrappers.keys())}") from err
639
+
640
+ def register_registry_handler(self, info: RegisteredRegistryHandlerInfo):
641
+
642
+ if (info.config_type in self._registered_memory_infos):
643
+ raise ValueError(
644
+ f"A Registry Handler with the same config type `{info.config_type}` has already been registered.")
645
+
646
+ self._registered_registry_handler_infos[info.config_type] = info
647
+ self._registered_channel_map[info.config_type.static_type()] = info
648
+
649
+ self._registration_changed()
650
+
651
+ def get_registry_handler(self, config_type: type[RegistryHandlerBaseConfig]) -> RegisteredRegistryHandlerInfo:
652
+
653
+ try:
654
+ return self._registered_registry_handler_infos[config_type]
655
+ except KeyError as err:
656
+ raise KeyError(f"Could not find a registered Registry Handler for config `{config_type}`. "
657
+ f"Registered configs: {set(self._registered_registry_handler_infos.keys())}") from err
658
+
659
+ def get_registered_registry_handlers(self) -> list[RegisteredInfo[RegistryHandlerBaseConfig]]:
660
+
661
+ return list(self._registered_registry_handler_infos.values())
662
+
663
+ def register_package(self, package_name: str, package_version: str | None = None):
664
+
665
+ discovery_metadata = DiscoveryMetadata.from_package_name(package_name=package_name,
666
+ package_version=package_version)
667
+ package = RegisteredPackage(discovery_metadata=discovery_metadata, package_name=package_name)
668
+ self._registered_packages[package.package_name] = package
669
+
670
+ self._registration_changed()
671
+
672
+ def get_infos_by_type(self, component_type: AIQComponentEnum) -> dict: # pylint: disable=R0911
673
+
674
+ if component_type == AIQComponentEnum.FRONT_END:
675
+ return self._registered_front_end_infos
676
+
677
+ if component_type == AIQComponentEnum.FUNCTION:
678
+ return self._registered_functions
679
+
680
+ if component_type == AIQComponentEnum.TOOL_WRAPPER:
681
+ return self._registered_tool_wrappers
682
+
683
+ if component_type == AIQComponentEnum.LLM_PROVIDER:
684
+ return self._registered_llm_provider_infos
685
+
686
+ if component_type == AIQComponentEnum.LLM_CLIENT:
687
+ leaf_llm_client_infos = {}
688
+ for framework in self._llm_client_provider_to_framework.values():
689
+ for info in framework.values():
690
+ leaf_llm_client_infos[info.discovery_metadata.component_name] = info
691
+ return leaf_llm_client_infos
692
+
693
+ if component_type == AIQComponentEnum.EMBEDDER_PROVIDER:
694
+ return self._registered_embedder_provider_infos
695
+
696
+ if component_type == AIQComponentEnum.EMBEDDER_CLIENT:
697
+ leaf_embedder_client_infos = {}
698
+ for framework in self._embedder_client_provider_to_framework.values():
699
+ for info in framework.values():
700
+ leaf_embedder_client_infos[info.discovery_metadata.component_name] = info
701
+ return leaf_embedder_client_infos
702
+
703
+ if component_type == AIQComponentEnum.RETRIEVER_PROVIDER:
704
+ return self._registered_retriever_provider_infos
705
+
706
+ if component_type == AIQComponentEnum.RETRIEVER_CLIENT:
707
+ leaf_retriever_client_infos = {}
708
+ for framework in self._retriever_client_provider_to_framework.values():
709
+ for info in framework.values():
710
+ leaf_retriever_client_infos[info.discovery_metadata.component_name] = info
711
+ return leaf_retriever_client_infos
712
+
713
+ if component_type == AIQComponentEnum.EVALUATOR:
714
+ return self._registered_evaluator_infos
715
+
716
+ if component_type == AIQComponentEnum.MEMORY:
717
+ return self._registered_memory_infos
718
+
719
+ if component_type == AIQComponentEnum.REGISTRY_HANDLER:
720
+ return self._registered_registry_handler_infos
721
+
722
+ if component_type == AIQComponentEnum.LOGGING:
723
+ return self._registered_logging_methods
724
+
725
+ if component_type == AIQComponentEnum.TRACING:
726
+ return self._registered_telemetry_exporters
727
+
728
+ if component_type == AIQComponentEnum.PACKAGE:
729
+ return self._registered_packages
730
+
731
+ raise ValueError(f"Supplied an unsupported component type {component_type}")
732
+
733
+ def get_registered_types_by_component_type( # pylint: disable=R0911
734
+ self, component_type: AIQComponentEnum) -> list[str]:
735
+
736
+ if component_type == AIQComponentEnum.FUNCTION:
737
+ return [i.static_type() for i in self._registered_functions]
738
+
739
+ if component_type == AIQComponentEnum.TOOL_WRAPPER:
740
+ return list(self._registered_tool_wrappers)
741
+
742
+ if component_type == AIQComponentEnum.LLM_PROVIDER:
743
+ return [i.static_type() for i in self._registered_llm_provider_infos]
744
+
745
+ if component_type == AIQComponentEnum.LLM_CLIENT:
746
+ leaf_client_provider_framework_types = []
747
+ for framework in self._llm_client_provider_to_framework.values():
748
+ for info in framework.values():
749
+ leaf_client_provider_framework_types.append([info.discovery_metadata.component_name])
750
+ return leaf_client_provider_framework_types
751
+
752
+ if component_type == AIQComponentEnum.EMBEDDER_PROVIDER:
753
+ return [i.static_type() for i in self._registered_embedder_provider_infos]
754
+
755
+ if component_type == AIQComponentEnum.EMBEDDER_CLIENT:
756
+ leaf_embedder_provider_framework_types = []
757
+ for framework in self._embedder_client_provider_to_framework.values():
758
+ for info in framework.values():
759
+ leaf_embedder_provider_framework_types.append([info.discovery_metadata.component_name])
760
+ return leaf_embedder_provider_framework_types
761
+
762
+ if component_type == AIQComponentEnum.EVALUATOR:
763
+ return [i.static_type() for i in self._registered_evaluator_infos]
764
+
765
+ if component_type == AIQComponentEnum.MEMORY:
766
+ return [i.static_type() for i in self._registered_memory_infos]
767
+
768
+ if component_type == AIQComponentEnum.REGISTRY_HANDLER:
769
+ return [i.static_type() for i in self._registered_registry_handler_infos]
770
+
771
+ if component_type == AIQComponentEnum.LOGGING:
772
+ return [i.static_type() for i in self._registered_logging_methods]
773
+
774
+ if component_type == AIQComponentEnum.TRACING:
775
+ return [i.static_type() for i in self._registered_telemetry_exporters]
776
+
777
+ if component_type == AIQComponentEnum.PACKAGE:
778
+ return list(self._registered_packages)
779
+
780
+ raise ValueError(f"Supplied an unsupported component type {component_type}")
781
+
782
+ def get_registered_channel_info_by_channel_type(self, channel_type: str) -> RegisteredRegistryHandlerInfo:
783
+ return self._registered_channel_map[channel_type]
784
+
785
+ def _do_compute_annotation(self, cls: type[TypedBaseModelT], registrations: list[RegisteredInfo[TypedBaseModelT]]):
786
+
787
+ while (len(registrations) < 2):
788
+ registrations.append(RegisteredInfo[TypedBaseModelT](full_type=f"_ignore/{len(registrations)}",
789
+ config_type=cls))
790
+
791
+ short_names: dict[str, int] = {}
792
+ type_list: list[tuple[str, type[TypedBaseModelT]]] = []
793
+
794
+ # For all keys in the list, split the key by / and increment the count of the last element
795
+ for key in registrations:
796
+ short_names[key.local_name] = short_names.get(key.local_name, 0) + 1
797
+
798
+ type_list.append((key.full_type, key.config_type))
799
+
800
+ # Now loop again and if the short name is unique, then create two entries, for the short and full name
801
+ for key in registrations:
802
+
803
+ if (short_names[key.local_name] == 1):
804
+ type_list.append((key.local_name, key.config_type))
805
+
806
+ # pylint: disable=consider-alternative-union-syntax
807
+ return typing.Union[tuple(typing.Annotated[x_type, Tag(x_id)] for x_id, x_type in type_list)]
808
+
809
+ def compute_annotation(self, cls: type[TypedBaseModelT]):
810
+
811
+ if issubclass(cls, EmbedderBaseConfig):
812
+ return self._do_compute_annotation(cls, self.get_registered_embedder_providers())
813
+
814
+ if issubclass(cls, EvaluatorBaseConfig):
815
+ return self._do_compute_annotation(cls, self.get_registered_evaluators())
816
+
817
+ if issubclass(cls, FrontEndBaseConfig):
818
+ return self._do_compute_annotation(cls, self.get_registered_front_ends())
819
+
820
+ if issubclass(cls, FunctionBaseConfig):
821
+ return self._do_compute_annotation(cls, self.get_registered_functions())
822
+
823
+ if issubclass(cls, LLMBaseConfig):
824
+ return self._do_compute_annotation(cls, self.get_registered_llm_providers())
825
+
826
+ if issubclass(cls, MemoryBaseConfig):
827
+ return self._do_compute_annotation(cls, self.get_registered_memorys())
828
+
829
+ if issubclass(cls, RegistryHandlerBaseConfig):
830
+ return self._do_compute_annotation(cls, self.get_registered_registry_handlers())
831
+
832
+ if issubclass(cls, RetrieverBaseConfig):
833
+ return self._do_compute_annotation(cls, self.get_registered_retriever_providers())
834
+
835
+ if issubclass(cls, TelemetryExporterBaseConfig):
836
+ return self._do_compute_annotation(cls, self.get_registered_telemetry_exporters())
837
+
838
+ if issubclass(cls, LoggingBaseConfig):
839
+ return self._do_compute_annotation(cls, self.get_registered_logging_method())
840
+
841
+ raise ValueError(f"Supplied an unsupported component type {cls}")
842
+
843
+
844
+ class GlobalTypeRegistry:
845
+
846
+ _global_registry: TypeRegistry = TypeRegistry()
847
+
848
+ @staticmethod
849
+ def get() -> TypeRegistry:
850
+ return GlobalTypeRegistry._global_registry
851
+
852
+ @staticmethod
853
+ @contextmanager
854
+ def push():
855
+
856
+ saved = GlobalTypeRegistry._global_registry
857
+ registry = deepcopy(saved)
858
+
859
+ try:
860
+ GlobalTypeRegistry._global_registry = registry
861
+
862
+ yield registry
863
+ finally:
864
+ GlobalTypeRegistry._global_registry = saved
865
+ GlobalTypeRegistry._global_registry._registration_changed()
866
+
867
+
868
+ # Finally, update the Config object each time the registry changes
869
+ GlobalTypeRegistry.get().add_registration_changed_hook(lambda: AIQConfig.rebuild_annotations())