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,125 @@
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 contextvars
17
+ import dataclasses
18
+ import logging
19
+
20
+ from aiq.data_models.intermediate_step import IntermediateStep
21
+ from aiq.data_models.intermediate_step import IntermediateStepPayload
22
+ from aiq.data_models.intermediate_step import IntermediateStepState
23
+ from aiq.data_models.invocation_node import InvocationNode
24
+ from aiq.utils.reactive.observable import OnComplete
25
+ from aiq.utils.reactive.observable import OnError
26
+ from aiq.utils.reactive.observable import OnNext
27
+ from aiq.utils.reactive.subscription import Subscription
28
+
29
+ logger = logging.getLogger(__name__)
30
+
31
+ _current_open_step_id = contextvars.ContextVar[str | None]("_current_open_step_id", default=None)
32
+
33
+
34
+ @dataclasses.dataclass
35
+ class OpenStep:
36
+ step_id: str
37
+ step_name: str
38
+ step_type: str
39
+ step_parent_id: str | None
40
+ context: contextvars.Context
41
+ token: contextvars.Token[str | None]
42
+
43
+
44
+ class IntermediateStepManager:
45
+ """
46
+ Manages updates to the AgentIQ Event Stream for intermediate steps
47
+ """
48
+
49
+ def __init__(self, context_state: "AIQContextState"): # noqa: F821
50
+ self._context_state = context_state
51
+
52
+ self._outstanding_start_steps: dict[str, OpenStep] = {}
53
+
54
+ def push_intermediate_step(self, payload: IntermediateStepPayload) -> None:
55
+ """
56
+ Pushes an intermediate step to the AgentIQ Event Stream
57
+ """
58
+
59
+ if not isinstance(payload, IntermediateStepPayload):
60
+ raise TypeError(f"Payload must be of type IntermediateStepPayload, not {type(payload)}")
61
+
62
+ parent_step_id = _current_open_step_id.get()
63
+
64
+ if (payload.event_state == IntermediateStepState.START):
65
+
66
+ token = _current_open_step_id.set(payload.UUID)
67
+
68
+ self._outstanding_start_steps[payload.UUID] = OpenStep(step_id=payload.UUID,
69
+ step_name=payload.name,
70
+ step_type=payload.event_type,
71
+ step_parent_id=parent_step_id,
72
+ context=contextvars.copy_context(),
73
+ token=token)
74
+
75
+ elif (payload.event_state == IntermediateStepState.END):
76
+
77
+ # Remove the current step from the outstanding steps
78
+ open_step = self._outstanding_start_steps.pop(payload.UUID, None)
79
+
80
+ if (open_step is None):
81
+ logger.warning("Step id %s not found in outstanding start steps", payload.UUID)
82
+ return
83
+
84
+ # If we are in the same coroutine, we should have the same parent step id. If so, unset the current step id.
85
+ if (parent_step_id == payload.UUID):
86
+ _current_open_step_id.reset(open_step.token)
87
+
88
+ else:
89
+ # Manually set the parent step ID. This happens when running on the thread pool
90
+ parent_step_id = open_step.step_parent_id
91
+ elif (payload.event_state == IntermediateStepState.CHUNK):
92
+
93
+ # Get the current step from the outstanding steps
94
+ open_step = self._outstanding_start_steps.get(payload.UUID, None)
95
+
96
+ # Generate a warning if the parent step id is not set to the current step id
97
+ if (open_step is None):
98
+ logger.warning(
99
+ "Created a chunk for step %s, but no matching start step was found. "
100
+ "Chunks must be created with the same ID as the start step.",
101
+ payload.UUID)
102
+ return
103
+
104
+ if (parent_step_id != payload.UUID):
105
+ # Manually set the parent step ID. This happens when running on the thread pool
106
+ parent_step_id = open_step.step_parent_id
107
+
108
+ function_ancestry = InvocationNode(function_name=self._context_state.active_function.get().function_name,
109
+ function_id=self._context_state.active_function.get().function_id,
110
+ parent_id=parent_step_id,
111
+ parent_name=self._context_state.active_function.get().parent_name)
112
+
113
+ intermediate_step = IntermediateStep(function_ancestry=function_ancestry, payload=payload)
114
+
115
+ self._context_state.event_stream.get().on_next(intermediate_step)
116
+
117
+ def subscribe(self,
118
+ on_next: OnNext[IntermediateStep],
119
+ on_error: OnError = None,
120
+ on_complete: OnComplete = None) -> Subscription:
121
+ """
122
+ Subscribes to the AgentIQ Event Stream for intermediate steps
123
+ """
124
+
125
+ return self._context_state.event_stream.get().subscribe(on_next, on_error, on_complete)
aiq/builder/llm.py ADDED
@@ -0,0 +1,25 @@
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
+ from aiq.data_models.llm import LLMBaseConfig
17
+
18
+
19
+ class LLMProviderInfo:
20
+
21
+ def __init__(self, *, config: LLMBaseConfig, description: str):
22
+
23
+ self.config = config
24
+ self.provider_type = type(config).static_type()
25
+ self.description = description
@@ -0,0 +1,25 @@
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 aiq.data_models.retriever import RetrieverBaseConfig
17
+
18
+
19
+ class RetrieverProviderInfo:
20
+
21
+ def __init__(self, *, config: RetrieverBaseConfig, description: str):
22
+
23
+ self.config = config
24
+ self.provider_type = type(config).static_type()
25
+ self.description = description
@@ -0,0 +1,71 @@
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 time
17
+ import uuid
18
+
19
+ from aiq.data_models.interactive import HumanPrompt
20
+ from aiq.data_models.interactive import HumanResponse
21
+ from aiq.data_models.interactive import InteractionPrompt
22
+ from aiq.data_models.interactive import InteractionResponse
23
+ from aiq.data_models.interactive import InteractionStatus
24
+
25
+
26
+ class AIQUserInteractionManager:
27
+ """
28
+ AIQUserInteractionManager is responsible for requesting user input
29
+ at runtime. It delegates the actual prompting to a callback function
30
+ stored in AIQContextState.user_input_callback.
31
+
32
+ Type is not imported in __init__ to prevent partial import.
33
+ """
34
+
35
+ def __init__(self, context_state: "AIQContextState") -> None: # noqa: F821
36
+ self._context_state = context_state
37
+
38
+ @staticmethod
39
+ async def default_callback_handler(prompt: InteractionPrompt) -> HumanResponse:
40
+ """
41
+ Default callback handler for user input. This is a no-op function
42
+ that simply returns the input text from the Interaction Content
43
+ object.
44
+
45
+ Args:
46
+ prompt (InteractionPrompt): The interaction to process.
47
+ """
48
+ raise NotImplementedError("No human prompt callback was registered. Unable to handle requested prompt.")
49
+
50
+ async def prompt_user_input(self, content: HumanPrompt) -> InteractionResponse:
51
+ """
52
+ Ask the user a question and wait for input. This calls out to
53
+ the callback from user_input_callback, which is typically
54
+ set by AIQSessionManager.
55
+
56
+ Returns the user's typed-in answer as a string.
57
+ """
58
+
59
+ uuid_req = str(uuid.uuid4())
60
+ status = InteractionStatus.IN_PROGRESS
61
+ timestamp = time.strftime("%Y-%m-%dT%H:%M:%SZ")
62
+ sys_human_interaction = InteractionPrompt(id=uuid_req, status=status, timestamp=timestamp, content=content)
63
+
64
+ resp = await self._context_state.user_input_callback.get()(sys_human_interaction)
65
+
66
+ # Rebuild a InteractionResponse object with the response
67
+ timestamp = time.strftime("%Y-%m-%dT%H:%M:%SZ")
68
+ status = InteractionStatus.COMPLETED
69
+ sys_human_interaction = InteractionResponse(id=uuid_req, status=status, timestamp=timestamp, content=resp)
70
+
71
+ return sys_human_interaction
@@ -0,0 +1,134 @@
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
+ from contextlib import asynccontextmanager
17
+ from contextvars import ContextVar
18
+ from typing import Any
19
+
20
+ from opentelemetry.sdk.trace.export import SpanExporter
21
+
22
+ from aiq.builder.context import AIQContextState
23
+ from aiq.builder.embedder import EmbedderProviderInfo
24
+ from aiq.builder.function import Function
25
+ from aiq.builder.function_base import FunctionBase
26
+ from aiq.builder.function_base import InputT
27
+ from aiq.builder.function_base import SingleOutputT
28
+ from aiq.builder.function_base import StreamingOutputT
29
+ from aiq.builder.llm import LLMProviderInfo
30
+ from aiq.builder.retriever import RetrieverProviderInfo
31
+ from aiq.data_models.config import AIQConfig
32
+ from aiq.memory.interfaces import MemoryEditor
33
+ from aiq.runtime.runner import AIQRunner
34
+
35
+ callback_handler_var: ContextVar[Any | None] = ContextVar("callback_handler_var", default=None)
36
+
37
+
38
+ class Workflow(FunctionBase[InputT, StreamingOutputT, SingleOutputT]):
39
+
40
+ def __init__(self,
41
+ *,
42
+ config: AIQConfig,
43
+ entry_fn: Function[InputT, StreamingOutputT, SingleOutputT],
44
+ functions: dict[str, Function] | None = None,
45
+ llms: dict[str, LLMProviderInfo] | None = None,
46
+ embeddings: dict[str, EmbedderProviderInfo] | None = None,
47
+ memory: dict[str, MemoryEditor] | None = None,
48
+ exporters: dict[str, SpanExporter] | None = None,
49
+ retrievers: dict[str | None, RetrieverProviderInfo] | None = None,
50
+ context_state: AIQContextState):
51
+
52
+ super().__init__(input_schema=entry_fn.input_schema,
53
+ streaming_output_schema=entry_fn.streaming_output_schema,
54
+ single_output_schema=entry_fn.single_output_schema)
55
+
56
+ self.config = config
57
+ self.functions = functions or {}
58
+ self.llms = llms or {}
59
+ self.embeddings = embeddings or {}
60
+ self.memory = memory or {}
61
+ self.retrievers = retrievers or {}
62
+
63
+ self._entry_fn = entry_fn
64
+
65
+ self._context_state = context_state
66
+
67
+ self._exporters = exporters or {}
68
+
69
+ @property
70
+ def has_streaming_output(self) -> bool:
71
+
72
+ return self._entry_fn.has_streaming_output
73
+
74
+ @property
75
+ def has_single_output(self) -> bool:
76
+
77
+ return self._entry_fn.has_single_output
78
+
79
+ @asynccontextmanager
80
+ async def run(self, message: InputT):
81
+ """
82
+ Called each time we start a new workflow run. We'll create
83
+ a new top-level workflow span here.
84
+ """
85
+ async with AIQRunner(input_message=message, entry_fn=self._entry_fn,
86
+ context_state=self._context_state) as runner:
87
+
88
+ # The caller can `yield runner` so they can do `runner.result()` or `runner.result_stream()`
89
+ yield runner
90
+
91
+ async def result_with_steps(self, message: InputT, to_type: type | None = None):
92
+
93
+ async with self.run(message) as runner:
94
+
95
+ from aiq.eval.runtime_event_subscriber import pull_intermediate
96
+
97
+ # Start the intermediate stream
98
+ pull_done, intermediate_steps = pull_intermediate()
99
+
100
+ # Wait on the result
101
+ result = await runner.result(to_type=to_type)
102
+
103
+ await pull_done.wait()
104
+
105
+ return result, intermediate_steps
106
+
107
+ @staticmethod
108
+ def from_entry_fn(*,
109
+ config: AIQConfig,
110
+ entry_fn: Function[InputT, StreamingOutputT, SingleOutputT],
111
+ functions: dict[str, Function] | None = None,
112
+ llms: dict[str, LLMProviderInfo] | None = None,
113
+ embeddings: dict[str, EmbedderProviderInfo] | None = None,
114
+ memory: dict[str, MemoryEditor] | None = None,
115
+ exporters: dict[str, SpanExporter] | None = None,
116
+ retrievers: dict[str | None, RetrieverProviderInfo] | None = None,
117
+ context_state: AIQContextState) -> 'Workflow[InputT, StreamingOutputT, SingleOutputT]':
118
+
119
+ input_type: type = entry_fn.input_type
120
+ streaming_output_type = entry_fn.streaming_output_type
121
+ single_output_type = entry_fn.single_output_type
122
+
123
+ class WorkflowImpl(Workflow[input_type, streaming_output_type, single_output_type]):
124
+ pass
125
+
126
+ return WorkflowImpl(config=config,
127
+ entry_fn=entry_fn,
128
+ functions=functions,
129
+ llms=llms,
130
+ embeddings=embeddings,
131
+ memory=memory,
132
+ exporters=exporters,
133
+ retrievers=retrievers,
134
+ context_state=context_state)