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,449 @@
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 copy
18
+ import logging
19
+ import weakref
20
+ from abc import abstractmethod
21
+ from collections.abc import AsyncGenerator
22
+ from collections.abc import Callable
23
+ from contextlib import asynccontextmanager
24
+ from typing import Any
25
+ from typing import Generic
26
+ from typing import TypeVar
27
+ from typing import overload
28
+
29
+ from aiq.builder.context import AIQContextState
30
+ from aiq.data_models.intermediate_step import IntermediateStep
31
+ from aiq.observability.exporter.exporter import Exporter
32
+ from aiq.utils.reactive.subject import Subject
33
+ from aiq.utils.type_utils import override
34
+
35
+ logger = logging.getLogger(__name__)
36
+
37
+ IsolatedAttributeT = TypeVar('IsolatedAttributeT')
38
+
39
+
40
+ class IsolatedAttribute(Generic[IsolatedAttributeT]):
41
+ """Descriptor for copy-on-write isolation.
42
+
43
+ This descriptor uses Python's descriptor protocol to automatically manage
44
+ attribute isolation during object copying. It enables efficient concurrent
45
+ execution by sharing expensive resources while isolating mutable state.
46
+
47
+ Performance Note: This pattern shares expensive resources (HTTP clients,
48
+ auth headers) while isolating cheap mutable state (task sets, events).
49
+ Tasks are tracked for monitoring but don't block shutdown - they complete
50
+ asynchronously in the event loop. Critical for high-throughput concurrent execution.
51
+
52
+ Implementation Note: Uses Python descriptor protocol (__get__, __set__, __set_name__)
53
+ for automatic attribute isolation on object copying.
54
+
55
+ Example:
56
+ class MyExporter(BaseExporter):
57
+ # Expensive HTTP client shared across instances
58
+ _client = expensive_http_client
59
+
60
+ # Cheap mutable state isolated per instance
61
+ _tasks: IsolatedAttribute[set] = IsolatedAttribute(set)
62
+
63
+ exporter1 = MyExporter(endpoint="https://api.service.com")
64
+ exporter2 = exporter1.create_isolated_instance(context)
65
+ # exporter2 shares _client but has isolated _tasks tracking
66
+ """
67
+
68
+ def __init__(self, factory: Callable[[], IsolatedAttributeT]):
69
+ self.factory = factory
70
+ self.name: str | None = None
71
+ self._private_name: str
72
+
73
+ def __set_name__(self, owner, name):
74
+ self.name = name
75
+ self._private_name = f"__{name}_isolated"
76
+
77
+ @overload
78
+ def __get__(self, obj: None, objtype: type[Any] | None = None) -> "IsolatedAttribute[IsolatedAttributeT]":
79
+ ...
80
+
81
+ @overload
82
+ def __get__(self, obj: Any, objtype: type[Any] | None = None) -> IsolatedAttributeT:
83
+ ...
84
+
85
+ def __get__(self, obj, objtype=None):
86
+ if obj is None:
87
+ return self
88
+
89
+ if not hasattr(obj, self._private_name):
90
+ setattr(obj, self._private_name, self.factory())
91
+
92
+ return getattr(obj, self._private_name)
93
+
94
+ def __set__(self, obj, value: IsolatedAttributeT):
95
+ setattr(obj, self._private_name, value)
96
+
97
+ def reset_for_copy(self, obj):
98
+ """Reset the attribute for a copied object."""
99
+ if hasattr(obj, self._private_name):
100
+ delattr(obj, self._private_name)
101
+
102
+
103
+ class BaseExporter(Exporter):
104
+ """Abstract base class for event exporters with isolated copy support.
105
+
106
+ This class provides the foundation for creating event exporters that can handle
107
+ concurrent execution through copy-on-write isolation. It manages the lifecycle
108
+ of event subscriptions and provides hooks for processing events.
109
+
110
+ The class supports isolation for concurrent execution by automatically resetting
111
+ mutable state when creating isolated copies using descriptors.
112
+
113
+ Performance Design:
114
+ - Export tasks run asynchronously in the event loop background
115
+ - stop() method does not wait for background tasks to complete
116
+ - Tasks are tracked for monitoring but cleaned up automatically
117
+ - This keeps observability "off the hot path" for optimal performance
118
+
119
+ Args:
120
+ context_state (AIQContextState, optional): The context state to use for the exporter. Defaults to None.
121
+ """
122
+
123
+ # Class-level tracking for debugging and monitoring
124
+ _instance_count: int = 0
125
+ _active_instances: set[weakref.ref] = set()
126
+ _isolated_instances: set[weakref.ref] = set()
127
+
128
+ # Use descriptors for automatic isolation with proper generic typing
129
+ _tasks: IsolatedAttribute[set[asyncio.Task]] = IsolatedAttribute(set)
130
+ _ready_event: IsolatedAttribute[asyncio.Event] = IsolatedAttribute(asyncio.Event)
131
+ _shutdown_event: IsolatedAttribute[asyncio.Event] = IsolatedAttribute(asyncio.Event)
132
+
133
+ def __init__(self, context_state: AIQContextState | None = None):
134
+ """Initialize the BaseExporter."""
135
+ if context_state is None:
136
+ context_state = AIQContextState.get()
137
+
138
+ self._context_state = context_state
139
+ self._subscription = None
140
+ self._running = False
141
+ # Get the event loop (set to None if not available, will be set later)
142
+ self._loop = None
143
+ self._is_isolated_instance = False
144
+
145
+ # Track instance creation
146
+ BaseExporter._instance_count += 1
147
+ BaseExporter._active_instances.add(weakref.ref(self, self._cleanup_instance_tracking))
148
+
149
+ # Note: _tasks, _ready_event, _shutdown_event are descriptors
150
+
151
+ @classmethod
152
+ def _cleanup_instance_tracking(cls, ref):
153
+ """Cleanup callback for weakref when instance is garbage collected."""
154
+ cls._active_instances.discard(ref)
155
+ cls._isolated_instances.discard(ref)
156
+
157
+ @classmethod
158
+ def get_active_instance_count(cls) -> int:
159
+ """Get the number of active BaseExporter instances.
160
+
161
+ Returns:
162
+ int: Number of active instances (cleaned up automatically via weakref)
163
+ """
164
+ # Clean up dead references automatically via weakref callback
165
+ return len(cls._active_instances)
166
+
167
+ @classmethod
168
+ def get_isolated_instance_count(cls) -> int:
169
+ """Get the number of active isolated BaseExporter instances.
170
+
171
+ Returns:
172
+ int: Number of active isolated instances
173
+ """
174
+ return len(cls._isolated_instances)
175
+
176
+ @classmethod
177
+ def log_instance_stats(cls) -> None:
178
+ """Log current instance statistics for debugging."""
179
+ total = cls.get_active_instance_count()
180
+ isolated = cls.get_isolated_instance_count()
181
+ original = total - isolated
182
+
183
+ logger.info("BaseExporter instances - Total: %d, Original: %d, Isolated: %d", total, original, isolated)
184
+
185
+ if isolated > 50: # Warn if we have many isolated instances
186
+ warning_msg = (f"High number of isolated BaseExporter instances ({isolated}). "
187
+ "Check for potential memory leaks.")
188
+ logger.warning(warning_msg)
189
+
190
+ def __del__(self):
191
+ """Destructor with memory leak warnings.
192
+
193
+ Warns if the exporter is being garbage collected while still running,
194
+ which indicates stop() was never called. Task tracking is used for
195
+ diagnostics but stop() doesn't wait for tasks to complete.
196
+
197
+ This method is defensive against partial initialization - if the object
198
+ failed to initialize completely, some attributes may not exist.
199
+ """
200
+ try:
201
+ # Check if object was fully initialized before checking for active resources
202
+ is_running = getattr(self, '_running', False)
203
+ has_tasks = hasattr(self, '__tasks_isolated') and bool(getattr(self, '_tasks', None))
204
+
205
+ if is_running or has_tasks:
206
+ # Safely get name and task count
207
+ try:
208
+ name = self.name
209
+ except (AttributeError, TypeError):
210
+ # Fallback if name property fails due to missing attributes
211
+ name = f"{self.__class__.__name__} (partially initialized)"
212
+
213
+ task_count = len(self._tasks) if has_tasks else 0
214
+
215
+ logger.warning(
216
+ "%s: Exporter being garbage collected with active resources. "
217
+ "Running: %s, Tasks: %s. "
218
+ "Call stop() explicitly to avoid memory leaks.",
219
+ name,
220
+ is_running,
221
+ task_count)
222
+
223
+ except Exception as e:
224
+ # Last resort: log that cleanup had issues but don't raise
225
+ # This prevents exceptions during garbage collection
226
+ try:
227
+ class_name = self.__class__.__name__
228
+ logger.debug("Exception during %s cleanup: %s", class_name, e)
229
+ except Exception:
230
+ # If even logging fails, silently ignore to prevent GC issues
231
+ pass
232
+
233
+ @property
234
+ def name(self) -> str:
235
+ """Get the name of the exporter.
236
+
237
+ Returns:
238
+ str: The unique name of the exporter.
239
+ """
240
+ try:
241
+ suffix = " (isolated)" if getattr(self, '_is_isolated_instance', False) else ""
242
+ return f"{self.__class__.__name__}{suffix}"
243
+ except AttributeError:
244
+ # Fallback for partially initialized objects
245
+ return f"{self.__class__.__name__} (partial)"
246
+
247
+ @property
248
+ def is_isolated_instance(self) -> bool:
249
+ """Check if this is an isolated instance.
250
+
251
+ Returns:
252
+ bool: True if this is an isolated instance, False otherwise
253
+ """
254
+ return self._is_isolated_instance
255
+
256
+ @abstractmethod
257
+ def export(self, event: IntermediateStep) -> None:
258
+ """This method is called on each event from the event stream to initiate the trace export.
259
+
260
+ This is the base implementation that can be overridden by subclasses.
261
+ By default, it does nothing - subclasses should implement their specific export logic.
262
+
263
+ Args:
264
+ event (IntermediateStep): The event to be exported.
265
+ """
266
+ pass
267
+
268
+ @override
269
+ def on_error(self, exc: Exception) -> None:
270
+ """Handle an error in the event subscription.
271
+
272
+ Args:
273
+ exc (Exception): The error to handle.
274
+ """
275
+ logger.error("Error in event subscription: %s", exc, exc_info=True)
276
+
277
+ @override
278
+ def on_complete(self) -> None:
279
+ """Handle the completion of the event stream.
280
+
281
+ This method is called when the event stream is complete.
282
+ """
283
+ logger.info("Event stream completed. No more events will arrive.")
284
+
285
+ def _start(self) -> Subject | None:
286
+ """Start the exporter.
287
+
288
+ Returns:
289
+ Subject | None: The subject to subscribe to.
290
+ """
291
+ subject = self._context_state.event_stream.get()
292
+ if subject is None:
293
+ return None
294
+
295
+ if not hasattr(subject, 'subscribe'):
296
+ logger.error("Event stream subject does not support subscription")
297
+ return None
298
+
299
+ def on_next_wrapper(event: IntermediateStep) -> None:
300
+ self.export(event)
301
+
302
+ self._subscription = subject.subscribe(
303
+ on_next=on_next_wrapper,
304
+ on_error=self.on_error,
305
+ on_complete=self.on_complete,
306
+ )
307
+
308
+ self._running = True
309
+ self._ready_event.set()
310
+ return subject
311
+
312
+ async def _pre_start(self):
313
+ """Called before the exporter starts."""
314
+ pass
315
+
316
+ @override
317
+ @asynccontextmanager
318
+ async def start(self) -> AsyncGenerator[None]:
319
+ """Start the exporter and yield control to the caller."""
320
+ try:
321
+ await self._pre_start()
322
+
323
+ if self._running:
324
+ logger.debug("Listener already running.")
325
+ yield
326
+ return
327
+
328
+ subject = self._start()
329
+ if subject is None:
330
+ logger.warning("No event stream available.")
331
+ yield
332
+ return
333
+
334
+ yield # let the caller do their workflow
335
+
336
+ finally:
337
+ await self.stop()
338
+
339
+ async def _cleanup(self):
340
+ """Clean up any resources."""
341
+ pass
342
+
343
+ async def _cancel_tasks(self):
344
+ """Cancel all scheduled tasks.
345
+
346
+ Note: This method is NOT called during normal stop() operation for performance.
347
+ It's available for special cases where explicit task completion is needed.
348
+ """
349
+ tasks_to_cancel = set(self._tasks)
350
+ for task in tasks_to_cancel:
351
+ if not task.done():
352
+ task.cancel()
353
+ try:
354
+ await task
355
+ except asyncio.CancelledError:
356
+ pass
357
+ except Exception as e:
358
+ logger.warning("Error while canceling task %s: %s", task.get_name(), e)
359
+
360
+ async def _wait_for_tasks(self, timeout: float = 5.0):
361
+ """Wait for all tracked tasks to complete with a timeout.
362
+
363
+ Note: This method is NOT called during normal stop() operation for performance.
364
+ It's available for special cases where explicit task completion is needed.
365
+
366
+ Args:
367
+ timeout (float, optional): The timeout in seconds. Defaults to 5.0.
368
+ """
369
+ if not self._tasks:
370
+ return
371
+
372
+ try:
373
+ # Wait for all tasks to complete with a timeout
374
+ await asyncio.wait_for(asyncio.gather(*self._tasks, return_exceptions=True), timeout=timeout)
375
+ except asyncio.TimeoutError:
376
+ logger.warning("%s: Some tasks did not complete within %s seconds", self.name, timeout)
377
+ except Exception as e:
378
+ logger.error("%s: Error while waiting for tasks: %s", self.name, e, exc_info=True)
379
+
380
+ @override
381
+ async def stop(self):
382
+ """Stop the exporter immediately without waiting for background tasks.
383
+
384
+ This method performs fast shutdown by:
385
+ 1. Setting running=False to prevent new export tasks
386
+ 2. Signaling shutdown to waiting code
387
+ 3. Cleaning up subscriptions and resources
388
+ 4. Clearing task tracking (tasks continue in event loop)
389
+
390
+ Performance: Does not block waiting for background export tasks to complete.
391
+ Background tasks will finish asynchronously and clean themselves up.
392
+
393
+ Note: This method is called when the exporter is no longer needed.
394
+ """
395
+ if not self._running:
396
+ return
397
+
398
+ self._running = False
399
+ self._shutdown_event.set()
400
+
401
+ await self._cleanup()
402
+
403
+ if self._subscription:
404
+ self._subscription.unsubscribe()
405
+ self._subscription = None
406
+
407
+ self._tasks.clear()
408
+
409
+ async def wait_ready(self):
410
+ """Wait for the exporter to be ready.
411
+
412
+ This method is called when the exporter is ready to export events.
413
+ """
414
+ await self._ready_event.wait()
415
+
416
+ def create_isolated_instance(self, context_state: AIQContextState) -> "BaseExporter":
417
+ """Create an isolated copy with automatic descriptor-based state reset.
418
+
419
+ This method creates a shallow copy that shares expensive resources
420
+ (HTTP clients, auth headers) while isolating mutable state through
421
+ the IsolatedAttribute descriptor pattern.
422
+
423
+ Args:
424
+ context_state: The isolated context state for the new instance
425
+
426
+ Returns:
427
+ BaseExporter: Isolated instance sharing expensive resources
428
+ """
429
+ # Create shallow copy
430
+ isolated_instance = copy.copy(self)
431
+
432
+ # Reset context state
433
+ isolated_instance._context_state = context_state
434
+
435
+ # Mark as isolated instance and track it
436
+ isolated_instance._is_isolated_instance = True
437
+ BaseExporter._isolated_instances.add(weakref.ref(isolated_instance, self._cleanup_instance_tracking))
438
+
439
+ # Reset IsolatedAttribute descriptors automatically
440
+ for attr_name in dir(type(self)):
441
+ attr_value = getattr(type(self), attr_name, None)
442
+ if isinstance(attr_value, IsolatedAttribute):
443
+ attr_value.reset_for_copy(isolated_instance)
444
+
445
+ # Reset basic attributes that aren't descriptors but need isolation
446
+ isolated_instance._subscription = None
447
+ isolated_instance._running = False
448
+
449
+ return isolated_instance
@@ -0,0 +1,78 @@
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 abc import ABC
18
+ from abc import abstractmethod
19
+ from collections.abc import AsyncGenerator
20
+
21
+ from aiq.data_models.intermediate_step import IntermediateStep
22
+
23
+ logger = logging.getLogger(__name__)
24
+
25
+
26
+ class Exporter(ABC):
27
+
28
+ @abstractmethod
29
+ async def start(self) -> AsyncGenerator[None]:
30
+ """Subscribes to event stream and starts the exporter.
31
+
32
+ This is an async context manager that should be used with 'async with'.
33
+ The exporter is automatically stopped when exiting the context.
34
+
35
+ Usage::
36
+
37
+ .. code-block:: python
38
+
39
+ async with exporter.start():
40
+ # Exporter is now running and subscribed to events
41
+ # Your workflow code here
42
+ pass
43
+
44
+ Note:
45
+ Implementations should use the @asynccontextmanager decorator.
46
+ """
47
+ pass
48
+
49
+ @abstractmethod
50
+ async def stop(self) -> None:
51
+ """Unsubscribes to the event stream and stops the exporter."""
52
+ pass
53
+
54
+ @abstractmethod
55
+ def export(self, event: IntermediateStep) -> None:
56
+ """This method is called on each event from the event stream to initiate the trace export.
57
+
58
+ Args:
59
+ event (IntermediateStep): The event to be exported.
60
+ """
61
+ pass
62
+
63
+ @abstractmethod
64
+ def on_error(self, exc: Exception) -> None:
65
+ """Handle an error in the event subscription.
66
+
67
+ Args:
68
+ exc (Exception): The error to handle.
69
+ """
70
+ pass
71
+
72
+ @abstractmethod
73
+ def on_complete(self) -> None:
74
+ """Handle the completion of the event stream.
75
+
76
+ This method is called when the event stream is complete.
77
+ """
78
+ pass
@@ -0,0 +1,33 @@
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.context import AIQContextState
19
+ from aiq.data_models.intermediate_step import IntermediateStep
20
+ from aiq.observability.exporter.raw_exporter import RawExporter
21
+ from aiq.observability.mixin.file_mixin import FileExportMixin
22
+ from aiq.observability.processor.intermediate_step_serializer import IntermediateStepSerializer
23
+
24
+ logger = logging.getLogger(__name__)
25
+
26
+
27
+ class FileExporter(FileExportMixin, RawExporter[IntermediateStep, str]): # pylint: disable=R0901
28
+ """A File exporter that exports telemetry traces to a local file."""
29
+
30
+ def __init__(self, context_state: AIQContextState | None = None, **file_kwargs):
31
+ super().__init__(context_state=context_state, **file_kwargs)
32
+ self._processor = IntermediateStepSerializer()
33
+ self.add_processor(self._processor)