nvidia-nat 1.1.0a20251020__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 (480) hide show
  1. aiq/__init__.py +66 -0
  2. nat/agent/__init__.py +0 -0
  3. nat/agent/base.py +265 -0
  4. nat/agent/dual_node.py +72 -0
  5. nat/agent/prompt_optimizer/__init__.py +0 -0
  6. nat/agent/prompt_optimizer/prompt.py +68 -0
  7. nat/agent/prompt_optimizer/register.py +149 -0
  8. nat/agent/react_agent/__init__.py +0 -0
  9. nat/agent/react_agent/agent.py +394 -0
  10. nat/agent/react_agent/output_parser.py +104 -0
  11. nat/agent/react_agent/prompt.py +44 -0
  12. nat/agent/react_agent/register.py +168 -0
  13. nat/agent/reasoning_agent/__init__.py +0 -0
  14. nat/agent/reasoning_agent/reasoning_agent.py +227 -0
  15. nat/agent/register.py +23 -0
  16. nat/agent/rewoo_agent/__init__.py +0 -0
  17. nat/agent/rewoo_agent/agent.py +593 -0
  18. nat/agent/rewoo_agent/prompt.py +107 -0
  19. nat/agent/rewoo_agent/register.py +175 -0
  20. nat/agent/tool_calling_agent/__init__.py +0 -0
  21. nat/agent/tool_calling_agent/agent.py +246 -0
  22. nat/agent/tool_calling_agent/register.py +129 -0
  23. nat/authentication/__init__.py +14 -0
  24. nat/authentication/api_key/__init__.py +14 -0
  25. nat/authentication/api_key/api_key_auth_provider.py +96 -0
  26. nat/authentication/api_key/api_key_auth_provider_config.py +124 -0
  27. nat/authentication/api_key/register.py +26 -0
  28. nat/authentication/credential_validator/__init__.py +14 -0
  29. nat/authentication/credential_validator/bearer_token_validator.py +557 -0
  30. nat/authentication/exceptions/__init__.py +14 -0
  31. nat/authentication/exceptions/api_key_exceptions.py +38 -0
  32. nat/authentication/http_basic_auth/__init__.py +0 -0
  33. nat/authentication/http_basic_auth/http_basic_auth_provider.py +81 -0
  34. nat/authentication/http_basic_auth/register.py +30 -0
  35. nat/authentication/interfaces.py +96 -0
  36. nat/authentication/oauth2/__init__.py +14 -0
  37. nat/authentication/oauth2/oauth2_auth_code_flow_provider.py +140 -0
  38. nat/authentication/oauth2/oauth2_auth_code_flow_provider_config.py +39 -0
  39. nat/authentication/oauth2/oauth2_resource_server_config.py +124 -0
  40. nat/authentication/oauth2/register.py +25 -0
  41. nat/authentication/register.py +20 -0
  42. nat/builder/__init__.py +0 -0
  43. nat/builder/builder.py +317 -0
  44. nat/builder/component_utils.py +320 -0
  45. nat/builder/context.py +321 -0
  46. nat/builder/embedder.py +24 -0
  47. nat/builder/eval_builder.py +166 -0
  48. nat/builder/evaluator.py +29 -0
  49. nat/builder/framework_enum.py +25 -0
  50. nat/builder/front_end.py +73 -0
  51. nat/builder/function.py +714 -0
  52. nat/builder/function_base.py +380 -0
  53. nat/builder/function_info.py +625 -0
  54. nat/builder/intermediate_step_manager.py +206 -0
  55. nat/builder/llm.py +25 -0
  56. nat/builder/retriever.py +25 -0
  57. nat/builder/user_interaction_manager.py +78 -0
  58. nat/builder/workflow.py +160 -0
  59. nat/builder/workflow_builder.py +1365 -0
  60. nat/cli/__init__.py +14 -0
  61. nat/cli/cli_utils/__init__.py +0 -0
  62. nat/cli/cli_utils/config_override.py +231 -0
  63. nat/cli/cli_utils/validation.py +37 -0
  64. nat/cli/commands/__init__.py +0 -0
  65. nat/cli/commands/configure/__init__.py +0 -0
  66. nat/cli/commands/configure/channel/__init__.py +0 -0
  67. nat/cli/commands/configure/channel/add.py +28 -0
  68. nat/cli/commands/configure/channel/channel.py +34 -0
  69. nat/cli/commands/configure/channel/remove.py +30 -0
  70. nat/cli/commands/configure/channel/update.py +30 -0
  71. nat/cli/commands/configure/configure.py +33 -0
  72. nat/cli/commands/evaluate.py +139 -0
  73. nat/cli/commands/info/__init__.py +14 -0
  74. nat/cli/commands/info/info.py +47 -0
  75. nat/cli/commands/info/list_channels.py +32 -0
  76. nat/cli/commands/info/list_components.py +128 -0
  77. nat/cli/commands/mcp/__init__.py +14 -0
  78. nat/cli/commands/mcp/mcp.py +986 -0
  79. nat/cli/commands/object_store/__init__.py +14 -0
  80. nat/cli/commands/object_store/object_store.py +227 -0
  81. nat/cli/commands/optimize.py +90 -0
  82. nat/cli/commands/registry/__init__.py +14 -0
  83. nat/cli/commands/registry/publish.py +88 -0
  84. nat/cli/commands/registry/pull.py +118 -0
  85. nat/cli/commands/registry/registry.py +36 -0
  86. nat/cli/commands/registry/remove.py +108 -0
  87. nat/cli/commands/registry/search.py +153 -0
  88. nat/cli/commands/sizing/__init__.py +14 -0
  89. nat/cli/commands/sizing/calc.py +297 -0
  90. nat/cli/commands/sizing/sizing.py +27 -0
  91. nat/cli/commands/start.py +257 -0
  92. nat/cli/commands/uninstall.py +81 -0
  93. nat/cli/commands/validate.py +47 -0
  94. nat/cli/commands/workflow/__init__.py +14 -0
  95. nat/cli/commands/workflow/templates/__init__.py.j2 +0 -0
  96. nat/cli/commands/workflow/templates/config.yml.j2 +17 -0
  97. nat/cli/commands/workflow/templates/pyproject.toml.j2 +25 -0
  98. nat/cli/commands/workflow/templates/register.py.j2 +4 -0
  99. nat/cli/commands/workflow/templates/workflow.py.j2 +50 -0
  100. nat/cli/commands/workflow/workflow.py +37 -0
  101. nat/cli/commands/workflow/workflow_commands.py +403 -0
  102. nat/cli/entrypoint.py +141 -0
  103. nat/cli/main.py +60 -0
  104. nat/cli/register_workflow.py +522 -0
  105. nat/cli/type_registry.py +1069 -0
  106. nat/control_flow/__init__.py +0 -0
  107. nat/control_flow/register.py +20 -0
  108. nat/control_flow/router_agent/__init__.py +0 -0
  109. nat/control_flow/router_agent/agent.py +329 -0
  110. nat/control_flow/router_agent/prompt.py +48 -0
  111. nat/control_flow/router_agent/register.py +91 -0
  112. nat/control_flow/sequential_executor.py +166 -0
  113. nat/data_models/__init__.py +14 -0
  114. nat/data_models/agent.py +34 -0
  115. nat/data_models/api_server.py +843 -0
  116. nat/data_models/authentication.py +245 -0
  117. nat/data_models/common.py +171 -0
  118. nat/data_models/component.py +60 -0
  119. nat/data_models/component_ref.py +179 -0
  120. nat/data_models/config.py +434 -0
  121. nat/data_models/dataset_handler.py +169 -0
  122. nat/data_models/discovery_metadata.py +305 -0
  123. nat/data_models/embedder.py +27 -0
  124. nat/data_models/evaluate.py +130 -0
  125. nat/data_models/evaluator.py +26 -0
  126. nat/data_models/front_end.py +26 -0
  127. nat/data_models/function.py +64 -0
  128. nat/data_models/function_dependencies.py +80 -0
  129. nat/data_models/gated_field_mixin.py +242 -0
  130. nat/data_models/interactive.py +246 -0
  131. nat/data_models/intermediate_step.py +302 -0
  132. nat/data_models/invocation_node.py +38 -0
  133. nat/data_models/llm.py +27 -0
  134. nat/data_models/logging.py +26 -0
  135. nat/data_models/memory.py +27 -0
  136. nat/data_models/object_store.py +44 -0
  137. nat/data_models/optimizable.py +119 -0
  138. nat/data_models/optimizer.py +149 -0
  139. nat/data_models/profiler.py +54 -0
  140. nat/data_models/registry_handler.py +26 -0
  141. nat/data_models/retriever.py +30 -0
  142. nat/data_models/retry_mixin.py +35 -0
  143. nat/data_models/span.py +228 -0
  144. nat/data_models/step_adaptor.py +64 -0
  145. nat/data_models/streaming.py +33 -0
  146. nat/data_models/swe_bench_model.py +54 -0
  147. nat/data_models/telemetry_exporter.py +26 -0
  148. nat/data_models/temperature_mixin.py +44 -0
  149. nat/data_models/thinking_mixin.py +86 -0
  150. nat/data_models/top_p_mixin.py +44 -0
  151. nat/data_models/ttc_strategy.py +30 -0
  152. nat/embedder/__init__.py +0 -0
  153. nat/embedder/azure_openai_embedder.py +46 -0
  154. nat/embedder/nim_embedder.py +59 -0
  155. nat/embedder/openai_embedder.py +42 -0
  156. nat/embedder/register.py +22 -0
  157. nat/eval/__init__.py +14 -0
  158. nat/eval/config.py +62 -0
  159. nat/eval/dataset_handler/__init__.py +0 -0
  160. nat/eval/dataset_handler/dataset_downloader.py +106 -0
  161. nat/eval/dataset_handler/dataset_filter.py +52 -0
  162. nat/eval/dataset_handler/dataset_handler.py +431 -0
  163. nat/eval/evaluate.py +565 -0
  164. nat/eval/evaluator/__init__.py +14 -0
  165. nat/eval/evaluator/base_evaluator.py +77 -0
  166. nat/eval/evaluator/evaluator_model.py +58 -0
  167. nat/eval/intermediate_step_adapter.py +99 -0
  168. nat/eval/rag_evaluator/__init__.py +0 -0
  169. nat/eval/rag_evaluator/evaluate.py +178 -0
  170. nat/eval/rag_evaluator/register.py +143 -0
  171. nat/eval/register.py +26 -0
  172. nat/eval/remote_workflow.py +133 -0
  173. nat/eval/runners/__init__.py +14 -0
  174. nat/eval/runners/config.py +39 -0
  175. nat/eval/runners/multi_eval_runner.py +54 -0
  176. nat/eval/runtime_evaluator/__init__.py +14 -0
  177. nat/eval/runtime_evaluator/evaluate.py +123 -0
  178. nat/eval/runtime_evaluator/register.py +100 -0
  179. nat/eval/runtime_event_subscriber.py +52 -0
  180. nat/eval/swe_bench_evaluator/__init__.py +0 -0
  181. nat/eval/swe_bench_evaluator/evaluate.py +215 -0
  182. nat/eval/swe_bench_evaluator/register.py +36 -0
  183. nat/eval/trajectory_evaluator/__init__.py +0 -0
  184. nat/eval/trajectory_evaluator/evaluate.py +75 -0
  185. nat/eval/trajectory_evaluator/register.py +40 -0
  186. nat/eval/tunable_rag_evaluator/__init__.py +0 -0
  187. nat/eval/tunable_rag_evaluator/evaluate.py +242 -0
  188. nat/eval/tunable_rag_evaluator/register.py +52 -0
  189. nat/eval/usage_stats.py +41 -0
  190. nat/eval/utils/__init__.py +0 -0
  191. nat/eval/utils/eval_trace_ctx.py +89 -0
  192. nat/eval/utils/output_uploader.py +140 -0
  193. nat/eval/utils/tqdm_position_registry.py +40 -0
  194. nat/eval/utils/weave_eval.py +193 -0
  195. nat/experimental/__init__.py +0 -0
  196. nat/experimental/decorators/__init__.py +0 -0
  197. nat/experimental/decorators/experimental_warning_decorator.py +154 -0
  198. nat/experimental/test_time_compute/__init__.py +0 -0
  199. nat/experimental/test_time_compute/editing/__init__.py +0 -0
  200. nat/experimental/test_time_compute/editing/iterative_plan_refinement_editor.py +147 -0
  201. nat/experimental/test_time_compute/editing/llm_as_a_judge_editor.py +204 -0
  202. nat/experimental/test_time_compute/editing/motivation_aware_summarization.py +107 -0
  203. nat/experimental/test_time_compute/functions/__init__.py +0 -0
  204. nat/experimental/test_time_compute/functions/execute_score_select_function.py +105 -0
  205. nat/experimental/test_time_compute/functions/plan_select_execute_function.py +228 -0
  206. nat/experimental/test_time_compute/functions/ttc_tool_orchestration_function.py +205 -0
  207. nat/experimental/test_time_compute/functions/ttc_tool_wrapper_function.py +146 -0
  208. nat/experimental/test_time_compute/models/__init__.py +0 -0
  209. nat/experimental/test_time_compute/models/editor_config.py +132 -0
  210. nat/experimental/test_time_compute/models/scoring_config.py +112 -0
  211. nat/experimental/test_time_compute/models/search_config.py +120 -0
  212. nat/experimental/test_time_compute/models/selection_config.py +154 -0
  213. nat/experimental/test_time_compute/models/stage_enums.py +43 -0
  214. nat/experimental/test_time_compute/models/strategy_base.py +67 -0
  215. nat/experimental/test_time_compute/models/tool_use_config.py +41 -0
  216. nat/experimental/test_time_compute/models/ttc_item.py +48 -0
  217. nat/experimental/test_time_compute/register.py +35 -0
  218. nat/experimental/test_time_compute/scoring/__init__.py +0 -0
  219. nat/experimental/test_time_compute/scoring/llm_based_agent_scorer.py +168 -0
  220. nat/experimental/test_time_compute/scoring/llm_based_plan_scorer.py +168 -0
  221. nat/experimental/test_time_compute/scoring/motivation_aware_scorer.py +111 -0
  222. nat/experimental/test_time_compute/search/__init__.py +0 -0
  223. nat/experimental/test_time_compute/search/multi_llm_planner.py +128 -0
  224. nat/experimental/test_time_compute/search/multi_query_retrieval_search.py +122 -0
  225. nat/experimental/test_time_compute/search/single_shot_multi_plan_planner.py +128 -0
  226. nat/experimental/test_time_compute/selection/__init__.py +0 -0
  227. nat/experimental/test_time_compute/selection/best_of_n_selector.py +63 -0
  228. nat/experimental/test_time_compute/selection/llm_based_agent_output_selector.py +131 -0
  229. nat/experimental/test_time_compute/selection/llm_based_output_merging_selector.py +157 -0
  230. nat/experimental/test_time_compute/selection/llm_based_plan_selector.py +128 -0
  231. nat/experimental/test_time_compute/selection/threshold_selector.py +58 -0
  232. nat/front_ends/__init__.py +14 -0
  233. nat/front_ends/console/__init__.py +14 -0
  234. nat/front_ends/console/authentication_flow_handler.py +285 -0
  235. nat/front_ends/console/console_front_end_config.py +32 -0
  236. nat/front_ends/console/console_front_end_plugin.py +108 -0
  237. nat/front_ends/console/register.py +25 -0
  238. nat/front_ends/cron/__init__.py +14 -0
  239. nat/front_ends/fastapi/__init__.py +14 -0
  240. nat/front_ends/fastapi/auth_flow_handlers/__init__.py +0 -0
  241. nat/front_ends/fastapi/auth_flow_handlers/http_flow_handler.py +27 -0
  242. nat/front_ends/fastapi/auth_flow_handlers/websocket_flow_handler.py +142 -0
  243. nat/front_ends/fastapi/dask_client_mixin.py +65 -0
  244. nat/front_ends/fastapi/fastapi_front_end_config.py +272 -0
  245. nat/front_ends/fastapi/fastapi_front_end_controller.py +68 -0
  246. nat/front_ends/fastapi/fastapi_front_end_plugin.py +247 -0
  247. nat/front_ends/fastapi/fastapi_front_end_plugin_worker.py +1257 -0
  248. nat/front_ends/fastapi/html_snippets/__init__.py +14 -0
  249. nat/front_ends/fastapi/html_snippets/auth_code_grant_success.py +35 -0
  250. nat/front_ends/fastapi/intermediate_steps_subscriber.py +80 -0
  251. nat/front_ends/fastapi/job_store.py +602 -0
  252. nat/front_ends/fastapi/main.py +64 -0
  253. nat/front_ends/fastapi/message_handler.py +344 -0
  254. nat/front_ends/fastapi/message_validator.py +351 -0
  255. nat/front_ends/fastapi/register.py +25 -0
  256. nat/front_ends/fastapi/response_helpers.py +195 -0
  257. nat/front_ends/fastapi/step_adaptor.py +319 -0
  258. nat/front_ends/fastapi/utils.py +57 -0
  259. nat/front_ends/mcp/__init__.py +14 -0
  260. nat/front_ends/mcp/introspection_token_verifier.py +73 -0
  261. nat/front_ends/mcp/mcp_front_end_config.py +90 -0
  262. nat/front_ends/mcp/mcp_front_end_plugin.py +113 -0
  263. nat/front_ends/mcp/mcp_front_end_plugin_worker.py +268 -0
  264. nat/front_ends/mcp/memory_profiler.py +320 -0
  265. nat/front_ends/mcp/register.py +27 -0
  266. nat/front_ends/mcp/tool_converter.py +290 -0
  267. nat/front_ends/register.py +21 -0
  268. nat/front_ends/simple_base/__init__.py +14 -0
  269. nat/front_ends/simple_base/simple_front_end_plugin_base.py +56 -0
  270. nat/llm/__init__.py +0 -0
  271. nat/llm/aws_bedrock_llm.py +69 -0
  272. nat/llm/azure_openai_llm.py +57 -0
  273. nat/llm/litellm_llm.py +69 -0
  274. nat/llm/nim_llm.py +58 -0
  275. nat/llm/openai_llm.py +54 -0
  276. nat/llm/register.py +27 -0
  277. nat/llm/utils/__init__.py +14 -0
  278. nat/llm/utils/env_config_value.py +93 -0
  279. nat/llm/utils/error.py +17 -0
  280. nat/llm/utils/thinking.py +215 -0
  281. nat/memory/__init__.py +20 -0
  282. nat/memory/interfaces.py +183 -0
  283. nat/memory/models.py +112 -0
  284. nat/meta/pypi.md +58 -0
  285. nat/object_store/__init__.py +20 -0
  286. nat/object_store/in_memory_object_store.py +76 -0
  287. nat/object_store/interfaces.py +84 -0
  288. nat/object_store/models.py +38 -0
  289. nat/object_store/register.py +19 -0
  290. nat/observability/__init__.py +14 -0
  291. nat/observability/exporter/__init__.py +14 -0
  292. nat/observability/exporter/base_exporter.py +449 -0
  293. nat/observability/exporter/exporter.py +78 -0
  294. nat/observability/exporter/file_exporter.py +33 -0
  295. nat/observability/exporter/processing_exporter.py +550 -0
  296. nat/observability/exporter/raw_exporter.py +52 -0
  297. nat/observability/exporter/span_exporter.py +308 -0
  298. nat/observability/exporter_manager.py +335 -0
  299. nat/observability/mixin/__init__.py +14 -0
  300. nat/observability/mixin/batch_config_mixin.py +26 -0
  301. nat/observability/mixin/collector_config_mixin.py +23 -0
  302. nat/observability/mixin/file_mixin.py +288 -0
  303. nat/observability/mixin/file_mode.py +23 -0
  304. nat/observability/mixin/redaction_config_mixin.py +42 -0
  305. nat/observability/mixin/resource_conflict_mixin.py +134 -0
  306. nat/observability/mixin/serialize_mixin.py +61 -0
  307. nat/observability/mixin/tagging_config_mixin.py +62 -0
  308. nat/observability/mixin/type_introspection_mixin.py +496 -0
  309. nat/observability/processor/__init__.py +14 -0
  310. nat/observability/processor/batching_processor.py +308 -0
  311. nat/observability/processor/callback_processor.py +42 -0
  312. nat/observability/processor/falsy_batch_filter_processor.py +55 -0
  313. nat/observability/processor/intermediate_step_serializer.py +28 -0
  314. nat/observability/processor/processor.py +74 -0
  315. nat/observability/processor/processor_factory.py +70 -0
  316. nat/observability/processor/redaction/__init__.py +24 -0
  317. nat/observability/processor/redaction/contextual_redaction_processor.py +125 -0
  318. nat/observability/processor/redaction/contextual_span_redaction_processor.py +66 -0
  319. nat/observability/processor/redaction/redaction_processor.py +177 -0
  320. nat/observability/processor/redaction/span_header_redaction_processor.py +92 -0
  321. nat/observability/processor/span_tagging_processor.py +68 -0
  322. nat/observability/register.py +114 -0
  323. nat/observability/utils/__init__.py +14 -0
  324. nat/observability/utils/dict_utils.py +236 -0
  325. nat/observability/utils/time_utils.py +31 -0
  326. nat/plugins/.namespace +1 -0
  327. nat/profiler/__init__.py +0 -0
  328. nat/profiler/calc/__init__.py +14 -0
  329. nat/profiler/calc/calc_runner.py +626 -0
  330. nat/profiler/calc/calculations.py +288 -0
  331. nat/profiler/calc/data_models.py +188 -0
  332. nat/profiler/calc/plot.py +345 -0
  333. nat/profiler/callbacks/__init__.py +0 -0
  334. nat/profiler/callbacks/agno_callback_handler.py +295 -0
  335. nat/profiler/callbacks/base_callback_class.py +20 -0
  336. nat/profiler/callbacks/langchain_callback_handler.py +297 -0
  337. nat/profiler/callbacks/llama_index_callback_handler.py +205 -0
  338. nat/profiler/callbacks/semantic_kernel_callback_handler.py +238 -0
  339. nat/profiler/callbacks/token_usage_base_model.py +27 -0
  340. nat/profiler/data_frame_row.py +51 -0
  341. nat/profiler/data_models.py +24 -0
  342. nat/profiler/decorators/__init__.py +0 -0
  343. nat/profiler/decorators/framework_wrapper.py +180 -0
  344. nat/profiler/decorators/function_tracking.py +411 -0
  345. nat/profiler/forecasting/__init__.py +0 -0
  346. nat/profiler/forecasting/config.py +18 -0
  347. nat/profiler/forecasting/model_trainer.py +75 -0
  348. nat/profiler/forecasting/models/__init__.py +22 -0
  349. nat/profiler/forecasting/models/forecasting_base_model.py +42 -0
  350. nat/profiler/forecasting/models/linear_model.py +197 -0
  351. nat/profiler/forecasting/models/random_forest_regressor.py +269 -0
  352. nat/profiler/inference_metrics_model.py +28 -0
  353. nat/profiler/inference_optimization/__init__.py +0 -0
  354. nat/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
  355. nat/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +460 -0
  356. nat/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +258 -0
  357. nat/profiler/inference_optimization/data_models.py +386 -0
  358. nat/profiler/inference_optimization/experimental/__init__.py +0 -0
  359. nat/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +468 -0
  360. nat/profiler/inference_optimization/experimental/prefix_span_analysis.py +404 -0
  361. nat/profiler/inference_optimization/llm_metrics.py +212 -0
  362. nat/profiler/inference_optimization/prompt_caching.py +163 -0
  363. nat/profiler/inference_optimization/token_uniqueness.py +107 -0
  364. nat/profiler/inference_optimization/workflow_runtimes.py +72 -0
  365. nat/profiler/intermediate_property_adapter.py +102 -0
  366. nat/profiler/parameter_optimization/__init__.py +0 -0
  367. nat/profiler/parameter_optimization/optimizable_utils.py +93 -0
  368. nat/profiler/parameter_optimization/optimizer_runtime.py +67 -0
  369. nat/profiler/parameter_optimization/parameter_optimizer.py +153 -0
  370. nat/profiler/parameter_optimization/parameter_selection.py +107 -0
  371. nat/profiler/parameter_optimization/pareto_visualizer.py +380 -0
  372. nat/profiler/parameter_optimization/prompt_optimizer.py +384 -0
  373. nat/profiler/parameter_optimization/update_helpers.py +66 -0
  374. nat/profiler/profile_runner.py +478 -0
  375. nat/profiler/utils.py +186 -0
  376. nat/registry_handlers/__init__.py +0 -0
  377. nat/registry_handlers/local/__init__.py +0 -0
  378. nat/registry_handlers/local/local_handler.py +176 -0
  379. nat/registry_handlers/local/register_local.py +37 -0
  380. nat/registry_handlers/metadata_factory.py +60 -0
  381. nat/registry_handlers/package_utils.py +570 -0
  382. nat/registry_handlers/pypi/__init__.py +0 -0
  383. nat/registry_handlers/pypi/pypi_handler.py +248 -0
  384. nat/registry_handlers/pypi/register_pypi.py +40 -0
  385. nat/registry_handlers/register.py +20 -0
  386. nat/registry_handlers/registry_handler_base.py +157 -0
  387. nat/registry_handlers/rest/__init__.py +0 -0
  388. nat/registry_handlers/rest/register_rest.py +56 -0
  389. nat/registry_handlers/rest/rest_handler.py +236 -0
  390. nat/registry_handlers/schemas/__init__.py +0 -0
  391. nat/registry_handlers/schemas/headers.py +42 -0
  392. nat/registry_handlers/schemas/package.py +68 -0
  393. nat/registry_handlers/schemas/publish.py +68 -0
  394. nat/registry_handlers/schemas/pull.py +82 -0
  395. nat/registry_handlers/schemas/remove.py +36 -0
  396. nat/registry_handlers/schemas/search.py +91 -0
  397. nat/registry_handlers/schemas/status.py +47 -0
  398. nat/retriever/__init__.py +0 -0
  399. nat/retriever/interface.py +41 -0
  400. nat/retriever/milvus/__init__.py +14 -0
  401. nat/retriever/milvus/register.py +81 -0
  402. nat/retriever/milvus/retriever.py +228 -0
  403. nat/retriever/models.py +77 -0
  404. nat/retriever/nemo_retriever/__init__.py +14 -0
  405. nat/retriever/nemo_retriever/register.py +60 -0
  406. nat/retriever/nemo_retriever/retriever.py +190 -0
  407. nat/retriever/register.py +21 -0
  408. nat/runtime/__init__.py +14 -0
  409. nat/runtime/loader.py +220 -0
  410. nat/runtime/runner.py +292 -0
  411. nat/runtime/session.py +223 -0
  412. nat/runtime/user_metadata.py +130 -0
  413. nat/settings/__init__.py +0 -0
  414. nat/settings/global_settings.py +329 -0
  415. nat/test/.namespace +1 -0
  416. nat/tool/__init__.py +0 -0
  417. nat/tool/chat_completion.py +77 -0
  418. nat/tool/code_execution/README.md +151 -0
  419. nat/tool/code_execution/__init__.py +0 -0
  420. nat/tool/code_execution/code_sandbox.py +267 -0
  421. nat/tool/code_execution/local_sandbox/.gitignore +1 -0
  422. nat/tool/code_execution/local_sandbox/Dockerfile.sandbox +60 -0
  423. nat/tool/code_execution/local_sandbox/__init__.py +13 -0
  424. nat/tool/code_execution/local_sandbox/local_sandbox_server.py +198 -0
  425. nat/tool/code_execution/local_sandbox/sandbox.requirements.txt +6 -0
  426. nat/tool/code_execution/local_sandbox/start_local_sandbox.sh +50 -0
  427. nat/tool/code_execution/register.py +74 -0
  428. nat/tool/code_execution/test_code_execution_sandbox.py +414 -0
  429. nat/tool/code_execution/utils.py +100 -0
  430. nat/tool/datetime_tools.py +82 -0
  431. nat/tool/document_search.py +141 -0
  432. nat/tool/github_tools.py +450 -0
  433. nat/tool/memory_tools/__init__.py +0 -0
  434. nat/tool/memory_tools/add_memory_tool.py +79 -0
  435. nat/tool/memory_tools/delete_memory_tool.py +66 -0
  436. nat/tool/memory_tools/get_memory_tool.py +72 -0
  437. nat/tool/nvidia_rag.py +95 -0
  438. nat/tool/register.py +31 -0
  439. nat/tool/retriever.py +95 -0
  440. nat/tool/server_tools.py +66 -0
  441. nat/utils/__init__.py +0 -0
  442. nat/utils/callable_utils.py +70 -0
  443. nat/utils/data_models/__init__.py +0 -0
  444. nat/utils/data_models/schema_validator.py +58 -0
  445. nat/utils/debugging_utils.py +43 -0
  446. nat/utils/decorators.py +210 -0
  447. nat/utils/dump_distro_mapping.py +32 -0
  448. nat/utils/exception_handlers/__init__.py +0 -0
  449. nat/utils/exception_handlers/automatic_retries.py +342 -0
  450. nat/utils/exception_handlers/schemas.py +114 -0
  451. nat/utils/io/__init__.py +0 -0
  452. nat/utils/io/model_processing.py +28 -0
  453. nat/utils/io/yaml_tools.py +119 -0
  454. nat/utils/log_levels.py +25 -0
  455. nat/utils/log_utils.py +37 -0
  456. nat/utils/metadata_utils.py +74 -0
  457. nat/utils/optional_imports.py +142 -0
  458. nat/utils/producer_consumer_queue.py +178 -0
  459. nat/utils/reactive/__init__.py +0 -0
  460. nat/utils/reactive/base/__init__.py +0 -0
  461. nat/utils/reactive/base/observable_base.py +65 -0
  462. nat/utils/reactive/base/observer_base.py +55 -0
  463. nat/utils/reactive/base/subject_base.py +79 -0
  464. nat/utils/reactive/observable.py +59 -0
  465. nat/utils/reactive/observer.py +76 -0
  466. nat/utils/reactive/subject.py +131 -0
  467. nat/utils/reactive/subscription.py +49 -0
  468. nat/utils/settings/__init__.py +0 -0
  469. nat/utils/settings/global_settings.py +195 -0
  470. nat/utils/string_utils.py +38 -0
  471. nat/utils/type_converter.py +299 -0
  472. nat/utils/type_utils.py +488 -0
  473. nat/utils/url_utils.py +27 -0
  474. nvidia_nat-1.1.0a20251020.dist-info/METADATA +195 -0
  475. nvidia_nat-1.1.0a20251020.dist-info/RECORD +480 -0
  476. nvidia_nat-1.1.0a20251020.dist-info/WHEEL +5 -0
  477. nvidia_nat-1.1.0a20251020.dist-info/entry_points.txt +22 -0
  478. nvidia_nat-1.1.0a20251020.dist-info/licenses/LICENSE-3rd-party.txt +5478 -0
  479. nvidia_nat-1.1.0a20251020.dist-info/licenses/LICENSE.md +201 -0
  480. nvidia_nat-1.1.0a20251020.dist-info/top_level.txt +2 -0
@@ -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 nat.utils.reactive.base.observer_base import ObserverBase
24
+ from nat.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)
29
+ _T = TypeVar("_T")
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)
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 nat.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 nat.utils.reactive.base.observable_base import ObservableBase
20
+ from nat.utils.reactive.base.observer_base import ObserverBase
21
+ from nat.utils.reactive.observer import Observer
22
+ from nat.utils.reactive.subscription import Subscription
23
+ from nat.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)
28
+ _T = TypeVar("_T")
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 nat.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)
27
+ _T = TypeVar("_T")
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)
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)
@@ -0,0 +1,131 @@
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 threading
17
+ from collections.abc import Callable
18
+ from typing import TypeVar
19
+
20
+ from nat.utils.reactive.base.subject_base import SubjectBase
21
+ from nat.utils.reactive.observable import Observable
22
+ from nat.utils.reactive.observer import Observer
23
+ from nat.utils.reactive.subscription import Subscription
24
+
25
+ T = TypeVar("T")
26
+
27
+ OnNext = Callable[[T], None]
28
+ OnError = Callable[[Exception], None]
29
+ OnComplete = Callable[[], None]
30
+
31
+
32
+ class Subject(Observable[T], Observer[T], SubjectBase[T]):
33
+ """
34
+ A Subject is both an Observer (receives events) and an Observable (sends events).
35
+ - Maintains a list of ObserverBase[T].
36
+ - No internal buffering or replay; events are only delivered to current subscribers.
37
+ - Thread-safe via a lock.
38
+
39
+ Once on_error or on_complete is called, the Subject is closed.
40
+ """
41
+
42
+ def __init__(self) -> None:
43
+ super().__init__()
44
+ self._lock = threading.RLock()
45
+ self._closed = False
46
+ self._error: Exception | None = None
47
+ self._observers: list[Observer[T]] = []
48
+ self._disposed = False
49
+
50
+ # ==========================================================================
51
+ # Observable[T] - for consumers
52
+ # ==========================================================================
53
+ def _subscribe_core(self, observer: Observer[T]) -> Subscription:
54
+ """
55
+ Subscribe to this subject. If disposed, returns a dummy subscription.
56
+ Otherwise, registers the given observer.
57
+ """
58
+ with self._lock:
59
+ if self._disposed:
60
+ # Already disposed => no subscription
61
+ return Subscription(self, None)
62
+
63
+ self._observers.append(observer)
64
+ return Subscription(self, observer)
65
+
66
+ # ==========================================================================
67
+ # ObserverBase[T] - for producers
68
+ # ==========================================================================
69
+ def on_next(self, value: T) -> None:
70
+ """
71
+ Called by producers to emit an item. Delivers synchronously to each observer.
72
+ If closed or disposed, do nothing.
73
+ """
74
+ with self._lock:
75
+ if self._closed or self._disposed:
76
+ return
77
+ # Copy the current observers to avoid mutation issues
78
+ current_observers = list(self._observers)
79
+
80
+ # Deliver outside the lock
81
+ for obs in current_observers:
82
+ obs.on_next(value)
83
+
84
+ def on_error(self, exc: Exception) -> None:
85
+ """
86
+ Called by producers to signal an error. Notifies all observers.
87
+ """
88
+ with self._lock:
89
+ if self._closed or self._disposed:
90
+ return
91
+ current_obs = list(self._observers)
92
+
93
+ for obs in current_obs:
94
+ obs.on_error(exc)
95
+
96
+ def on_complete(self) -> None:
97
+ """
98
+ Called by producers to signal completion. Notifies all observers, then
99
+ clears them. Subject is closed.
100
+ """
101
+ with self._lock:
102
+ if self._closed or self._disposed:
103
+ return
104
+ current_observers = list(self._observers)
105
+ self.dispose()
106
+
107
+ for obs in current_observers:
108
+ obs.on_complete()
109
+
110
+ # ==========================================================================
111
+ # SubjectBase - internal unsubscribing
112
+ # ==========================================================================
113
+ def _unsubscribe_observer(self, observer: Observer[T]) -> None:
114
+ with self._lock:
115
+ if not self._disposed and observer in self._observers:
116
+ self._observers.remove(observer)
117
+
118
+ # ==========================================================================
119
+ # Disposal
120
+ # ==========================================================================
121
+ def dispose(self) -> None:
122
+ """
123
+ Immediately close the Subject. No future on_next, on_error, or on_complete.
124
+ Clears all observers.
125
+ """
126
+ with self._lock:
127
+ if not self._disposed:
128
+ self._disposed = True
129
+ self._observers.clear()
130
+ self._closed = True
131
+ self._error = None