aiqtoolkit 1.1.0rc6__py3-none-any.whl → 1.2.0__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 (319) hide show
  1. aiqtoolkit-1.2.0.dist-info/METADATA +29 -0
  2. aiqtoolkit-1.2.0.dist-info/RECORD +4 -0
  3. {aiqtoolkit-1.1.0rc6.dist-info → aiqtoolkit-1.2.0.dist-info}/WHEEL +1 -1
  4. aiqtoolkit-1.2.0.dist-info/top_level.txt +1 -0
  5. aiq/agent/__init__.py +0 -0
  6. aiq/agent/base.py +0 -76
  7. aiq/agent/dual_node.py +0 -67
  8. aiq/agent/react_agent/__init__.py +0 -0
  9. aiq/agent/react_agent/agent.py +0 -322
  10. aiq/agent/react_agent/output_parser.py +0 -104
  11. aiq/agent/react_agent/prompt.py +0 -46
  12. aiq/agent/react_agent/register.py +0 -148
  13. aiq/agent/reasoning_agent/__init__.py +0 -0
  14. aiq/agent/reasoning_agent/reasoning_agent.py +0 -224
  15. aiq/agent/register.py +0 -23
  16. aiq/agent/rewoo_agent/__init__.py +0 -0
  17. aiq/agent/rewoo_agent/agent.py +0 -410
  18. aiq/agent/rewoo_agent/prompt.py +0 -108
  19. aiq/agent/rewoo_agent/register.py +0 -158
  20. aiq/agent/tool_calling_agent/__init__.py +0 -0
  21. aiq/agent/tool_calling_agent/agent.py +0 -123
  22. aiq/agent/tool_calling_agent/register.py +0 -105
  23. aiq/builder/__init__.py +0 -0
  24. aiq/builder/builder.py +0 -223
  25. aiq/builder/component_utils.py +0 -303
  26. aiq/builder/context.py +0 -227
  27. aiq/builder/embedder.py +0 -24
  28. aiq/builder/eval_builder.py +0 -120
  29. aiq/builder/evaluator.py +0 -29
  30. aiq/builder/framework_enum.py +0 -24
  31. aiq/builder/front_end.py +0 -73
  32. aiq/builder/function.py +0 -297
  33. aiq/builder/function_base.py +0 -376
  34. aiq/builder/function_info.py +0 -627
  35. aiq/builder/intermediate_step_manager.py +0 -176
  36. aiq/builder/llm.py +0 -25
  37. aiq/builder/retriever.py +0 -25
  38. aiq/builder/user_interaction_manager.py +0 -71
  39. aiq/builder/workflow.py +0 -143
  40. aiq/builder/workflow_builder.py +0 -757
  41. aiq/cli/__init__.py +0 -14
  42. aiq/cli/cli_utils/__init__.py +0 -0
  43. aiq/cli/cli_utils/config_override.py +0 -231
  44. aiq/cli/cli_utils/validation.py +0 -37
  45. aiq/cli/commands/__init__.py +0 -0
  46. aiq/cli/commands/configure/__init__.py +0 -0
  47. aiq/cli/commands/configure/channel/__init__.py +0 -0
  48. aiq/cli/commands/configure/channel/add.py +0 -28
  49. aiq/cli/commands/configure/channel/channel.py +0 -36
  50. aiq/cli/commands/configure/channel/remove.py +0 -30
  51. aiq/cli/commands/configure/channel/update.py +0 -30
  52. aiq/cli/commands/configure/configure.py +0 -33
  53. aiq/cli/commands/evaluate.py +0 -139
  54. aiq/cli/commands/info/__init__.py +0 -14
  55. aiq/cli/commands/info/info.py +0 -39
  56. aiq/cli/commands/info/list_channels.py +0 -32
  57. aiq/cli/commands/info/list_components.py +0 -129
  58. aiq/cli/commands/info/list_mcp.py +0 -126
  59. aiq/cli/commands/registry/__init__.py +0 -14
  60. aiq/cli/commands/registry/publish.py +0 -88
  61. aiq/cli/commands/registry/pull.py +0 -118
  62. aiq/cli/commands/registry/registry.py +0 -38
  63. aiq/cli/commands/registry/remove.py +0 -108
  64. aiq/cli/commands/registry/search.py +0 -155
  65. aiq/cli/commands/start.py +0 -250
  66. aiq/cli/commands/uninstall.py +0 -83
  67. aiq/cli/commands/validate.py +0 -47
  68. aiq/cli/commands/workflow/__init__.py +0 -14
  69. aiq/cli/commands/workflow/templates/__init__.py.j2 +0 -0
  70. aiq/cli/commands/workflow/templates/config.yml.j2 +0 -16
  71. aiq/cli/commands/workflow/templates/pyproject.toml.j2 +0 -22
  72. aiq/cli/commands/workflow/templates/register.py.j2 +0 -5
  73. aiq/cli/commands/workflow/templates/workflow.py.j2 +0 -36
  74. aiq/cli/commands/workflow/workflow.py +0 -37
  75. aiq/cli/commands/workflow/workflow_commands.py +0 -313
  76. aiq/cli/entrypoint.py +0 -133
  77. aiq/cli/main.py +0 -44
  78. aiq/cli/register_workflow.py +0 -408
  79. aiq/cli/type_registry.py +0 -879
  80. aiq/data_models/__init__.py +0 -14
  81. aiq/data_models/api_server.py +0 -588
  82. aiq/data_models/common.py +0 -143
  83. aiq/data_models/component.py +0 -46
  84. aiq/data_models/component_ref.py +0 -135
  85. aiq/data_models/config.py +0 -349
  86. aiq/data_models/dataset_handler.py +0 -122
  87. aiq/data_models/discovery_metadata.py +0 -286
  88. aiq/data_models/embedder.py +0 -26
  89. aiq/data_models/evaluate.py +0 -104
  90. aiq/data_models/evaluator.py +0 -26
  91. aiq/data_models/front_end.py +0 -26
  92. aiq/data_models/function.py +0 -30
  93. aiq/data_models/function_dependencies.py +0 -64
  94. aiq/data_models/interactive.py +0 -237
  95. aiq/data_models/intermediate_step.py +0 -269
  96. aiq/data_models/invocation_node.py +0 -38
  97. aiq/data_models/llm.py +0 -26
  98. aiq/data_models/logging.py +0 -26
  99. aiq/data_models/memory.py +0 -26
  100. aiq/data_models/profiler.py +0 -53
  101. aiq/data_models/registry_handler.py +0 -26
  102. aiq/data_models/retriever.py +0 -30
  103. aiq/data_models/step_adaptor.py +0 -64
  104. aiq/data_models/streaming.py +0 -33
  105. aiq/data_models/swe_bench_model.py +0 -54
  106. aiq/data_models/telemetry_exporter.py +0 -26
  107. aiq/embedder/__init__.py +0 -0
  108. aiq/embedder/langchain_client.py +0 -41
  109. aiq/embedder/nim_embedder.py +0 -58
  110. aiq/embedder/openai_embedder.py +0 -42
  111. aiq/embedder/register.py +0 -24
  112. aiq/eval/__init__.py +0 -14
  113. aiq/eval/config.py +0 -42
  114. aiq/eval/dataset_handler/__init__.py +0 -0
  115. aiq/eval/dataset_handler/dataset_downloader.py +0 -106
  116. aiq/eval/dataset_handler/dataset_filter.py +0 -52
  117. aiq/eval/dataset_handler/dataset_handler.py +0 -169
  118. aiq/eval/evaluate.py +0 -325
  119. aiq/eval/evaluator/__init__.py +0 -14
  120. aiq/eval/evaluator/evaluator_model.py +0 -44
  121. aiq/eval/intermediate_step_adapter.py +0 -93
  122. aiq/eval/rag_evaluator/__init__.py +0 -0
  123. aiq/eval/rag_evaluator/evaluate.py +0 -138
  124. aiq/eval/rag_evaluator/register.py +0 -138
  125. aiq/eval/register.py +0 -23
  126. aiq/eval/remote_workflow.py +0 -128
  127. aiq/eval/runtime_event_subscriber.py +0 -52
  128. aiq/eval/swe_bench_evaluator/__init__.py +0 -0
  129. aiq/eval/swe_bench_evaluator/evaluate.py +0 -215
  130. aiq/eval/swe_bench_evaluator/register.py +0 -36
  131. aiq/eval/trajectory_evaluator/__init__.py +0 -0
  132. aiq/eval/trajectory_evaluator/evaluate.py +0 -118
  133. aiq/eval/trajectory_evaluator/register.py +0 -40
  134. aiq/eval/tunable_rag_evaluator/__init__.py +0 -0
  135. aiq/eval/tunable_rag_evaluator/evaluate.py +0 -263
  136. aiq/eval/tunable_rag_evaluator/register.py +0 -50
  137. aiq/eval/utils/__init__.py +0 -0
  138. aiq/eval/utils/output_uploader.py +0 -131
  139. aiq/eval/utils/tqdm_position_registry.py +0 -40
  140. aiq/front_ends/__init__.py +0 -14
  141. aiq/front_ends/console/__init__.py +0 -14
  142. aiq/front_ends/console/console_front_end_config.py +0 -32
  143. aiq/front_ends/console/console_front_end_plugin.py +0 -107
  144. aiq/front_ends/console/register.py +0 -25
  145. aiq/front_ends/cron/__init__.py +0 -14
  146. aiq/front_ends/fastapi/__init__.py +0 -14
  147. aiq/front_ends/fastapi/fastapi_front_end_config.py +0 -150
  148. aiq/front_ends/fastapi/fastapi_front_end_plugin.py +0 -103
  149. aiq/front_ends/fastapi/fastapi_front_end_plugin_worker.py +0 -607
  150. aiq/front_ends/fastapi/intermediate_steps_subscriber.py +0 -80
  151. aiq/front_ends/fastapi/job_store.py +0 -161
  152. aiq/front_ends/fastapi/main.py +0 -70
  153. aiq/front_ends/fastapi/message_handler.py +0 -279
  154. aiq/front_ends/fastapi/message_validator.py +0 -345
  155. aiq/front_ends/fastapi/register.py +0 -25
  156. aiq/front_ends/fastapi/response_helpers.py +0 -195
  157. aiq/front_ends/fastapi/step_adaptor.py +0 -320
  158. aiq/front_ends/fastapi/websocket.py +0 -148
  159. aiq/front_ends/mcp/__init__.py +0 -14
  160. aiq/front_ends/mcp/mcp_front_end_config.py +0 -32
  161. aiq/front_ends/mcp/mcp_front_end_plugin.py +0 -93
  162. aiq/front_ends/mcp/register.py +0 -27
  163. aiq/front_ends/mcp/tool_converter.py +0 -242
  164. aiq/front_ends/register.py +0 -22
  165. aiq/front_ends/simple_base/__init__.py +0 -14
  166. aiq/front_ends/simple_base/simple_front_end_plugin_base.py +0 -52
  167. aiq/llm/__init__.py +0 -0
  168. aiq/llm/nim_llm.py +0 -45
  169. aiq/llm/openai_llm.py +0 -45
  170. aiq/llm/register.py +0 -22
  171. aiq/llm/utils/__init__.py +0 -14
  172. aiq/llm/utils/env_config_value.py +0 -94
  173. aiq/llm/utils/error.py +0 -17
  174. aiq/memory/__init__.py +0 -20
  175. aiq/memory/interfaces.py +0 -183
  176. aiq/memory/models.py +0 -112
  177. aiq/meta/module_to_distro.json +0 -3
  178. aiq/meta/pypi.md +0 -58
  179. aiq/observability/__init__.py +0 -0
  180. aiq/observability/async_otel_listener.py +0 -429
  181. aiq/observability/register.py +0 -99
  182. aiq/plugins/.namespace +0 -1
  183. aiq/profiler/__init__.py +0 -0
  184. aiq/profiler/callbacks/__init__.py +0 -0
  185. aiq/profiler/callbacks/agno_callback_handler.py +0 -295
  186. aiq/profiler/callbacks/base_callback_class.py +0 -20
  187. aiq/profiler/callbacks/langchain_callback_handler.py +0 -278
  188. aiq/profiler/callbacks/llama_index_callback_handler.py +0 -205
  189. aiq/profiler/callbacks/semantic_kernel_callback_handler.py +0 -238
  190. aiq/profiler/callbacks/token_usage_base_model.py +0 -27
  191. aiq/profiler/data_frame_row.py +0 -51
  192. aiq/profiler/decorators/__init__.py +0 -0
  193. aiq/profiler/decorators/framework_wrapper.py +0 -131
  194. aiq/profiler/decorators/function_tracking.py +0 -254
  195. aiq/profiler/forecasting/__init__.py +0 -0
  196. aiq/profiler/forecasting/config.py +0 -18
  197. aiq/profiler/forecasting/model_trainer.py +0 -75
  198. aiq/profiler/forecasting/models/__init__.py +0 -22
  199. aiq/profiler/forecasting/models/forecasting_base_model.py +0 -40
  200. aiq/profiler/forecasting/models/linear_model.py +0 -196
  201. aiq/profiler/forecasting/models/random_forest_regressor.py +0 -268
  202. aiq/profiler/inference_metrics_model.py +0 -25
  203. aiq/profiler/inference_optimization/__init__.py +0 -0
  204. aiq/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
  205. aiq/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +0 -452
  206. aiq/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +0 -258
  207. aiq/profiler/inference_optimization/data_models.py +0 -386
  208. aiq/profiler/inference_optimization/experimental/__init__.py +0 -0
  209. aiq/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +0 -468
  210. aiq/profiler/inference_optimization/experimental/prefix_span_analysis.py +0 -405
  211. aiq/profiler/inference_optimization/llm_metrics.py +0 -212
  212. aiq/profiler/inference_optimization/prompt_caching.py +0 -163
  213. aiq/profiler/inference_optimization/token_uniqueness.py +0 -107
  214. aiq/profiler/inference_optimization/workflow_runtimes.py +0 -72
  215. aiq/profiler/intermediate_property_adapter.py +0 -102
  216. aiq/profiler/profile_runner.py +0 -433
  217. aiq/profiler/utils.py +0 -184
  218. aiq/registry_handlers/__init__.py +0 -0
  219. aiq/registry_handlers/local/__init__.py +0 -0
  220. aiq/registry_handlers/local/local_handler.py +0 -176
  221. aiq/registry_handlers/local/register_local.py +0 -37
  222. aiq/registry_handlers/metadata_factory.py +0 -60
  223. aiq/registry_handlers/package_utils.py +0 -198
  224. aiq/registry_handlers/pypi/__init__.py +0 -0
  225. aiq/registry_handlers/pypi/pypi_handler.py +0 -251
  226. aiq/registry_handlers/pypi/register_pypi.py +0 -40
  227. aiq/registry_handlers/register.py +0 -21
  228. aiq/registry_handlers/registry_handler_base.py +0 -157
  229. aiq/registry_handlers/rest/__init__.py +0 -0
  230. aiq/registry_handlers/rest/register_rest.py +0 -56
  231. aiq/registry_handlers/rest/rest_handler.py +0 -237
  232. aiq/registry_handlers/schemas/__init__.py +0 -0
  233. aiq/registry_handlers/schemas/headers.py +0 -42
  234. aiq/registry_handlers/schemas/package.py +0 -68
  235. aiq/registry_handlers/schemas/publish.py +0 -63
  236. aiq/registry_handlers/schemas/pull.py +0 -82
  237. aiq/registry_handlers/schemas/remove.py +0 -36
  238. aiq/registry_handlers/schemas/search.py +0 -91
  239. aiq/registry_handlers/schemas/status.py +0 -47
  240. aiq/retriever/__init__.py +0 -0
  241. aiq/retriever/interface.py +0 -37
  242. aiq/retriever/milvus/__init__.py +0 -14
  243. aiq/retriever/milvus/register.py +0 -81
  244. aiq/retriever/milvus/retriever.py +0 -228
  245. aiq/retriever/models.py +0 -74
  246. aiq/retriever/nemo_retriever/__init__.py +0 -14
  247. aiq/retriever/nemo_retriever/register.py +0 -60
  248. aiq/retriever/nemo_retriever/retriever.py +0 -190
  249. aiq/retriever/register.py +0 -22
  250. aiq/runtime/__init__.py +0 -14
  251. aiq/runtime/loader.py +0 -188
  252. aiq/runtime/runner.py +0 -176
  253. aiq/runtime/session.py +0 -140
  254. aiq/runtime/user_metadata.py +0 -131
  255. aiq/settings/__init__.py +0 -0
  256. aiq/settings/global_settings.py +0 -318
  257. aiq/test/.namespace +0 -1
  258. aiq/tool/__init__.py +0 -0
  259. aiq/tool/code_execution/__init__.py +0 -0
  260. aiq/tool/code_execution/code_sandbox.py +0 -188
  261. aiq/tool/code_execution/local_sandbox/Dockerfile.sandbox +0 -60
  262. aiq/tool/code_execution/local_sandbox/__init__.py +0 -13
  263. aiq/tool/code_execution/local_sandbox/local_sandbox_server.py +0 -83
  264. aiq/tool/code_execution/local_sandbox/sandbox.requirements.txt +0 -4
  265. aiq/tool/code_execution/local_sandbox/start_local_sandbox.sh +0 -25
  266. aiq/tool/code_execution/register.py +0 -70
  267. aiq/tool/code_execution/utils.py +0 -100
  268. aiq/tool/datetime_tools.py +0 -42
  269. aiq/tool/document_search.py +0 -141
  270. aiq/tool/github_tools/__init__.py +0 -0
  271. aiq/tool/github_tools/create_github_commit.py +0 -133
  272. aiq/tool/github_tools/create_github_issue.py +0 -87
  273. aiq/tool/github_tools/create_github_pr.py +0 -106
  274. aiq/tool/github_tools/get_github_file.py +0 -106
  275. aiq/tool/github_tools/get_github_issue.py +0 -166
  276. aiq/tool/github_tools/get_github_pr.py +0 -256
  277. aiq/tool/github_tools/update_github_issue.py +0 -100
  278. aiq/tool/mcp/__init__.py +0 -14
  279. aiq/tool/mcp/mcp_client.py +0 -220
  280. aiq/tool/mcp/mcp_tool.py +0 -95
  281. aiq/tool/memory_tools/__init__.py +0 -0
  282. aiq/tool/memory_tools/add_memory_tool.py +0 -79
  283. aiq/tool/memory_tools/delete_memory_tool.py +0 -67
  284. aiq/tool/memory_tools/get_memory_tool.py +0 -72
  285. aiq/tool/nvidia_rag.py +0 -95
  286. aiq/tool/register.py +0 -37
  287. aiq/tool/retriever.py +0 -89
  288. aiq/tool/server_tools.py +0 -63
  289. aiq/utils/__init__.py +0 -0
  290. aiq/utils/data_models/__init__.py +0 -0
  291. aiq/utils/data_models/schema_validator.py +0 -58
  292. aiq/utils/debugging_utils.py +0 -43
  293. aiq/utils/exception_handlers/__init__.py +0 -0
  294. aiq/utils/exception_handlers/schemas.py +0 -114
  295. aiq/utils/io/__init__.py +0 -0
  296. aiq/utils/io/yaml_tools.py +0 -119
  297. aiq/utils/metadata_utils.py +0 -74
  298. aiq/utils/optional_imports.py +0 -142
  299. aiq/utils/producer_consumer_queue.py +0 -178
  300. aiq/utils/reactive/__init__.py +0 -0
  301. aiq/utils/reactive/base/__init__.py +0 -0
  302. aiq/utils/reactive/base/observable_base.py +0 -65
  303. aiq/utils/reactive/base/observer_base.py +0 -55
  304. aiq/utils/reactive/base/subject_base.py +0 -79
  305. aiq/utils/reactive/observable.py +0 -59
  306. aiq/utils/reactive/observer.py +0 -76
  307. aiq/utils/reactive/subject.py +0 -131
  308. aiq/utils/reactive/subscription.py +0 -49
  309. aiq/utils/settings/__init__.py +0 -0
  310. aiq/utils/settings/global_settings.py +0 -197
  311. aiq/utils/type_converter.py +0 -232
  312. aiq/utils/type_utils.py +0 -397
  313. aiq/utils/url_utils.py +0 -27
  314. aiqtoolkit-1.1.0rc6.dist-info/METADATA +0 -331
  315. aiqtoolkit-1.1.0rc6.dist-info/RECORD +0 -316
  316. aiqtoolkit-1.1.0rc6.dist-info/entry_points.txt +0 -17
  317. aiqtoolkit-1.1.0rc6.dist-info/licenses/LICENSE-3rd-party.txt +0 -3686
  318. aiqtoolkit-1.1.0rc6.dist-info/licenses/LICENSE.md +0 -201
  319. aiqtoolkit-1.1.0rc6.dist-info/top_level.txt +0 -1
@@ -1,320 +0,0 @@
1
- # SPDX-FileCopyrightText: Copyright (c) 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 html
17
- import logging
18
- from functools import reduce
19
- from textwrap import dedent
20
-
21
- from aiq.data_models.api_server import AIQResponseIntermediateStep
22
- from aiq.data_models.api_server import AIQResponseSerializable
23
- from aiq.data_models.intermediate_step import IntermediateStep
24
- from aiq.data_models.intermediate_step import IntermediateStepCategory
25
- from aiq.data_models.intermediate_step import IntermediateStepPayload
26
- from aiq.data_models.intermediate_step import IntermediateStepType
27
- from aiq.data_models.invocation_node import InvocationNode
28
- from aiq.data_models.step_adaptor import StepAdaptorConfig
29
- from aiq.data_models.step_adaptor import StepAdaptorMode
30
- from aiq.utils.type_utils import is_valid_json
31
-
32
- logger = logging.getLogger(__name__)
33
-
34
-
35
- class StepAdaptor:
36
-
37
- def __init__(self, config: StepAdaptorConfig):
38
-
39
- self._history: list[IntermediateStep] = []
40
- self.config = config
41
-
42
- def _step_matches_filter(self, step: IntermediateStep, config: StepAdaptorConfig) -> bool:
43
- """
44
- Returns True if this intermediate step should be included (based on the config.mode).
45
- """
46
-
47
- if config.mode == StepAdaptorMode.OFF:
48
- return False
49
-
50
- if config.mode == StepAdaptorMode.DEFAULT:
51
- # default existing behavior: show LLM events + TOOL_END + FUNCTION events
52
- if step.event_category == IntermediateStepCategory.LLM:
53
- return True
54
- if step.event_category == IntermediateStepCategory.TOOL:
55
- return True
56
- if step.event_category == IntermediateStepCategory.FUNCTION:
57
- return True
58
- return False
59
-
60
- if config.mode == StepAdaptorMode.CUSTOM:
61
- # pass only what the user explicitly listed
62
- return step.event_type in config.custom_event_types
63
-
64
- return False
65
-
66
- def _handle_llm(self, step: IntermediateStepPayload, ancestry: InvocationNode) -> AIQResponseSerializable | None:
67
- input_str: str | None = None
68
- output_str: str | None = None
69
-
70
- # Find the start in the history with matching run_id
71
- start_step = next(
72
- (x for x in self._history if x.event_type == IntermediateStepType.LLM_START and x.UUID == step.UUID), None)
73
-
74
- if not start_step:
75
- # If we don't have a start step, we can't do anything
76
- return None
77
-
78
- input_str = str(start_step.data.input)
79
-
80
- if step.event_type == IntermediateStepType.LLM_NEW_TOKEN:
81
-
82
- # Find all of the previous LLM chunks and concatenate them
83
- output_str = reduce(
84
- lambda x, y: x + y,
85
- (str(x.data.chunk)
86
- for x in self._history if x.event_type == IntermediateStepType.LLM_NEW_TOKEN and x.UUID == step.UUID),
87
- "")
88
-
89
- elif step.event_type == IntermediateStepType.LLM_END:
90
- output_str = str(step.data.output)
91
-
92
- if not input_str and not output_str:
93
- return None
94
-
95
- escaped_input = html.escape(input_str, quote=False)
96
-
97
- # Dont use f-strings here because the payload is markdown and screws up the dedent
98
- payload = dedent("""
99
- **Input:**
100
- ```python
101
- {input_value}
102
- ```
103
- """).strip("\n").format(input_value=escaped_input)
104
-
105
- if (output_str):
106
- escaped_output = html.escape(output_str, quote=False) if output_str else ""
107
-
108
- # Dont use f-strings here because the payload is markdown and screws up the dedent
109
- payload = dedent("""
110
- {payload}
111
-
112
- **Output:**
113
- {output_value}
114
- """).strip("\n").format(payload=payload, output_value=escaped_output)
115
-
116
- event = AIQResponseIntermediateStep(id=step.UUID,
117
- name=step.name or "",
118
- payload=payload,
119
- parent_id=ancestry.function_id)
120
-
121
- return event
122
-
123
- def _handle_tool(self, step: IntermediateStepPayload, ancestry: InvocationNode) -> AIQResponseSerializable | None:
124
- """
125
- Handles both TOOL_START and TOOL_END events
126
- """
127
- input_str: str | None = None
128
- output_str: str | None = None
129
-
130
- # Find the start in the history with matching run_id
131
- start_step = next(
132
- (x for x in self._history if x.event_type == IntermediateStepType.TOOL_START and x.UUID == step.UUID), None)
133
-
134
- if not start_step:
135
- # If we don't have a start step, we can't do anything
136
- return None
137
-
138
- input_str = str(start_step.data.input)
139
-
140
- if step.event_type == IntermediateStepType.TOOL_END:
141
- output_str = str(step.data.output)
142
-
143
- if not input_str and not output_str:
144
- return None
145
-
146
- escaped_input = html.escape(input_str, quote=False)
147
- format_input_type = "json" if is_valid_json(escaped_input) else "python"
148
-
149
- # Dont use f-strings here because the payload is markdown and screws up the dedent
150
- payload = dedent("""
151
- **Input:**
152
- ```{format_input_type}
153
- {input_value}
154
- ```
155
- """).strip("\n").format(input_value=escaped_input, format_input_type=format_input_type)
156
-
157
- if output_str:
158
- escaped_output = html.escape(output_str, quote=False)
159
- format_output_type = "json" if is_valid_json(escaped_output) else "python"
160
-
161
- # Dont use f-strings here because the payload is markdown and screws up the dedent
162
- payload = dedent("""
163
- {payload}
164
-
165
- **Output:**
166
- ```{format_output_type}
167
- {output_value}
168
- ```
169
- """).strip("\n").format(payload=payload, output_value=escaped_output, format_output_type=format_output_type)
170
-
171
- event = AIQResponseIntermediateStep(id=step.UUID,
172
- name=f"Tool: {step.name}",
173
- payload=payload,
174
- parent_id=ancestry.function_id)
175
-
176
- return event
177
-
178
- def _handle_function(self, step: IntermediateStepPayload,
179
- ancestry: InvocationNode) -> AIQResponseSerializable | None:
180
- """
181
- Handles the FUNCTION_START and FUNCTION_END events
182
- """
183
- input_str: str | None = None
184
- output_str: str | None = None
185
-
186
- if step.event_type == IntermediateStepType.FUNCTION_START:
187
- # For function start events, display input data
188
- if step.data and hasattr(step.data, 'input'):
189
- input_str = str(step.data.input)
190
- elif step.data:
191
- input_str = str(step.data)
192
-
193
- if not input_str:
194
- return None
195
-
196
- escaped_input = html.escape(input_str, quote=False)
197
- format_input_type = "json" if is_valid_json(escaped_input) else "python"
198
-
199
- # Create payload for function start
200
- payload_str = dedent("""
201
- **Function Input:**
202
- ```{format_input_type}
203
- {input_value}
204
- ```
205
- """).strip("\n").format(input_value=escaped_input, format_input_type=format_input_type)
206
-
207
- event = AIQResponseIntermediateStep(id=step.UUID,
208
- name=f"Function Start: {step.name}",
209
- payload=payload_str,
210
- parent_id=ancestry.parent_id)
211
- return event
212
-
213
- if step.event_type == IntermediateStepType.FUNCTION_END:
214
- # Find the start event with matching UUID
215
- start_step = next(
216
- (x
217
- for x in self._history if x.event_type == IntermediateStepType.FUNCTION_START and x.UUID == step.UUID),
218
- None)
219
-
220
- # For function end events, display output data
221
- if step.data and hasattr(step.data, 'output'):
222
- output_str = str(step.data.output)
223
- elif step.data:
224
- output_str = str(step.data)
225
-
226
- if not output_str:
227
- return None
228
-
229
- escaped_output = html.escape(output_str, quote=False)
230
- format_output_type = "json" if is_valid_json(escaped_output) else "python"
231
-
232
- # Get input from start step if available
233
- input_payload = ""
234
- if start_step and start_step.data:
235
- if hasattr(start_step.data, 'input'):
236
- input_str = str(start_step.data.input)
237
- else:
238
- input_str = str(start_step.data)
239
-
240
- if input_str:
241
- escaped_input = html.escape(input_str, quote=False)
242
- format_input_type = "json" if is_valid_json(escaped_input) else "python"
243
- input_payload = dedent("""
244
- **Function Input:**
245
- ```{format_input_type}
246
- {input_value}
247
- ```
248
- """).strip("\n").format(input_value=escaped_input, format_input_type=format_input_type)
249
-
250
- # Create payload for function end
251
- payload_str = dedent("""
252
- {input_payload}**Function Output:**
253
- ```{format_output_type}
254
- {output_value}
255
- ```
256
- """).strip("\n").format(input_payload=input_payload,
257
- output_value=escaped_output,
258
- format_output_type=format_output_type)
259
-
260
- event = AIQResponseIntermediateStep(id=step.UUID,
261
- name=f"Function Complete: {step.name}",
262
- payload=payload_str,
263
- parent_id=ancestry.parent_id)
264
- return event
265
-
266
- return None
267
-
268
- def _handle_custom(self, payload: IntermediateStepPayload,
269
- ancestry: InvocationNode) -> AIQResponseSerializable | None:
270
- """
271
- Handles the CUSTOM event
272
- """
273
- escaped_payload = html.escape(str(payload), quote=False)
274
- escaped_payload = escaped_payload.replace("\n", "")
275
-
276
- # Attempt to determine type
277
- format_type = "json" if is_valid_json(escaped_payload) else "python"
278
-
279
- # Don't use f-strings here because the payload is markdown and screws up the dedent
280
- payload_str = dedent("""
281
- ```{format_type}
282
- {payload}
283
- ```
284
- """).strip("\n").format(payload=escaped_payload, format_type=format_type)
285
-
286
- # Return the event
287
- event = AIQResponseIntermediateStep(id=payload.UUID,
288
- name=f"{payload.event_type}",
289
- payload=payload_str,
290
- parent_id=ancestry.function_id)
291
-
292
- return event
293
-
294
- def process(self, step: IntermediateStep) -> AIQResponseSerializable | None:
295
- # Track the chunk
296
- self._history.append(step)
297
- payload = step.payload
298
- ancestry = step.function_ancestry
299
-
300
- if not self._step_matches_filter(step, self.config):
301
- return None
302
-
303
- try:
304
-
305
- if step.event_category == IntermediateStepCategory.LLM:
306
- return self._handle_llm(payload, ancestry)
307
-
308
- if step.event_category == IntermediateStepCategory.TOOL:
309
- return self._handle_tool(payload, ancestry)
310
-
311
- if step.event_category == IntermediateStepCategory.FUNCTION:
312
- return self._handle_function(payload, ancestry)
313
-
314
- if step.event_category == IntermediateStepCategory.CUSTOM:
315
- return self._handle_custom(payload, ancestry)
316
-
317
- except Exception as e:
318
- logger.error("Error processing intermediate step: %s", e, exc_info=True)
319
-
320
- return None
@@ -1,148 +0,0 @@
1
- # SPDX-FileCopyrightText: Copyright (c) 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 asyncio
17
- import logging
18
- import typing
19
- from collections.abc import Awaitable
20
- from collections.abc import Callable
21
- from typing import Any
22
-
23
- from fastapi import WebSocket
24
- from fastapi import WebSocketException
25
- from starlette.endpoints import WebSocketEndpoint
26
- from starlette.websockets import WebSocketDisconnect
27
-
28
- from aiq.data_models.api_server import AIQChatRequest
29
- from aiq.data_models.api_server import AIQChatResponse
30
- from aiq.data_models.api_server import AIQChatResponseChunk
31
- from aiq.data_models.api_server import AIQResponsePayloadOutput
32
- from aiq.data_models.api_server import AIQResponseSerializable
33
- from aiq.data_models.api_server import WebSocketMessageStatus
34
- from aiq.data_models.api_server import WorkflowSchemaType
35
- from aiq.front_ends.fastapi.message_handler import MessageHandler
36
- from aiq.front_ends.fastapi.response_helpers import generate_streaming_response
37
- from aiq.front_ends.fastapi.step_adaptor import StepAdaptor
38
- from aiq.runtime.session import AIQSessionManager
39
-
40
- logger = logging.getLogger(__name__)
41
-
42
-
43
- class AIQWebSocket(WebSocketEndpoint):
44
- encoding = "json"
45
-
46
- def __init__(self, session_manager: AIQSessionManager, step_adaptor: StepAdaptor, *args, **kwargs):
47
- self._session_manager: AIQSessionManager = session_manager
48
- self._message_handler: MessageHandler = MessageHandler(self)
49
- self._process_response_event: asyncio.Event = asyncio.Event()
50
- self._workflow_schema_type: dict[str, Callable[..., Awaitable[Any]]] = {
51
- WorkflowSchemaType.GENERATE_STREAM: self.process_generate_stream,
52
- WorkflowSchemaType.CHAT_STREAM: self.process_chat_stream,
53
- WorkflowSchemaType.GENERATE: self.process_generate,
54
- WorkflowSchemaType.CHAT: self.process_chat
55
- }
56
- self._step_adaptor = step_adaptor
57
- super().__init__(*args, **kwargs)
58
-
59
- @property
60
- def workflow_schema_type(self) -> dict[str, Callable[..., Awaitable[Any]]]:
61
- return self._workflow_schema_type
62
-
63
- @property
64
- def process_response_event(self) -> asyncio.Event:
65
- return self._process_response_event
66
-
67
- async def on_connect(self, websocket: WebSocket):
68
- try:
69
- # Accept the websocket connection
70
- await websocket.accept()
71
- try:
72
- # Start background message processors
73
- self._message_handler.process_messages_task = asyncio.create_task(
74
- self._message_handler.process_messages())
75
-
76
- self._message_handler.process_out_going_messages_task = asyncio.create_task(
77
- self._message_handler.process_out_going_messages(websocket))
78
-
79
- except asyncio.CancelledError:
80
- pass
81
-
82
- except (WebSocketDisconnect, WebSocketException):
83
- logger.error("A WebSocket error occured during `on_connect`. Ignoring the connection.", exc_info=True)
84
-
85
- async def on_send(self, websocket: WebSocket, data: dict[str, str]):
86
- try:
87
- await websocket.send_json(data)
88
- except (WebSocketDisconnect, WebSocketException, Exception):
89
- logger.error("A WebSocket error occurred during `on_send`. Ignoring the connection.", exc_info=True)
90
-
91
- async def on_receive(self, websocket: WebSocket, data: dict[str, Any]):
92
- try:
93
- await self._message_handler.messages_queue.put(data)
94
- except (Exception):
95
- logger.error("An unxpected error occurred during `on_receive`. Ignoring the exception", exc_info=True)
96
-
97
- async def on_disconnect(self, websocket: WebSocket, close_code: Any):
98
- try:
99
- if self._message_handler.process_messages_task:
100
- self._message_handler.process_messages_task.cancel()
101
-
102
- if self._message_handler.process_out_going_messages_task:
103
- self._message_handler.process_out_going_messages_task.cancel()
104
-
105
- if self._message_handler.background_task:
106
- self._message_handler.background_task.cancel()
107
-
108
- except (WebSocketDisconnect, asyncio.CancelledError):
109
- pass
110
-
111
- async def _process_message(self,
112
- payload: typing.Any,
113
- result_type: type | None = None,
114
- output_type: type | None = None) -> None:
115
-
116
- async with self._session_manager.session(
117
- user_input_callback=self._message_handler.human_interaction) as session:
118
-
119
- async for value in generate_streaming_response(payload,
120
- session_manager=session,
121
- streaming=True,
122
- step_adaptor=self._step_adaptor,
123
- result_type=result_type,
124
- output_type=output_type):
125
-
126
- await self._process_response_event.wait()
127
-
128
- if not isinstance(value, AIQResponseSerializable):
129
- value = AIQResponsePayloadOutput(payload=value)
130
-
131
- await self._message_handler.create_websocket_message(data_model=value,
132
- status=WebSocketMessageStatus.IN_PROGRESS)
133
-
134
- async def process_generate_stream(self, payload: str):
135
-
136
- return await self._process_message(payload, result_type=None, output_type=None)
137
-
138
- async def process_chat_stream(self, payload: AIQChatRequest):
139
-
140
- return await self._process_message(payload, result_type=AIQChatResponse, output_type=AIQChatResponseChunk)
141
-
142
- async def process_generate(self, payload: typing.Any):
143
-
144
- return await self._process_message(payload)
145
-
146
- async def process_chat(self, payload: AIQChatRequest):
147
-
148
- return await self._process_message(payload, result_type=AIQChatResponse)
@@ -1,14 +0,0 @@
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.
@@ -1,32 +0,0 @@
1
- # SPDX-FileCopyrightText: Copyright (c) 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
- from pydantic import Field
17
-
18
- from aiq.data_models.front_end import FrontEndBaseConfig
19
-
20
-
21
- class MCPFrontEndConfig(FrontEndBaseConfig, name="mcp"):
22
- """MCP front end configuration.
23
-
24
- A simple MCP (Modular Communication Protocol) front end for AIQ.
25
- """
26
-
27
- name: str = Field(default="AIQ MCP", description="Name of the MCP server")
28
- host: str = Field(default="localhost", description="Host to bind the server to")
29
- port: int = Field(default=9901, description="Port to bind the server to", ge=0, le=65535)
30
- debug: bool = Field(default=False, description="Enable debug mode")
31
- log_level: str = Field(default="INFO", description="Log level for the MCP server")
32
- tool_names: list[str] = Field(default_factory=list, description="The list of tools MCP server will expose.")
@@ -1,93 +0,0 @@
1
- # SPDX-FileCopyrightText: Copyright (c) 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
-
18
- from aiq.builder.front_end import FrontEndBase
19
- from aiq.builder.function import Function
20
- from aiq.builder.workflow import Workflow
21
- from aiq.builder.workflow_builder import WorkflowBuilder
22
- from aiq.front_ends.mcp.mcp_front_end_config import MCPFrontEndConfig
23
-
24
- logger = logging.getLogger(__name__)
25
-
26
-
27
- class MCPFrontEndPlugin(FrontEndBase[MCPFrontEndConfig]):
28
- """MCP front end plugin implementation."""
29
-
30
- async def run(self) -> None:
31
- """Run the MCP server."""
32
- # Import FastMCP
33
- from mcp.server.fastmcp import FastMCP
34
-
35
- from aiq.front_ends.mcp.tool_converter import register_function_with_mcp
36
-
37
- # Create an MCP server with the configured parameters
38
- mcp = FastMCP(
39
- self.front_end_config.name,
40
- host=self.front_end_config.host,
41
- port=self.front_end_config.port,
42
- debug=self.front_end_config.debug,
43
- log_level=self.front_end_config.log_level,
44
- )
45
-
46
- # Build the workflow and register all functions with MCP
47
- async with WorkflowBuilder.from_config(config=self.full_config) as builder:
48
- # Build the workflow
49
- workflow = builder.build()
50
-
51
- # Get all functions from the workflow
52
- functions = self._get_all_functions(workflow)
53
-
54
- # Filter functions based on tool_names if provided
55
- if self.front_end_config.tool_names:
56
- logger.info(f"Filtering functions based on tool_names: {self.front_end_config.tool_names}")
57
- filtered_functions: dict[str, Function] = {}
58
- for function_name, function in functions.items():
59
- if function_name in self.front_end_config.tool_names:
60
- filtered_functions[function_name] = function
61
- else:
62
- logger.debug(f"Skipping function {function_name} as it's not in tool_names")
63
- functions = filtered_functions
64
-
65
- # Register each function with MCP
66
- for function_name, function in functions.items():
67
- register_function_with_mcp(mcp, function_name, function)
68
-
69
- # Add a simple fallback function if no functions were found
70
- if not functions:
71
- raise RuntimeError("No functions found in workflow. Please check your configuration.")
72
-
73
- # Start the MCP server
74
- await mcp.run_sse_async()
75
-
76
- def _get_all_functions(self, workflow: Workflow) -> dict[str, Function]:
77
- """Get all functions from the workflow.
78
-
79
- Args:
80
- workflow: The AIQ workflow.
81
-
82
- Returns:
83
- Dict mapping function names to Function objects.
84
- """
85
- functions: dict[str, Function] = {}
86
-
87
- # Extract all functions from the workflow
88
- for function_name, function in workflow.functions.items():
89
- functions[function_name] = function
90
-
91
- functions[workflow.config.workflow.type] = workflow
92
-
93
- return functions
@@ -1,27 +0,0 @@
1
- # SPDX-FileCopyrightText: Copyright (c) 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
- from collections.abc import AsyncIterator
17
-
18
- from aiq.cli.register_workflow import register_front_end
19
- from aiq.data_models.config import AIQConfig
20
- from aiq.front_ends.mcp.mcp_front_end_config import MCPFrontEndConfig
21
-
22
-
23
- @register_front_end(config_type=MCPFrontEndConfig)
24
- async def register_mcp_front_end(config: MCPFrontEndConfig, full_config: AIQConfig) -> AsyncIterator:
25
- from aiq.front_ends.mcp.mcp_front_end_plugin import MCPFrontEndPlugin
26
-
27
- yield MCPFrontEndPlugin(full_config=full_config)