nvidia-nat 1.2.0rc5__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.
Files changed (435) hide show
  1. aiq/agent/__init__.py +0 -0
  2. aiq/agent/base.py +239 -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 +355 -0
  6. aiq/agent/react_agent/output_parser.py +104 -0
  7. aiq/agent/react_agent/prompt.py +41 -0
  8. aiq/agent/react_agent/register.py +149 -0
  9. aiq/agent/reasoning_agent/__init__.py +0 -0
  10. aiq/agent/reasoning_agent/reasoning_agent.py +225 -0
  11. aiq/agent/register.py +23 -0
  12. aiq/agent/rewoo_agent/__init__.py +0 -0
  13. aiq/agent/rewoo_agent/agent.py +411 -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 +119 -0
  18. aiq/agent/tool_calling_agent/register.py +106 -0
  19. aiq/authentication/__init__.py +14 -0
  20. aiq/authentication/api_key/__init__.py +14 -0
  21. aiq/authentication/api_key/api_key_auth_provider.py +96 -0
  22. aiq/authentication/api_key/api_key_auth_provider_config.py +124 -0
  23. aiq/authentication/api_key/register.py +26 -0
  24. aiq/authentication/exceptions/__init__.py +14 -0
  25. aiq/authentication/exceptions/api_key_exceptions.py +38 -0
  26. aiq/authentication/http_basic_auth/__init__.py +0 -0
  27. aiq/authentication/http_basic_auth/http_basic_auth_provider.py +81 -0
  28. aiq/authentication/http_basic_auth/register.py +30 -0
  29. aiq/authentication/interfaces.py +93 -0
  30. aiq/authentication/oauth2/__init__.py +14 -0
  31. aiq/authentication/oauth2/oauth2_auth_code_flow_provider.py +107 -0
  32. aiq/authentication/oauth2/oauth2_auth_code_flow_provider_config.py +39 -0
  33. aiq/authentication/oauth2/register.py +25 -0
  34. aiq/authentication/register.py +21 -0
  35. aiq/builder/__init__.py +0 -0
  36. aiq/builder/builder.py +285 -0
  37. aiq/builder/component_utils.py +316 -0
  38. aiq/builder/context.py +264 -0
  39. aiq/builder/embedder.py +24 -0
  40. aiq/builder/eval_builder.py +161 -0
  41. aiq/builder/evaluator.py +29 -0
  42. aiq/builder/framework_enum.py +24 -0
  43. aiq/builder/front_end.py +73 -0
  44. aiq/builder/function.py +344 -0
  45. aiq/builder/function_base.py +380 -0
  46. aiq/builder/function_info.py +627 -0
  47. aiq/builder/intermediate_step_manager.py +174 -0
  48. aiq/builder/llm.py +25 -0
  49. aiq/builder/retriever.py +25 -0
  50. aiq/builder/user_interaction_manager.py +74 -0
  51. aiq/builder/workflow.py +148 -0
  52. aiq/builder/workflow_builder.py +1117 -0
  53. aiq/cli/__init__.py +14 -0
  54. aiq/cli/cli_utils/__init__.py +0 -0
  55. aiq/cli/cli_utils/config_override.py +231 -0
  56. aiq/cli/cli_utils/validation.py +37 -0
  57. aiq/cli/commands/__init__.py +0 -0
  58. aiq/cli/commands/configure/__init__.py +0 -0
  59. aiq/cli/commands/configure/channel/__init__.py +0 -0
  60. aiq/cli/commands/configure/channel/add.py +28 -0
  61. aiq/cli/commands/configure/channel/channel.py +36 -0
  62. aiq/cli/commands/configure/channel/remove.py +30 -0
  63. aiq/cli/commands/configure/channel/update.py +30 -0
  64. aiq/cli/commands/configure/configure.py +33 -0
  65. aiq/cli/commands/evaluate.py +139 -0
  66. aiq/cli/commands/info/__init__.py +14 -0
  67. aiq/cli/commands/info/info.py +39 -0
  68. aiq/cli/commands/info/list_channels.py +32 -0
  69. aiq/cli/commands/info/list_components.py +129 -0
  70. aiq/cli/commands/info/list_mcp.py +213 -0
  71. aiq/cli/commands/registry/__init__.py +14 -0
  72. aiq/cli/commands/registry/publish.py +88 -0
  73. aiq/cli/commands/registry/pull.py +118 -0
  74. aiq/cli/commands/registry/registry.py +38 -0
  75. aiq/cli/commands/registry/remove.py +108 -0
  76. aiq/cli/commands/registry/search.py +155 -0
  77. aiq/cli/commands/sizing/__init__.py +14 -0
  78. aiq/cli/commands/sizing/calc.py +297 -0
  79. aiq/cli/commands/sizing/sizing.py +27 -0
  80. aiq/cli/commands/start.py +246 -0
  81. aiq/cli/commands/uninstall.py +81 -0
  82. aiq/cli/commands/validate.py +47 -0
  83. aiq/cli/commands/workflow/__init__.py +14 -0
  84. aiq/cli/commands/workflow/templates/__init__.py.j2 +0 -0
  85. aiq/cli/commands/workflow/templates/config.yml.j2 +16 -0
  86. aiq/cli/commands/workflow/templates/pyproject.toml.j2 +22 -0
  87. aiq/cli/commands/workflow/templates/register.py.j2 +5 -0
  88. aiq/cli/commands/workflow/templates/workflow.py.j2 +36 -0
  89. aiq/cli/commands/workflow/workflow.py +37 -0
  90. aiq/cli/commands/workflow/workflow_commands.py +313 -0
  91. aiq/cli/entrypoint.py +135 -0
  92. aiq/cli/main.py +44 -0
  93. aiq/cli/register_workflow.py +488 -0
  94. aiq/cli/type_registry.py +1000 -0
  95. aiq/data_models/__init__.py +14 -0
  96. aiq/data_models/api_server.py +694 -0
  97. aiq/data_models/authentication.py +231 -0
  98. aiq/data_models/common.py +171 -0
  99. aiq/data_models/component.py +54 -0
  100. aiq/data_models/component_ref.py +168 -0
  101. aiq/data_models/config.py +406 -0
  102. aiq/data_models/dataset_handler.py +123 -0
  103. aiq/data_models/discovery_metadata.py +335 -0
  104. aiq/data_models/embedder.py +27 -0
  105. aiq/data_models/evaluate.py +127 -0
  106. aiq/data_models/evaluator.py +26 -0
  107. aiq/data_models/front_end.py +26 -0
  108. aiq/data_models/function.py +30 -0
  109. aiq/data_models/function_dependencies.py +72 -0
  110. aiq/data_models/interactive.py +246 -0
  111. aiq/data_models/intermediate_step.py +302 -0
  112. aiq/data_models/invocation_node.py +38 -0
  113. aiq/data_models/llm.py +27 -0
  114. aiq/data_models/logging.py +26 -0
  115. aiq/data_models/memory.py +27 -0
  116. aiq/data_models/object_store.py +44 -0
  117. aiq/data_models/profiler.py +54 -0
  118. aiq/data_models/registry_handler.py +26 -0
  119. aiq/data_models/retriever.py +30 -0
  120. aiq/data_models/retry_mixin.py +35 -0
  121. aiq/data_models/span.py +187 -0
  122. aiq/data_models/step_adaptor.py +64 -0
  123. aiq/data_models/streaming.py +33 -0
  124. aiq/data_models/swe_bench_model.py +54 -0
  125. aiq/data_models/telemetry_exporter.py +26 -0
  126. aiq/data_models/ttc_strategy.py +30 -0
  127. aiq/embedder/__init__.py +0 -0
  128. aiq/embedder/langchain_client.py +41 -0
  129. aiq/embedder/nim_embedder.py +59 -0
  130. aiq/embedder/openai_embedder.py +43 -0
  131. aiq/embedder/register.py +24 -0
  132. aiq/eval/__init__.py +14 -0
  133. aiq/eval/config.py +60 -0
  134. aiq/eval/dataset_handler/__init__.py +0 -0
  135. aiq/eval/dataset_handler/dataset_downloader.py +106 -0
  136. aiq/eval/dataset_handler/dataset_filter.py +52 -0
  137. aiq/eval/dataset_handler/dataset_handler.py +254 -0
  138. aiq/eval/evaluate.py +506 -0
  139. aiq/eval/evaluator/__init__.py +14 -0
  140. aiq/eval/evaluator/base_evaluator.py +73 -0
  141. aiq/eval/evaluator/evaluator_model.py +45 -0
  142. aiq/eval/intermediate_step_adapter.py +99 -0
  143. aiq/eval/rag_evaluator/__init__.py +0 -0
  144. aiq/eval/rag_evaluator/evaluate.py +178 -0
  145. aiq/eval/rag_evaluator/register.py +143 -0
  146. aiq/eval/register.py +23 -0
  147. aiq/eval/remote_workflow.py +133 -0
  148. aiq/eval/runners/__init__.py +14 -0
  149. aiq/eval/runners/config.py +39 -0
  150. aiq/eval/runners/multi_eval_runner.py +54 -0
  151. aiq/eval/runtime_event_subscriber.py +52 -0
  152. aiq/eval/swe_bench_evaluator/__init__.py +0 -0
  153. aiq/eval/swe_bench_evaluator/evaluate.py +215 -0
  154. aiq/eval/swe_bench_evaluator/register.py +36 -0
  155. aiq/eval/trajectory_evaluator/__init__.py +0 -0
  156. aiq/eval/trajectory_evaluator/evaluate.py +75 -0
  157. aiq/eval/trajectory_evaluator/register.py +40 -0
  158. aiq/eval/tunable_rag_evaluator/__init__.py +0 -0
  159. aiq/eval/tunable_rag_evaluator/evaluate.py +245 -0
  160. aiq/eval/tunable_rag_evaluator/register.py +52 -0
  161. aiq/eval/usage_stats.py +41 -0
  162. aiq/eval/utils/__init__.py +0 -0
  163. aiq/eval/utils/output_uploader.py +140 -0
  164. aiq/eval/utils/tqdm_position_registry.py +40 -0
  165. aiq/eval/utils/weave_eval.py +184 -0
  166. aiq/experimental/__init__.py +0 -0
  167. aiq/experimental/decorators/__init__.py +0 -0
  168. aiq/experimental/decorators/experimental_warning_decorator.py +130 -0
  169. aiq/experimental/test_time_compute/__init__.py +0 -0
  170. aiq/experimental/test_time_compute/editing/__init__.py +0 -0
  171. aiq/experimental/test_time_compute/editing/iterative_plan_refinement_editor.py +147 -0
  172. aiq/experimental/test_time_compute/editing/llm_as_a_judge_editor.py +204 -0
  173. aiq/experimental/test_time_compute/editing/motivation_aware_summarization.py +107 -0
  174. aiq/experimental/test_time_compute/functions/__init__.py +0 -0
  175. aiq/experimental/test_time_compute/functions/execute_score_select_function.py +105 -0
  176. aiq/experimental/test_time_compute/functions/its_tool_orchestration_function.py +205 -0
  177. aiq/experimental/test_time_compute/functions/its_tool_wrapper_function.py +146 -0
  178. aiq/experimental/test_time_compute/functions/plan_select_execute_function.py +224 -0
  179. aiq/experimental/test_time_compute/models/__init__.py +0 -0
  180. aiq/experimental/test_time_compute/models/editor_config.py +132 -0
  181. aiq/experimental/test_time_compute/models/scoring_config.py +112 -0
  182. aiq/experimental/test_time_compute/models/search_config.py +120 -0
  183. aiq/experimental/test_time_compute/models/selection_config.py +154 -0
  184. aiq/experimental/test_time_compute/models/stage_enums.py +43 -0
  185. aiq/experimental/test_time_compute/models/strategy_base.py +66 -0
  186. aiq/experimental/test_time_compute/models/tool_use_config.py +41 -0
  187. aiq/experimental/test_time_compute/models/ttc_item.py +48 -0
  188. aiq/experimental/test_time_compute/register.py +36 -0
  189. aiq/experimental/test_time_compute/scoring/__init__.py +0 -0
  190. aiq/experimental/test_time_compute/scoring/llm_based_agent_scorer.py +168 -0
  191. aiq/experimental/test_time_compute/scoring/llm_based_plan_scorer.py +168 -0
  192. aiq/experimental/test_time_compute/scoring/motivation_aware_scorer.py +111 -0
  193. aiq/experimental/test_time_compute/search/__init__.py +0 -0
  194. aiq/experimental/test_time_compute/search/multi_llm_planner.py +128 -0
  195. aiq/experimental/test_time_compute/search/multi_query_retrieval_search.py +122 -0
  196. aiq/experimental/test_time_compute/search/single_shot_multi_plan_planner.py +128 -0
  197. aiq/experimental/test_time_compute/selection/__init__.py +0 -0
  198. aiq/experimental/test_time_compute/selection/best_of_n_selector.py +63 -0
  199. aiq/experimental/test_time_compute/selection/llm_based_agent_output_selector.py +131 -0
  200. aiq/experimental/test_time_compute/selection/llm_based_output_merging_selector.py +159 -0
  201. aiq/experimental/test_time_compute/selection/llm_based_plan_selector.py +128 -0
  202. aiq/experimental/test_time_compute/selection/threshold_selector.py +58 -0
  203. aiq/front_ends/__init__.py +14 -0
  204. aiq/front_ends/console/__init__.py +14 -0
  205. aiq/front_ends/console/authentication_flow_handler.py +233 -0
  206. aiq/front_ends/console/console_front_end_config.py +32 -0
  207. aiq/front_ends/console/console_front_end_plugin.py +96 -0
  208. aiq/front_ends/console/register.py +25 -0
  209. aiq/front_ends/cron/__init__.py +14 -0
  210. aiq/front_ends/fastapi/__init__.py +14 -0
  211. aiq/front_ends/fastapi/auth_flow_handlers/__init__.py +0 -0
  212. aiq/front_ends/fastapi/auth_flow_handlers/http_flow_handler.py +27 -0
  213. aiq/front_ends/fastapi/auth_flow_handlers/websocket_flow_handler.py +107 -0
  214. aiq/front_ends/fastapi/fastapi_front_end_config.py +234 -0
  215. aiq/front_ends/fastapi/fastapi_front_end_controller.py +68 -0
  216. aiq/front_ends/fastapi/fastapi_front_end_plugin.py +116 -0
  217. aiq/front_ends/fastapi/fastapi_front_end_plugin_worker.py +1092 -0
  218. aiq/front_ends/fastapi/html_snippets/__init__.py +14 -0
  219. aiq/front_ends/fastapi/html_snippets/auth_code_grant_success.py +35 -0
  220. aiq/front_ends/fastapi/intermediate_steps_subscriber.py +80 -0
  221. aiq/front_ends/fastapi/job_store.py +183 -0
  222. aiq/front_ends/fastapi/main.py +72 -0
  223. aiq/front_ends/fastapi/message_handler.py +298 -0
  224. aiq/front_ends/fastapi/message_validator.py +345 -0
  225. aiq/front_ends/fastapi/register.py +25 -0
  226. aiq/front_ends/fastapi/response_helpers.py +195 -0
  227. aiq/front_ends/fastapi/step_adaptor.py +321 -0
  228. aiq/front_ends/mcp/__init__.py +14 -0
  229. aiq/front_ends/mcp/mcp_front_end_config.py +32 -0
  230. aiq/front_ends/mcp/mcp_front_end_plugin.py +93 -0
  231. aiq/front_ends/mcp/register.py +27 -0
  232. aiq/front_ends/mcp/tool_converter.py +242 -0
  233. aiq/front_ends/register.py +22 -0
  234. aiq/front_ends/simple_base/__init__.py +14 -0
  235. aiq/front_ends/simple_base/simple_front_end_plugin_base.py +54 -0
  236. aiq/llm/__init__.py +0 -0
  237. aiq/llm/aws_bedrock_llm.py +57 -0
  238. aiq/llm/nim_llm.py +46 -0
  239. aiq/llm/openai_llm.py +46 -0
  240. aiq/llm/register.py +23 -0
  241. aiq/llm/utils/__init__.py +14 -0
  242. aiq/llm/utils/env_config_value.py +94 -0
  243. aiq/llm/utils/error.py +17 -0
  244. aiq/memory/__init__.py +20 -0
  245. aiq/memory/interfaces.py +183 -0
  246. aiq/memory/models.py +112 -0
  247. aiq/meta/module_to_distro.json +3 -0
  248. aiq/meta/pypi.md +58 -0
  249. aiq/object_store/__init__.py +20 -0
  250. aiq/object_store/in_memory_object_store.py +76 -0
  251. aiq/object_store/interfaces.py +84 -0
  252. aiq/object_store/models.py +36 -0
  253. aiq/object_store/register.py +20 -0
  254. aiq/observability/__init__.py +14 -0
  255. aiq/observability/exporter/__init__.py +14 -0
  256. aiq/observability/exporter/base_exporter.py +449 -0
  257. aiq/observability/exporter/exporter.py +78 -0
  258. aiq/observability/exporter/file_exporter.py +33 -0
  259. aiq/observability/exporter/processing_exporter.py +322 -0
  260. aiq/observability/exporter/raw_exporter.py +52 -0
  261. aiq/observability/exporter/span_exporter.py +265 -0
  262. aiq/observability/exporter_manager.py +335 -0
  263. aiq/observability/mixin/__init__.py +14 -0
  264. aiq/observability/mixin/batch_config_mixin.py +26 -0
  265. aiq/observability/mixin/collector_config_mixin.py +23 -0
  266. aiq/observability/mixin/file_mixin.py +288 -0
  267. aiq/observability/mixin/file_mode.py +23 -0
  268. aiq/observability/mixin/resource_conflict_mixin.py +134 -0
  269. aiq/observability/mixin/serialize_mixin.py +61 -0
  270. aiq/observability/mixin/type_introspection_mixin.py +183 -0
  271. aiq/observability/processor/__init__.py +14 -0
  272. aiq/observability/processor/batching_processor.py +310 -0
  273. aiq/observability/processor/callback_processor.py +42 -0
  274. aiq/observability/processor/intermediate_step_serializer.py +28 -0
  275. aiq/observability/processor/processor.py +71 -0
  276. aiq/observability/register.py +96 -0
  277. aiq/observability/utils/__init__.py +14 -0
  278. aiq/observability/utils/dict_utils.py +236 -0
  279. aiq/observability/utils/time_utils.py +31 -0
  280. aiq/plugins/.namespace +1 -0
  281. aiq/profiler/__init__.py +0 -0
  282. aiq/profiler/calc/__init__.py +14 -0
  283. aiq/profiler/calc/calc_runner.py +627 -0
  284. aiq/profiler/calc/calculations.py +288 -0
  285. aiq/profiler/calc/data_models.py +188 -0
  286. aiq/profiler/calc/plot.py +345 -0
  287. aiq/profiler/callbacks/__init__.py +0 -0
  288. aiq/profiler/callbacks/agno_callback_handler.py +295 -0
  289. aiq/profiler/callbacks/base_callback_class.py +20 -0
  290. aiq/profiler/callbacks/langchain_callback_handler.py +290 -0
  291. aiq/profiler/callbacks/llama_index_callback_handler.py +205 -0
  292. aiq/profiler/callbacks/semantic_kernel_callback_handler.py +238 -0
  293. aiq/profiler/callbacks/token_usage_base_model.py +27 -0
  294. aiq/profiler/data_frame_row.py +51 -0
  295. aiq/profiler/data_models.py +24 -0
  296. aiq/profiler/decorators/__init__.py +0 -0
  297. aiq/profiler/decorators/framework_wrapper.py +131 -0
  298. aiq/profiler/decorators/function_tracking.py +254 -0
  299. aiq/profiler/forecasting/__init__.py +0 -0
  300. aiq/profiler/forecasting/config.py +18 -0
  301. aiq/profiler/forecasting/model_trainer.py +75 -0
  302. aiq/profiler/forecasting/models/__init__.py +22 -0
  303. aiq/profiler/forecasting/models/forecasting_base_model.py +40 -0
  304. aiq/profiler/forecasting/models/linear_model.py +196 -0
  305. aiq/profiler/forecasting/models/random_forest_regressor.py +268 -0
  306. aiq/profiler/inference_metrics_model.py +28 -0
  307. aiq/profiler/inference_optimization/__init__.py +0 -0
  308. aiq/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
  309. aiq/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +460 -0
  310. aiq/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +258 -0
  311. aiq/profiler/inference_optimization/data_models.py +386 -0
  312. aiq/profiler/inference_optimization/experimental/__init__.py +0 -0
  313. aiq/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +468 -0
  314. aiq/profiler/inference_optimization/experimental/prefix_span_analysis.py +405 -0
  315. aiq/profiler/inference_optimization/llm_metrics.py +212 -0
  316. aiq/profiler/inference_optimization/prompt_caching.py +163 -0
  317. aiq/profiler/inference_optimization/token_uniqueness.py +107 -0
  318. aiq/profiler/inference_optimization/workflow_runtimes.py +72 -0
  319. aiq/profiler/intermediate_property_adapter.py +102 -0
  320. aiq/profiler/profile_runner.py +473 -0
  321. aiq/profiler/utils.py +184 -0
  322. aiq/registry_handlers/__init__.py +0 -0
  323. aiq/registry_handlers/local/__init__.py +0 -0
  324. aiq/registry_handlers/local/local_handler.py +176 -0
  325. aiq/registry_handlers/local/register_local.py +37 -0
  326. aiq/registry_handlers/metadata_factory.py +60 -0
  327. aiq/registry_handlers/package_utils.py +567 -0
  328. aiq/registry_handlers/pypi/__init__.py +0 -0
  329. aiq/registry_handlers/pypi/pypi_handler.py +251 -0
  330. aiq/registry_handlers/pypi/register_pypi.py +40 -0
  331. aiq/registry_handlers/register.py +21 -0
  332. aiq/registry_handlers/registry_handler_base.py +157 -0
  333. aiq/registry_handlers/rest/__init__.py +0 -0
  334. aiq/registry_handlers/rest/register_rest.py +56 -0
  335. aiq/registry_handlers/rest/rest_handler.py +237 -0
  336. aiq/registry_handlers/schemas/__init__.py +0 -0
  337. aiq/registry_handlers/schemas/headers.py +42 -0
  338. aiq/registry_handlers/schemas/package.py +68 -0
  339. aiq/registry_handlers/schemas/publish.py +63 -0
  340. aiq/registry_handlers/schemas/pull.py +82 -0
  341. aiq/registry_handlers/schemas/remove.py +36 -0
  342. aiq/registry_handlers/schemas/search.py +91 -0
  343. aiq/registry_handlers/schemas/status.py +47 -0
  344. aiq/retriever/__init__.py +0 -0
  345. aiq/retriever/interface.py +37 -0
  346. aiq/retriever/milvus/__init__.py +14 -0
  347. aiq/retriever/milvus/register.py +81 -0
  348. aiq/retriever/milvus/retriever.py +228 -0
  349. aiq/retriever/models.py +74 -0
  350. aiq/retriever/nemo_retriever/__init__.py +14 -0
  351. aiq/retriever/nemo_retriever/register.py +60 -0
  352. aiq/retriever/nemo_retriever/retriever.py +190 -0
  353. aiq/retriever/register.py +22 -0
  354. aiq/runtime/__init__.py +14 -0
  355. aiq/runtime/loader.py +215 -0
  356. aiq/runtime/runner.py +190 -0
  357. aiq/runtime/session.py +158 -0
  358. aiq/runtime/user_metadata.py +130 -0
  359. aiq/settings/__init__.py +0 -0
  360. aiq/settings/global_settings.py +318 -0
  361. aiq/test/.namespace +1 -0
  362. aiq/tool/__init__.py +0 -0
  363. aiq/tool/chat_completion.py +74 -0
  364. aiq/tool/code_execution/README.md +151 -0
  365. aiq/tool/code_execution/__init__.py +0 -0
  366. aiq/tool/code_execution/code_sandbox.py +267 -0
  367. aiq/tool/code_execution/local_sandbox/.gitignore +1 -0
  368. aiq/tool/code_execution/local_sandbox/Dockerfile.sandbox +60 -0
  369. aiq/tool/code_execution/local_sandbox/__init__.py +13 -0
  370. aiq/tool/code_execution/local_sandbox/local_sandbox_server.py +198 -0
  371. aiq/tool/code_execution/local_sandbox/sandbox.requirements.txt +6 -0
  372. aiq/tool/code_execution/local_sandbox/start_local_sandbox.sh +50 -0
  373. aiq/tool/code_execution/register.py +74 -0
  374. aiq/tool/code_execution/test_code_execution_sandbox.py +414 -0
  375. aiq/tool/code_execution/utils.py +100 -0
  376. aiq/tool/datetime_tools.py +42 -0
  377. aiq/tool/document_search.py +141 -0
  378. aiq/tool/github_tools/__init__.py +0 -0
  379. aiq/tool/github_tools/create_github_commit.py +133 -0
  380. aiq/tool/github_tools/create_github_issue.py +87 -0
  381. aiq/tool/github_tools/create_github_pr.py +106 -0
  382. aiq/tool/github_tools/get_github_file.py +106 -0
  383. aiq/tool/github_tools/get_github_issue.py +166 -0
  384. aiq/tool/github_tools/get_github_pr.py +256 -0
  385. aiq/tool/github_tools/update_github_issue.py +100 -0
  386. aiq/tool/mcp/__init__.py +14 -0
  387. aiq/tool/mcp/exceptions.py +142 -0
  388. aiq/tool/mcp/mcp_client.py +255 -0
  389. aiq/tool/mcp/mcp_tool.py +96 -0
  390. aiq/tool/memory_tools/__init__.py +0 -0
  391. aiq/tool/memory_tools/add_memory_tool.py +79 -0
  392. aiq/tool/memory_tools/delete_memory_tool.py +67 -0
  393. aiq/tool/memory_tools/get_memory_tool.py +72 -0
  394. aiq/tool/nvidia_rag.py +95 -0
  395. aiq/tool/register.py +38 -0
  396. aiq/tool/retriever.py +89 -0
  397. aiq/tool/server_tools.py +66 -0
  398. aiq/utils/__init__.py +0 -0
  399. aiq/utils/data_models/__init__.py +0 -0
  400. aiq/utils/data_models/schema_validator.py +58 -0
  401. aiq/utils/debugging_utils.py +43 -0
  402. aiq/utils/dump_distro_mapping.py +32 -0
  403. aiq/utils/exception_handlers/__init__.py +0 -0
  404. aiq/utils/exception_handlers/automatic_retries.py +289 -0
  405. aiq/utils/exception_handlers/mcp.py +211 -0
  406. aiq/utils/exception_handlers/schemas.py +114 -0
  407. aiq/utils/io/__init__.py +0 -0
  408. aiq/utils/io/model_processing.py +28 -0
  409. aiq/utils/io/yaml_tools.py +119 -0
  410. aiq/utils/log_utils.py +37 -0
  411. aiq/utils/metadata_utils.py +74 -0
  412. aiq/utils/optional_imports.py +142 -0
  413. aiq/utils/producer_consumer_queue.py +178 -0
  414. aiq/utils/reactive/__init__.py +0 -0
  415. aiq/utils/reactive/base/__init__.py +0 -0
  416. aiq/utils/reactive/base/observable_base.py +65 -0
  417. aiq/utils/reactive/base/observer_base.py +55 -0
  418. aiq/utils/reactive/base/subject_base.py +79 -0
  419. aiq/utils/reactive/observable.py +59 -0
  420. aiq/utils/reactive/observer.py +76 -0
  421. aiq/utils/reactive/subject.py +131 -0
  422. aiq/utils/reactive/subscription.py +49 -0
  423. aiq/utils/settings/__init__.py +0 -0
  424. aiq/utils/settings/global_settings.py +197 -0
  425. aiq/utils/string_utils.py +38 -0
  426. aiq/utils/type_converter.py +290 -0
  427. aiq/utils/type_utils.py +484 -0
  428. aiq/utils/url_utils.py +27 -0
  429. nvidia_nat-1.2.0rc5.dist-info/METADATA +363 -0
  430. nvidia_nat-1.2.0rc5.dist-info/RECORD +435 -0
  431. nvidia_nat-1.2.0rc5.dist-info/WHEEL +5 -0
  432. nvidia_nat-1.2.0rc5.dist-info/entry_points.txt +20 -0
  433. nvidia_nat-1.2.0rc5.dist-info/licenses/LICENSE-3rd-party.txt +3686 -0
  434. nvidia_nat-1.2.0rc5.dist-info/licenses/LICENSE.md +201 -0
  435. nvidia_nat-1.2.0rc5.dist-info/top_level.txt +1 -0
@@ -0,0 +1,142 @@
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 importlib
17
+ import logging
18
+ from types import ModuleType
19
+
20
+ logger = logging.getLogger(__name__)
21
+
22
+
23
+ class OptionalImportError(Exception):
24
+ """Raised when an optional import fails."""
25
+
26
+ def __init__(self, module_name: str, additional_message: str = ""):
27
+ super().__init__(f"Optional dependency '{module_name}' is not installed. {additional_message}")
28
+
29
+
30
+ class TelemetryOptionalImportError(OptionalImportError):
31
+ """Raised when an optional import of telemetry dependencies fails."""
32
+
33
+ def __init__(self, module_name: str):
34
+ super().__init__(
35
+ module_name,
36
+ "But the configuration file contains tracing exporters. "
37
+ "If you want to use this feature, please install it with: uv pip install -e '.[telemetry]'",
38
+ )
39
+
40
+
41
+ def optional_import(module_name: str) -> ModuleType:
42
+ """Attempt to import a module, raising OptionalImportError if it fails."""
43
+ try:
44
+ return importlib.import_module(module_name)
45
+ except ImportError as e:
46
+ raise OptionalImportError(module_name) from e
47
+
48
+
49
+ def telemetry_optional_import(module_name: str) -> ModuleType:
50
+ """Attempt to import a module, raising TelemetryOptionalImportError if it fails."""
51
+ try:
52
+ return importlib.import_module(module_name)
53
+ except ImportError as e:
54
+ raise TelemetryOptionalImportError(module_name) from e
55
+
56
+
57
+ def try_import_opentelemetry() -> ModuleType:
58
+ """Get the opentelemetry module if available."""
59
+ return telemetry_optional_import("opentelemetry")
60
+
61
+
62
+ def try_import_phoenix() -> ModuleType:
63
+ """Get the phoenix module if available."""
64
+ return telemetry_optional_import("phoenix")
65
+
66
+
67
+ # Dummy OpenTelemetry classes for when the package is not available
68
+ class DummySpan:
69
+ """Dummy span class that does nothing when OpenTelemetry is not available."""
70
+
71
+ def __init__(self, *args, **kwargs):
72
+ pass
73
+
74
+ def end(self, *args, **kwargs):
75
+ pass
76
+
77
+ def set_attribute(self, *args, **kwargs):
78
+ pass
79
+
80
+
81
+ class DummyTracer:
82
+ """Dummy tracer class that returns dummy spans."""
83
+
84
+ def start_span(self, *args, **kwargs):
85
+ return DummySpan()
86
+
87
+
88
+ class DummyTracerProvider:
89
+ """Dummy tracer provider that returns dummy tracers."""
90
+
91
+ @staticmethod
92
+ def get_tracer(*args, **kwargs):
93
+ return DummyTracer()
94
+
95
+ @staticmethod
96
+ def add_span_processor(*args, **kwargs):
97
+ pass
98
+
99
+
100
+ class DummyTrace:
101
+ """Dummy trace module that returns dummy tracer providers."""
102
+
103
+ @staticmethod
104
+ def get_tracer_provider():
105
+ return DummyTracerProvider()
106
+
107
+ @staticmethod
108
+ def set_tracer_provider(*args, **kwargs):
109
+ pass
110
+
111
+ @staticmethod
112
+ def get_tracer(*args, **kwargs):
113
+ return DummyTracer()
114
+
115
+
116
+ class DummySpanExporter:
117
+ """Dummy span exporter that does nothing."""
118
+
119
+ @staticmethod
120
+ def export(*args, **kwargs):
121
+ pass
122
+
123
+ @staticmethod
124
+ def shutdown(*args, **kwargs):
125
+ pass
126
+
127
+
128
+ class DummyBatchSpanProcessor:
129
+ """Dummy implementation of BatchSpanProcessor for when OpenTelemetry is not available."""
130
+
131
+ def __init__(self, *args, **kwargs):
132
+ pass
133
+
134
+ @staticmethod
135
+ def shutdown(*args, **kwargs):
136
+ pass
137
+
138
+
139
+ # Dummy functions for when OpenTelemetry is not available
140
+ def dummy_set_span_in_context(*args, **kwargs) -> None:
141
+ """Dummy function that does nothing."""
142
+ return None
@@ -0,0 +1,178 @@
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 typing
18
+
19
+ _T = typing.TypeVar("_T")
20
+
21
+
22
+ class QueueClosed(Exception):
23
+ 'Exception raised when the queue is closed'
24
+ pass
25
+
26
+
27
+ class AsyncIOProducerConsumerQueue(asyncio.Queue, typing.Generic[_T]):
28
+ """
29
+ Custom queue.Queue implementation which supports closing and uses recursive locks
30
+ """
31
+
32
+ def __init__(self, maxsize=0) -> None:
33
+ super().__init__(maxsize=maxsize)
34
+
35
+ self._closed = asyncio.Event()
36
+ self._is_closed = False
37
+
38
+ async def __aiter__(self):
39
+ try:
40
+ while True:
41
+ yield await self.get()
42
+ except QueueClosed:
43
+ return
44
+
45
+ async def join(self):
46
+ """Block until all items in the queue have been gotten and processed.
47
+
48
+ The count of unfinished tasks goes up whenever an item is added to the
49
+ queue. The count goes down whenever a consumer calls task_done() to
50
+ indicate that the item was retrieved and all work on it is complete.
51
+ When the count of unfinished tasks drops to zero, join() unblocks.
52
+ """
53
+
54
+ # First wait for the closed flag to be set
55
+ await self._closed.wait()
56
+
57
+ if self._unfinished_tasks > 0:
58
+ await self._finished.wait()
59
+
60
+ async def put(self, item):
61
+ """Put an item into the queue.
62
+
63
+ Put an item into the queue. If the queue is full, wait until a free
64
+ slot is available before adding item.
65
+ """
66
+ while self.full() and not self._is_closed:
67
+ putter = self._get_loop().create_future()
68
+ self._putters.append(putter)
69
+ try:
70
+ await putter
71
+ except Exception:
72
+ putter.cancel() # Just in case putter is not done yet.
73
+ try:
74
+ # Clean self._putters from canceled putters.
75
+ self._putters.remove(putter)
76
+ except ValueError:
77
+ # The putter could be removed from self._putters by a
78
+ # previous get_nowait call.
79
+ pass
80
+ if not self.full() and not putter.cancelled():
81
+ # We were woken up by get_nowait(), but can't take
82
+ # the call. Wake up the next in line.
83
+ self._wakeup_next(self._putters)
84
+ raise
85
+
86
+ if (self._is_closed):
87
+ raise QueueClosed # @IgnoreException
88
+
89
+ return self.put_nowait(item)
90
+
91
+ async def get(self) -> _T:
92
+ """Remove and return an item from the queue.
93
+
94
+ If queue is empty, wait until an item is available.
95
+ """
96
+ while self.empty() and not self._is_closed:
97
+ getter = self._get_loop().create_future()
98
+ self._getters.append(getter)
99
+ try:
100
+ await getter
101
+ except Exception:
102
+ getter.cancel() # Just in case getter is not done yet.
103
+ try:
104
+ # Clean self._getters from canceled getters.
105
+ self._getters.remove(getter)
106
+ except ValueError:
107
+ # The getter could be removed from self._getters by a
108
+ # previous put_nowait call.
109
+ pass
110
+ if not self.empty() and not getter.cancelled():
111
+ # We were woken up by put_nowait(), but can't take
112
+ # the call. Wake up the next in line.
113
+ self._wakeup_next(self._getters)
114
+ raise
115
+
116
+ if (self.empty() and self._is_closed):
117
+ raise QueueClosed # @IgnoreException
118
+
119
+ return self.get_nowait()
120
+
121
+ def put_blocking(self, item: _T):
122
+ """
123
+ Synchronously block until the item can be put.
124
+ This method creates or uses an event loop internally to call the async put().
125
+ If the queue is closed, it raises QueueClosed.
126
+
127
+ NOTE: If you already have an event loop running in this same thread, calling
128
+ `run_until_complete` can cause conflicts or an error. Typically, you only
129
+ want to do this from a pure synchronous environment.
130
+ """
131
+
132
+ # If the queue is already closed, raise immediately
133
+ if self._is_closed:
134
+ raise QueueClosed("Queue is closed, cannot put more items.")
135
+
136
+ # Quick check: if there's space, just put_nowait() and exit
137
+ # (This covers the trivial case with no blocking)
138
+ if not self.full():
139
+ self.put_nowait(item)
140
+ return None
141
+
142
+ # If we do need to block, we run self.put(...) in an event loop
143
+ # We'll attempt to get the currently running loop if there is one,
144
+ # otherwise create a new one. If there's an existing loop, we might get
145
+ # an error if that loop is in the same thread. Adjust logic as needed.
146
+
147
+ try:
148
+ # If a loop is already running in this thread, get_running_loop() will succeed.
149
+ loop = asyncio.get_running_loop()
150
+ except RuntimeError:
151
+ # Means no running event loop in this thread -> create a new loop
152
+ loop = asyncio.new_event_loop()
153
+ try:
154
+ result = loop.run_until_complete(self.put(item))
155
+ finally:
156
+ loop.close()
157
+ return None
158
+
159
+ # If we got a running loop, but we aren't inside an async function,
160
+ # do a "blocking" wait by scheduling the put and waiting:
161
+ future = asyncio.run_coroutine_threadsafe(self.put(item), loop)
162
+ result = future.result() # blocks until done
163
+ return result
164
+
165
+ async def close(self):
166
+ """Close the queue."""
167
+ if (not self._is_closed):
168
+ self._is_closed = True
169
+
170
+ # Hit the flag
171
+ self._closed.set()
172
+
173
+ self._wakeup_next(self._putters)
174
+ self._wakeup_next(self._getters)
175
+
176
+ def is_closed(self) -> bool:
177
+ """Check if the queue is closed."""
178
+ return self._is_closed
File without changes
File without changes
@@ -0,0 +1,65 @@
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 typing
17
+ from abc import ABC
18
+ from abc import abstractmethod
19
+ from collections.abc import Callable
20
+ from typing import Generic
21
+ from typing import TypeVar
22
+
23
+ from aiq.utils.reactive.base.observer_base import ObserverBase
24
+ from aiq.utils.reactive.subscription import Subscription
25
+
26
+ # Covariant type param: An Observable producing type X can also produce
27
+ # a subtype of X.
28
+ _T_out_co = TypeVar("_T_out_co", covariant=True) # pylint: disable=invalid-name
29
+ _T = TypeVar("_T") # pylint: disable=invalid-name
30
+
31
+ OnNext = Callable[[_T], None]
32
+ OnError = Callable[[Exception], None]
33
+ OnComplete = Callable[[], None]
34
+
35
+
36
+ class ObservableBase(Generic[_T_out_co], ABC):
37
+ """
38
+ Abstract base class for an Observable that can be subscribed to.
39
+ Produces items of type _T_out for its subscribers.
40
+ """
41
+
42
+ @typing.overload
43
+ def subscribe(self, on_next: ObserverBase[_T_out_co]) -> Subscription:
44
+ ...
45
+
46
+ @typing.overload
47
+ def subscribe(self,
48
+ on_next: OnNext[_T_out_co] | None = None,
49
+ on_error: OnError | None = None,
50
+ on_complete: OnComplete | None = None) -> Subscription:
51
+ ...
52
+
53
+ @abstractmethod
54
+ def subscribe(self,
55
+ on_next: ObserverBase[_T_out_co] | OnNext[_T_out_co] | None = None,
56
+ on_error: OnError | None = None,
57
+ on_complete: OnComplete | None = None) -> Subscription:
58
+ """
59
+ Subscribes an Observer or callbacks to this Observable.
60
+
61
+ If an Observer is provided, it will be subscribed to this Observable.
62
+ If callbacks are provided, they will be wrapped into an Observer and
63
+ subscribed to this Observable.
64
+ """
65
+ pass
@@ -0,0 +1,55 @@
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 abc import ABC
17
+ from abc import abstractmethod
18
+ from typing import Generic
19
+ from typing import TypeVar
20
+
21
+ # Contravariant type param: An Observer that can accept type X can also
22
+ # accept any supertype of X.
23
+ _T_in_contra = TypeVar("_T_in_contra", contravariant=True) # pylint: disable=invalid-name
24
+
25
+
26
+ class ObserverBase(Generic[_T_in_contra], ABC):
27
+ """
28
+ Abstract base class for an Observer that can receive events of type _T_in.
29
+
30
+ Once on_error or on_complete is called, the observer is considered stopped.
31
+ """
32
+
33
+ @abstractmethod
34
+ def on_next(self, value: _T_in_contra) -> None:
35
+ """
36
+ Called when a new item is produced. If the observer is stopped,
37
+ this call should be ignored or raise an error.
38
+ """
39
+ pass
40
+
41
+ @abstractmethod
42
+ def on_error(self, exc: Exception) -> None:
43
+ """
44
+ Called when the producer signals an unrecoverable error.
45
+ After this call, the observer is stopped.
46
+ """
47
+ pass
48
+
49
+ @abstractmethod
50
+ def on_complete(self) -> None:
51
+ """
52
+ Called when the producer signals completion (no more items).
53
+ After this call, the observer is stopped.
54
+ """
55
+ pass
@@ -0,0 +1,79 @@
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 typing
17
+ from abc import abstractmethod
18
+ from collections.abc import Callable
19
+ from typing import TypeVar
20
+
21
+ from .observable_base import ObservableBase
22
+ from .observer_base import ObserverBase
23
+
24
+ if typing.TYPE_CHECKING:
25
+ from aiq.utils.reactive.subscription import Subscription
26
+
27
+ T = TypeVar("T")
28
+
29
+ OnNext = Callable[[T], None]
30
+ OnError = Callable[[Exception], None]
31
+ OnComplete = Callable[[], None]
32
+
33
+
34
+ class SubjectBase(ObserverBase[T], ObservableBase[T]):
35
+ """
36
+ Minimal interface we expect from the Subject for unsubscribing logic.
37
+ """
38
+
39
+ @abstractmethod
40
+ def _unsubscribe_observer(self, observer: object) -> None:
41
+ pass
42
+
43
+ @abstractmethod
44
+ def subscribe(self,
45
+ on_next: ObserverBase[T] | OnNext[T] | None = None,
46
+ on_error: OnError | None = None,
47
+ on_complete: OnComplete | None = None) -> "Subscription":
48
+ """
49
+ Subscribes an Observer or callbacks to this Observable.
50
+
51
+ If an Observer is provided, it will be subscribed to this Observable.
52
+ If callbacks are provided, they will be wrapped into an Observer and
53
+ subscribed to this Observable.
54
+ """
55
+ pass
56
+
57
+ @abstractmethod
58
+ def on_next(self, value: T) -> None:
59
+ """
60
+ Called when a new item is produced. If the observer is stopped,
61
+ this call should be ignored or raise an error.
62
+ """
63
+ pass
64
+
65
+ @abstractmethod
66
+ def on_error(self, exc: Exception) -> None:
67
+ """
68
+ Called when the producer signals an unrecoverable error.
69
+ After this call, the observer is stopped.
70
+ """
71
+ pass
72
+
73
+ @abstractmethod
74
+ def on_complete(self) -> None:
75
+ """
76
+ Called when the producer signals completion (no more items).
77
+ After this call, the observer is stopped.
78
+ """
79
+ pass
@@ -0,0 +1,59 @@
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 Callable
17
+ from typing import TypeVar
18
+
19
+ from aiq.utils.reactive.base.observable_base import ObservableBase
20
+ from aiq.utils.reactive.base.observer_base import ObserverBase
21
+ from aiq.utils.reactive.observer import Observer
22
+ from aiq.utils.reactive.subscription import Subscription
23
+ from aiq.utils.type_utils import override
24
+
25
+ # Covariant type param: An Observable producing type X can also produce
26
+ # a subtype of X.
27
+ _T_out_co = TypeVar("_T_out_co", covariant=True) # pylint: disable=invalid-name
28
+ _T = TypeVar("_T") # pylint: disable=invalid-name
29
+
30
+ OnNext = Callable[[_T], None]
31
+ OnError = Callable[[Exception], None]
32
+ OnComplete = Callable[[], None]
33
+
34
+
35
+ class Observable(ObservableBase[_T_out_co]):
36
+ """
37
+ Concrete base Observable that implements subscribe, deferring actual hooking
38
+ logic to _subscribe_core.
39
+ """
40
+
41
+ __slots__ = ()
42
+
43
+ def _subscribe_core(self, observer: ObserverBase) -> Subscription:
44
+ """
45
+ By default, does nothing. Subclasses should override this to
46
+ attach the observer to their emission logic.
47
+ """
48
+ raise NotImplementedError("Observable._subscribe_core must be implemented by subclasses")
49
+
50
+ @override
51
+ def subscribe(self,
52
+ on_next: ObserverBase[_T_out_co] | OnNext[_T_out_co] | None = None,
53
+ on_error: OnError | None = None,
54
+ on_complete: OnComplete | None = None) -> "Subscription":
55
+
56
+ if isinstance(on_next, ObserverBase):
57
+ return self._subscribe_core(on_next)
58
+
59
+ return self._subscribe_core(Observer(on_next, on_error, on_complete))
@@ -0,0 +1,76 @@
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
+ from collections.abc import Callable
18
+ from typing import TypeVar
19
+
20
+ from aiq.utils.reactive.base.observer_base import ObserverBase
21
+
22
+ logger = logging.getLogger(__name__)
23
+
24
+ # Contravariant type param: An Observer that can accept type X can also
25
+ # accept any supertype of X.
26
+ _T_in_contra = TypeVar("_T_in_contra", contravariant=True) # pylint: disable=invalid-name
27
+ _T = TypeVar("_T") # pylint: disable=invalid-name
28
+
29
+ OnNext = Callable[[_T], None]
30
+ OnError = Callable[[Exception], None]
31
+ OnComplete = Callable[[], None]
32
+
33
+
34
+ class Observer(ObserverBase[_T_in_contra]):
35
+ """
36
+ Concrete Observer that wraps user-provided callbacks into an ObserverBase.
37
+ """
38
+
39
+ def __init__(
40
+ self,
41
+ on_next: OnNext | None = None,
42
+ on_error: OnError | None = None,
43
+ on_complete: OnComplete | None = None,
44
+ ) -> None:
45
+ self._on_next = on_next
46
+ self._on_error = on_error
47
+ self._on_complete = on_complete
48
+ self._stopped = False
49
+
50
+ def on_next(self, value: _T) -> None:
51
+ if self._stopped:
52
+ return
53
+ if self._on_next is None:
54
+ return
55
+ try:
56
+ self._on_next(value)
57
+ except Exception as exc:
58
+ # If the callback itself raises, treat that as an error
59
+ self.on_error(exc)
60
+
61
+ def on_error(self, exc: Exception) -> None:
62
+ if not self._stopped:
63
+ if self._on_error:
64
+ try:
65
+ self._on_error(exc)
66
+ except Exception as e:
67
+ logger.exception("Error in on_error callback: %s", e, exc_info=True)
68
+
69
+ def on_complete(self) -> None:
70
+ if not self._stopped:
71
+ self._stopped = True
72
+ if self._on_complete:
73
+ try:
74
+ self._on_complete()
75
+ except Exception as e:
76
+ logger.exception("Error in on_complete callback: %s", e, exc_info=True)