nvidia-nat 1.2.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- aiq/__init__.py +66 -0
- nat/agent/__init__.py +0 -0
- nat/agent/base.py +256 -0
- nat/agent/dual_node.py +67 -0
- nat/agent/react_agent/__init__.py +0 -0
- nat/agent/react_agent/agent.py +363 -0
- nat/agent/react_agent/output_parser.py +104 -0
- nat/agent/react_agent/prompt.py +44 -0
- nat/agent/react_agent/register.py +149 -0
- nat/agent/reasoning_agent/__init__.py +0 -0
- nat/agent/reasoning_agent/reasoning_agent.py +225 -0
- nat/agent/register.py +23 -0
- nat/agent/rewoo_agent/__init__.py +0 -0
- nat/agent/rewoo_agent/agent.py +415 -0
- nat/agent/rewoo_agent/prompt.py +110 -0
- nat/agent/rewoo_agent/register.py +157 -0
- nat/agent/tool_calling_agent/__init__.py +0 -0
- nat/agent/tool_calling_agent/agent.py +119 -0
- nat/agent/tool_calling_agent/register.py +106 -0
- nat/authentication/__init__.py +14 -0
- nat/authentication/api_key/__init__.py +14 -0
- nat/authentication/api_key/api_key_auth_provider.py +96 -0
- nat/authentication/api_key/api_key_auth_provider_config.py +124 -0
- nat/authentication/api_key/register.py +26 -0
- nat/authentication/exceptions/__init__.py +14 -0
- nat/authentication/exceptions/api_key_exceptions.py +38 -0
- nat/authentication/http_basic_auth/__init__.py +0 -0
- nat/authentication/http_basic_auth/http_basic_auth_provider.py +81 -0
- nat/authentication/http_basic_auth/register.py +30 -0
- nat/authentication/interfaces.py +93 -0
- nat/authentication/oauth2/__init__.py +14 -0
- nat/authentication/oauth2/oauth2_auth_code_flow_provider.py +107 -0
- nat/authentication/oauth2/oauth2_auth_code_flow_provider_config.py +39 -0
- nat/authentication/oauth2/register.py +25 -0
- nat/authentication/register.py +21 -0
- nat/builder/__init__.py +0 -0
- nat/builder/builder.py +285 -0
- nat/builder/component_utils.py +316 -0
- nat/builder/context.py +270 -0
- nat/builder/embedder.py +24 -0
- nat/builder/eval_builder.py +161 -0
- nat/builder/evaluator.py +29 -0
- nat/builder/framework_enum.py +24 -0
- nat/builder/front_end.py +73 -0
- nat/builder/function.py +344 -0
- nat/builder/function_base.py +380 -0
- nat/builder/function_info.py +627 -0
- nat/builder/intermediate_step_manager.py +174 -0
- nat/builder/llm.py +25 -0
- nat/builder/retriever.py +25 -0
- nat/builder/user_interaction_manager.py +78 -0
- nat/builder/workflow.py +148 -0
- nat/builder/workflow_builder.py +1117 -0
- nat/cli/__init__.py +14 -0
- nat/cli/cli_utils/__init__.py +0 -0
- nat/cli/cli_utils/config_override.py +231 -0
- nat/cli/cli_utils/validation.py +37 -0
- nat/cli/commands/__init__.py +0 -0
- nat/cli/commands/configure/__init__.py +0 -0
- nat/cli/commands/configure/channel/__init__.py +0 -0
- nat/cli/commands/configure/channel/add.py +28 -0
- nat/cli/commands/configure/channel/channel.py +34 -0
- nat/cli/commands/configure/channel/remove.py +30 -0
- nat/cli/commands/configure/channel/update.py +30 -0
- nat/cli/commands/configure/configure.py +33 -0
- nat/cli/commands/evaluate.py +139 -0
- nat/cli/commands/info/__init__.py +14 -0
- nat/cli/commands/info/info.py +37 -0
- nat/cli/commands/info/list_channels.py +32 -0
- nat/cli/commands/info/list_components.py +129 -0
- nat/cli/commands/info/list_mcp.py +304 -0
- nat/cli/commands/registry/__init__.py +14 -0
- nat/cli/commands/registry/publish.py +88 -0
- nat/cli/commands/registry/pull.py +118 -0
- nat/cli/commands/registry/registry.py +36 -0
- nat/cli/commands/registry/remove.py +108 -0
- nat/cli/commands/registry/search.py +155 -0
- nat/cli/commands/sizing/__init__.py +14 -0
- nat/cli/commands/sizing/calc.py +297 -0
- nat/cli/commands/sizing/sizing.py +27 -0
- nat/cli/commands/start.py +246 -0
- nat/cli/commands/uninstall.py +81 -0
- nat/cli/commands/validate.py +47 -0
- nat/cli/commands/workflow/__init__.py +14 -0
- nat/cli/commands/workflow/templates/__init__.py.j2 +0 -0
- nat/cli/commands/workflow/templates/config.yml.j2 +16 -0
- nat/cli/commands/workflow/templates/pyproject.toml.j2 +22 -0
- nat/cli/commands/workflow/templates/register.py.j2 +5 -0
- nat/cli/commands/workflow/templates/workflow.py.j2 +36 -0
- nat/cli/commands/workflow/workflow.py +37 -0
- nat/cli/commands/workflow/workflow_commands.py +317 -0
- nat/cli/entrypoint.py +135 -0
- nat/cli/main.py +57 -0
- nat/cli/register_workflow.py +488 -0
- nat/cli/type_registry.py +1000 -0
- nat/data_models/__init__.py +14 -0
- nat/data_models/api_server.py +716 -0
- nat/data_models/authentication.py +231 -0
- nat/data_models/common.py +171 -0
- nat/data_models/component.py +58 -0
- nat/data_models/component_ref.py +168 -0
- nat/data_models/config.py +410 -0
- nat/data_models/dataset_handler.py +169 -0
- nat/data_models/discovery_metadata.py +305 -0
- nat/data_models/embedder.py +27 -0
- nat/data_models/evaluate.py +127 -0
- nat/data_models/evaluator.py +26 -0
- nat/data_models/front_end.py +26 -0
- nat/data_models/function.py +30 -0
- nat/data_models/function_dependencies.py +72 -0
- nat/data_models/interactive.py +246 -0
- nat/data_models/intermediate_step.py +302 -0
- nat/data_models/invocation_node.py +38 -0
- nat/data_models/llm.py +27 -0
- nat/data_models/logging.py +26 -0
- nat/data_models/memory.py +27 -0
- nat/data_models/object_store.py +44 -0
- nat/data_models/profiler.py +54 -0
- nat/data_models/registry_handler.py +26 -0
- nat/data_models/retriever.py +30 -0
- nat/data_models/retry_mixin.py +35 -0
- nat/data_models/span.py +190 -0
- nat/data_models/step_adaptor.py +64 -0
- nat/data_models/streaming.py +33 -0
- nat/data_models/swe_bench_model.py +54 -0
- nat/data_models/telemetry_exporter.py +26 -0
- nat/data_models/ttc_strategy.py +30 -0
- nat/embedder/__init__.py +0 -0
- nat/embedder/nim_embedder.py +59 -0
- nat/embedder/openai_embedder.py +43 -0
- nat/embedder/register.py +22 -0
- nat/eval/__init__.py +14 -0
- nat/eval/config.py +60 -0
- nat/eval/dataset_handler/__init__.py +0 -0
- nat/eval/dataset_handler/dataset_downloader.py +106 -0
- nat/eval/dataset_handler/dataset_filter.py +52 -0
- nat/eval/dataset_handler/dataset_handler.py +367 -0
- nat/eval/evaluate.py +510 -0
- nat/eval/evaluator/__init__.py +14 -0
- nat/eval/evaluator/base_evaluator.py +77 -0
- nat/eval/evaluator/evaluator_model.py +45 -0
- nat/eval/intermediate_step_adapter.py +99 -0
- nat/eval/rag_evaluator/__init__.py +0 -0
- nat/eval/rag_evaluator/evaluate.py +178 -0
- nat/eval/rag_evaluator/register.py +143 -0
- nat/eval/register.py +23 -0
- nat/eval/remote_workflow.py +133 -0
- nat/eval/runners/__init__.py +14 -0
- nat/eval/runners/config.py +39 -0
- nat/eval/runners/multi_eval_runner.py +54 -0
- nat/eval/runtime_event_subscriber.py +52 -0
- nat/eval/swe_bench_evaluator/__init__.py +0 -0
- nat/eval/swe_bench_evaluator/evaluate.py +215 -0
- nat/eval/swe_bench_evaluator/register.py +36 -0
- nat/eval/trajectory_evaluator/__init__.py +0 -0
- nat/eval/trajectory_evaluator/evaluate.py +75 -0
- nat/eval/trajectory_evaluator/register.py +40 -0
- nat/eval/tunable_rag_evaluator/__init__.py +0 -0
- nat/eval/tunable_rag_evaluator/evaluate.py +245 -0
- nat/eval/tunable_rag_evaluator/register.py +52 -0
- nat/eval/usage_stats.py +41 -0
- nat/eval/utils/__init__.py +0 -0
- nat/eval/utils/output_uploader.py +140 -0
- nat/eval/utils/tqdm_position_registry.py +40 -0
- nat/eval/utils/weave_eval.py +184 -0
- nat/experimental/__init__.py +0 -0
- nat/experimental/decorators/__init__.py +0 -0
- nat/experimental/decorators/experimental_warning_decorator.py +134 -0
- nat/experimental/test_time_compute/__init__.py +0 -0
- nat/experimental/test_time_compute/editing/__init__.py +0 -0
- nat/experimental/test_time_compute/editing/iterative_plan_refinement_editor.py +147 -0
- nat/experimental/test_time_compute/editing/llm_as_a_judge_editor.py +204 -0
- nat/experimental/test_time_compute/editing/motivation_aware_summarization.py +107 -0
- nat/experimental/test_time_compute/functions/__init__.py +0 -0
- nat/experimental/test_time_compute/functions/execute_score_select_function.py +105 -0
- nat/experimental/test_time_compute/functions/plan_select_execute_function.py +224 -0
- nat/experimental/test_time_compute/functions/ttc_tool_orchestration_function.py +205 -0
- nat/experimental/test_time_compute/functions/ttc_tool_wrapper_function.py +146 -0
- nat/experimental/test_time_compute/models/__init__.py +0 -0
- nat/experimental/test_time_compute/models/editor_config.py +132 -0
- nat/experimental/test_time_compute/models/scoring_config.py +112 -0
- nat/experimental/test_time_compute/models/search_config.py +120 -0
- nat/experimental/test_time_compute/models/selection_config.py +154 -0
- nat/experimental/test_time_compute/models/stage_enums.py +43 -0
- nat/experimental/test_time_compute/models/strategy_base.py +66 -0
- nat/experimental/test_time_compute/models/tool_use_config.py +41 -0
- nat/experimental/test_time_compute/models/ttc_item.py +48 -0
- nat/experimental/test_time_compute/register.py +36 -0
- nat/experimental/test_time_compute/scoring/__init__.py +0 -0
- nat/experimental/test_time_compute/scoring/llm_based_agent_scorer.py +168 -0
- nat/experimental/test_time_compute/scoring/llm_based_plan_scorer.py +168 -0
- nat/experimental/test_time_compute/scoring/motivation_aware_scorer.py +111 -0
- nat/experimental/test_time_compute/search/__init__.py +0 -0
- nat/experimental/test_time_compute/search/multi_llm_planner.py +128 -0
- nat/experimental/test_time_compute/search/multi_query_retrieval_search.py +122 -0
- nat/experimental/test_time_compute/search/single_shot_multi_plan_planner.py +128 -0
- nat/experimental/test_time_compute/selection/__init__.py +0 -0
- nat/experimental/test_time_compute/selection/best_of_n_selector.py +63 -0
- nat/experimental/test_time_compute/selection/llm_based_agent_output_selector.py +131 -0
- nat/experimental/test_time_compute/selection/llm_based_output_merging_selector.py +159 -0
- nat/experimental/test_time_compute/selection/llm_based_plan_selector.py +128 -0
- nat/experimental/test_time_compute/selection/threshold_selector.py +58 -0
- nat/front_ends/__init__.py +14 -0
- nat/front_ends/console/__init__.py +14 -0
- nat/front_ends/console/authentication_flow_handler.py +233 -0
- nat/front_ends/console/console_front_end_config.py +32 -0
- nat/front_ends/console/console_front_end_plugin.py +96 -0
- nat/front_ends/console/register.py +25 -0
- nat/front_ends/cron/__init__.py +14 -0
- nat/front_ends/fastapi/__init__.py +14 -0
- nat/front_ends/fastapi/auth_flow_handlers/__init__.py +0 -0
- nat/front_ends/fastapi/auth_flow_handlers/http_flow_handler.py +27 -0
- nat/front_ends/fastapi/auth_flow_handlers/websocket_flow_handler.py +107 -0
- nat/front_ends/fastapi/fastapi_front_end_config.py +241 -0
- nat/front_ends/fastapi/fastapi_front_end_controller.py +68 -0
- nat/front_ends/fastapi/fastapi_front_end_plugin.py +116 -0
- nat/front_ends/fastapi/fastapi_front_end_plugin_worker.py +1087 -0
- nat/front_ends/fastapi/html_snippets/__init__.py +14 -0
- nat/front_ends/fastapi/html_snippets/auth_code_grant_success.py +35 -0
- nat/front_ends/fastapi/intermediate_steps_subscriber.py +80 -0
- nat/front_ends/fastapi/job_store.py +183 -0
- nat/front_ends/fastapi/main.py +72 -0
- nat/front_ends/fastapi/message_handler.py +320 -0
- nat/front_ends/fastapi/message_validator.py +352 -0
- nat/front_ends/fastapi/register.py +25 -0
- nat/front_ends/fastapi/response_helpers.py +195 -0
- nat/front_ends/fastapi/step_adaptor.py +319 -0
- nat/front_ends/mcp/__init__.py +14 -0
- nat/front_ends/mcp/mcp_front_end_config.py +36 -0
- nat/front_ends/mcp/mcp_front_end_plugin.py +81 -0
- nat/front_ends/mcp/mcp_front_end_plugin_worker.py +143 -0
- nat/front_ends/mcp/register.py +27 -0
- nat/front_ends/mcp/tool_converter.py +241 -0
- nat/front_ends/register.py +22 -0
- nat/front_ends/simple_base/__init__.py +14 -0
- nat/front_ends/simple_base/simple_front_end_plugin_base.py +54 -0
- nat/llm/__init__.py +0 -0
- nat/llm/aws_bedrock_llm.py +57 -0
- nat/llm/nim_llm.py +46 -0
- nat/llm/openai_llm.py +46 -0
- nat/llm/register.py +23 -0
- nat/llm/utils/__init__.py +14 -0
- nat/llm/utils/env_config_value.py +94 -0
- nat/llm/utils/error.py +17 -0
- nat/memory/__init__.py +20 -0
- nat/memory/interfaces.py +183 -0
- nat/memory/models.py +112 -0
- nat/meta/pypi.md +58 -0
- nat/object_store/__init__.py +20 -0
- nat/object_store/in_memory_object_store.py +76 -0
- nat/object_store/interfaces.py +84 -0
- nat/object_store/models.py +38 -0
- nat/object_store/register.py +20 -0
- nat/observability/__init__.py +14 -0
- nat/observability/exporter/__init__.py +14 -0
- nat/observability/exporter/base_exporter.py +449 -0
- nat/observability/exporter/exporter.py +78 -0
- nat/observability/exporter/file_exporter.py +33 -0
- nat/observability/exporter/processing_exporter.py +322 -0
- nat/observability/exporter/raw_exporter.py +52 -0
- nat/observability/exporter/span_exporter.py +288 -0
- nat/observability/exporter_manager.py +335 -0
- nat/observability/mixin/__init__.py +14 -0
- nat/observability/mixin/batch_config_mixin.py +26 -0
- nat/observability/mixin/collector_config_mixin.py +23 -0
- nat/observability/mixin/file_mixin.py +288 -0
- nat/observability/mixin/file_mode.py +23 -0
- nat/observability/mixin/resource_conflict_mixin.py +134 -0
- nat/observability/mixin/serialize_mixin.py +61 -0
- nat/observability/mixin/type_introspection_mixin.py +183 -0
- nat/observability/processor/__init__.py +14 -0
- nat/observability/processor/batching_processor.py +310 -0
- nat/observability/processor/callback_processor.py +42 -0
- nat/observability/processor/intermediate_step_serializer.py +28 -0
- nat/observability/processor/processor.py +71 -0
- nat/observability/register.py +96 -0
- nat/observability/utils/__init__.py +14 -0
- nat/observability/utils/dict_utils.py +236 -0
- nat/observability/utils/time_utils.py +31 -0
- nat/plugins/.namespace +1 -0
- nat/profiler/__init__.py +0 -0
- nat/profiler/calc/__init__.py +14 -0
- nat/profiler/calc/calc_runner.py +627 -0
- nat/profiler/calc/calculations.py +288 -0
- nat/profiler/calc/data_models.py +188 -0
- nat/profiler/calc/plot.py +345 -0
- nat/profiler/callbacks/__init__.py +0 -0
- nat/profiler/callbacks/agno_callback_handler.py +295 -0
- nat/profiler/callbacks/base_callback_class.py +20 -0
- nat/profiler/callbacks/langchain_callback_handler.py +290 -0
- nat/profiler/callbacks/llama_index_callback_handler.py +205 -0
- nat/profiler/callbacks/semantic_kernel_callback_handler.py +238 -0
- nat/profiler/callbacks/token_usage_base_model.py +27 -0
- nat/profiler/data_frame_row.py +51 -0
- nat/profiler/data_models.py +24 -0
- nat/profiler/decorators/__init__.py +0 -0
- nat/profiler/decorators/framework_wrapper.py +131 -0
- nat/profiler/decorators/function_tracking.py +254 -0
- nat/profiler/forecasting/__init__.py +0 -0
- nat/profiler/forecasting/config.py +18 -0
- nat/profiler/forecasting/model_trainer.py +75 -0
- nat/profiler/forecasting/models/__init__.py +22 -0
- nat/profiler/forecasting/models/forecasting_base_model.py +40 -0
- nat/profiler/forecasting/models/linear_model.py +197 -0
- nat/profiler/forecasting/models/random_forest_regressor.py +269 -0
- nat/profiler/inference_metrics_model.py +28 -0
- nat/profiler/inference_optimization/__init__.py +0 -0
- nat/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
- nat/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +460 -0
- nat/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +258 -0
- nat/profiler/inference_optimization/data_models.py +386 -0
- nat/profiler/inference_optimization/experimental/__init__.py +0 -0
- nat/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +468 -0
- nat/profiler/inference_optimization/experimental/prefix_span_analysis.py +405 -0
- nat/profiler/inference_optimization/llm_metrics.py +212 -0
- nat/profiler/inference_optimization/prompt_caching.py +163 -0
- nat/profiler/inference_optimization/token_uniqueness.py +107 -0
- nat/profiler/inference_optimization/workflow_runtimes.py +72 -0
- nat/profiler/intermediate_property_adapter.py +102 -0
- nat/profiler/profile_runner.py +473 -0
- nat/profiler/utils.py +184 -0
- nat/registry_handlers/__init__.py +0 -0
- nat/registry_handlers/local/__init__.py +0 -0
- nat/registry_handlers/local/local_handler.py +176 -0
- nat/registry_handlers/local/register_local.py +37 -0
- nat/registry_handlers/metadata_factory.py +60 -0
- nat/registry_handlers/package_utils.py +571 -0
- nat/registry_handlers/pypi/__init__.py +0 -0
- nat/registry_handlers/pypi/pypi_handler.py +251 -0
- nat/registry_handlers/pypi/register_pypi.py +40 -0
- nat/registry_handlers/register.py +21 -0
- nat/registry_handlers/registry_handler_base.py +157 -0
- nat/registry_handlers/rest/__init__.py +0 -0
- nat/registry_handlers/rest/register_rest.py +56 -0
- nat/registry_handlers/rest/rest_handler.py +237 -0
- nat/registry_handlers/schemas/__init__.py +0 -0
- nat/registry_handlers/schemas/headers.py +42 -0
- nat/registry_handlers/schemas/package.py +68 -0
- nat/registry_handlers/schemas/publish.py +68 -0
- nat/registry_handlers/schemas/pull.py +82 -0
- nat/registry_handlers/schemas/remove.py +36 -0
- nat/registry_handlers/schemas/search.py +91 -0
- nat/registry_handlers/schemas/status.py +47 -0
- nat/retriever/__init__.py +0 -0
- nat/retriever/interface.py +41 -0
- nat/retriever/milvus/__init__.py +14 -0
- nat/retriever/milvus/register.py +81 -0
- nat/retriever/milvus/retriever.py +228 -0
- nat/retriever/models.py +77 -0
- nat/retriever/nemo_retriever/__init__.py +14 -0
- nat/retriever/nemo_retriever/register.py +60 -0
- nat/retriever/nemo_retriever/retriever.py +190 -0
- nat/retriever/register.py +22 -0
- nat/runtime/__init__.py +14 -0
- nat/runtime/loader.py +220 -0
- nat/runtime/runner.py +195 -0
- nat/runtime/session.py +162 -0
- nat/runtime/user_metadata.py +130 -0
- nat/settings/__init__.py +0 -0
- nat/settings/global_settings.py +318 -0
- nat/test/.namespace +1 -0
- nat/tool/__init__.py +0 -0
- nat/tool/chat_completion.py +74 -0
- nat/tool/code_execution/README.md +151 -0
- nat/tool/code_execution/__init__.py +0 -0
- nat/tool/code_execution/code_sandbox.py +267 -0
- nat/tool/code_execution/local_sandbox/.gitignore +1 -0
- nat/tool/code_execution/local_sandbox/Dockerfile.sandbox +60 -0
- nat/tool/code_execution/local_sandbox/__init__.py +13 -0
- nat/tool/code_execution/local_sandbox/local_sandbox_server.py +198 -0
- nat/tool/code_execution/local_sandbox/sandbox.requirements.txt +6 -0
- nat/tool/code_execution/local_sandbox/start_local_sandbox.sh +50 -0
- nat/tool/code_execution/register.py +74 -0
- nat/tool/code_execution/test_code_execution_sandbox.py +414 -0
- nat/tool/code_execution/utils.py +100 -0
- nat/tool/datetime_tools.py +42 -0
- nat/tool/document_search.py +141 -0
- nat/tool/github_tools/__init__.py +0 -0
- nat/tool/github_tools/create_github_commit.py +133 -0
- nat/tool/github_tools/create_github_issue.py +87 -0
- nat/tool/github_tools/create_github_pr.py +106 -0
- nat/tool/github_tools/get_github_file.py +106 -0
- nat/tool/github_tools/get_github_issue.py +166 -0
- nat/tool/github_tools/get_github_pr.py +256 -0
- nat/tool/github_tools/update_github_issue.py +100 -0
- nat/tool/mcp/__init__.py +14 -0
- nat/tool/mcp/exceptions.py +142 -0
- nat/tool/mcp/mcp_client.py +255 -0
- nat/tool/mcp/mcp_tool.py +96 -0
- nat/tool/memory_tools/__init__.py +0 -0
- nat/tool/memory_tools/add_memory_tool.py +79 -0
- nat/tool/memory_tools/delete_memory_tool.py +67 -0
- nat/tool/memory_tools/get_memory_tool.py +72 -0
- nat/tool/nvidia_rag.py +95 -0
- nat/tool/register.py +38 -0
- nat/tool/retriever.py +94 -0
- nat/tool/server_tools.py +66 -0
- nat/utils/__init__.py +0 -0
- nat/utils/data_models/__init__.py +0 -0
- nat/utils/data_models/schema_validator.py +58 -0
- nat/utils/debugging_utils.py +43 -0
- nat/utils/dump_distro_mapping.py +32 -0
- nat/utils/exception_handlers/__init__.py +0 -0
- nat/utils/exception_handlers/automatic_retries.py +289 -0
- nat/utils/exception_handlers/mcp.py +211 -0
- nat/utils/exception_handlers/schemas.py +114 -0
- nat/utils/io/__init__.py +0 -0
- nat/utils/io/model_processing.py +28 -0
- nat/utils/io/yaml_tools.py +119 -0
- nat/utils/log_utils.py +37 -0
- nat/utils/metadata_utils.py +74 -0
- nat/utils/optional_imports.py +142 -0
- nat/utils/producer_consumer_queue.py +178 -0
- nat/utils/reactive/__init__.py +0 -0
- nat/utils/reactive/base/__init__.py +0 -0
- nat/utils/reactive/base/observable_base.py +65 -0
- nat/utils/reactive/base/observer_base.py +55 -0
- nat/utils/reactive/base/subject_base.py +79 -0
- nat/utils/reactive/observable.py +59 -0
- nat/utils/reactive/observer.py +76 -0
- nat/utils/reactive/subject.py +131 -0
- nat/utils/reactive/subscription.py +49 -0
- nat/utils/settings/__init__.py +0 -0
- nat/utils/settings/global_settings.py +197 -0
- nat/utils/string_utils.py +38 -0
- nat/utils/type_converter.py +290 -0
- nat/utils/type_utils.py +484 -0
- nat/utils/url_utils.py +27 -0
- nvidia_nat-1.2.0.dist-info/METADATA +365 -0
- nvidia_nat-1.2.0.dist-info/RECORD +435 -0
- nvidia_nat-1.2.0.dist-info/WHEEL +5 -0
- nvidia_nat-1.2.0.dist-info/entry_points.txt +21 -0
- nvidia_nat-1.2.0.dist-info/licenses/LICENSE-3rd-party.txt +5478 -0
- nvidia_nat-1.2.0.dist-info/licenses/LICENSE.md +201 -0
- nvidia_nat-1.2.0.dist-info/top_level.txt +2 -0
|
@@ -0,0 +1,211 @@
|
|
|
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
|
+
import ssl
|
|
18
|
+
import sys
|
|
19
|
+
from collections.abc import Callable
|
|
20
|
+
from functools import wraps
|
|
21
|
+
from typing import Any
|
|
22
|
+
|
|
23
|
+
import httpx
|
|
24
|
+
|
|
25
|
+
from nat.tool.mcp.exceptions import MCPAuthenticationError
|
|
26
|
+
from nat.tool.mcp.exceptions import MCPConnectionError
|
|
27
|
+
from nat.tool.mcp.exceptions import MCPError
|
|
28
|
+
from nat.tool.mcp.exceptions import MCPProtocolError
|
|
29
|
+
from nat.tool.mcp.exceptions import MCPRequestError
|
|
30
|
+
from nat.tool.mcp.exceptions import MCPSSLError
|
|
31
|
+
from nat.tool.mcp.exceptions import MCPTimeoutError
|
|
32
|
+
from nat.tool.mcp.exceptions import MCPToolNotFoundError
|
|
33
|
+
|
|
34
|
+
logger = logging.getLogger(__name__)
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def format_mcp_error(error: MCPError, include_traceback: bool = False) -> None:
|
|
38
|
+
"""Format MCP errors for CLI display with structured logging and user guidance.
|
|
39
|
+
|
|
40
|
+
Logs structured error information for debugging and displays user-friendly
|
|
41
|
+
error messages with actionable suggestions to stderr.
|
|
42
|
+
|
|
43
|
+
Args:
|
|
44
|
+
error (MCPError): MCPError instance containing message, url, category, suggestions, and original_exception
|
|
45
|
+
include_traceback (bool, optional): Whether to include the traceback in the error message. Defaults to False.
|
|
46
|
+
"""
|
|
47
|
+
# Log structured error information for debugging
|
|
48
|
+
logger.error("MCP operation failed: %s", error, exc_info=include_traceback)
|
|
49
|
+
|
|
50
|
+
# Display user-friendly suggestions
|
|
51
|
+
for suggestion in error.suggestions:
|
|
52
|
+
print(f" → {suggestion}", file=sys.stderr)
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def _extract_url(args: tuple, kwargs: dict[str, Any], url_param: str, func_name: str) -> str:
|
|
56
|
+
"""Extract URL from function arguments using clean fallback chain.
|
|
57
|
+
|
|
58
|
+
Args:
|
|
59
|
+
args: Function positional arguments
|
|
60
|
+
kwargs: Function keyword arguments
|
|
61
|
+
url_param (str): Parameter name containing the URL
|
|
62
|
+
func_name (str): Function name for logging
|
|
63
|
+
|
|
64
|
+
Returns:
|
|
65
|
+
str: URL string or "unknown" if extraction fails
|
|
66
|
+
"""
|
|
67
|
+
# Try keyword arguments first
|
|
68
|
+
if url_param in kwargs:
|
|
69
|
+
return kwargs[url_param]
|
|
70
|
+
|
|
71
|
+
# Try self attribute (e.g., self.url)
|
|
72
|
+
if args and hasattr(args[0], url_param):
|
|
73
|
+
return getattr(args[0], url_param)
|
|
74
|
+
|
|
75
|
+
# Try common case: url as second parameter after self
|
|
76
|
+
if len(args) > 1 and url_param == "url":
|
|
77
|
+
return args[1]
|
|
78
|
+
|
|
79
|
+
# Fallback with warning
|
|
80
|
+
logger.warning("Could not extract URL for error handling in %s", func_name)
|
|
81
|
+
return "unknown"
|
|
82
|
+
|
|
83
|
+
|
|
84
|
+
def extract_primary_exception(exceptions: list[Exception]) -> Exception:
|
|
85
|
+
"""Extract the most relevant exception from a group.
|
|
86
|
+
|
|
87
|
+
Prioritizes connection errors over others for better user experience.
|
|
88
|
+
|
|
89
|
+
Args:
|
|
90
|
+
exceptions (list[Exception]): List of exceptions from ExceptionGroup
|
|
91
|
+
|
|
92
|
+
Returns:
|
|
93
|
+
Exception: Most relevant exception for user feedback
|
|
94
|
+
"""
|
|
95
|
+
# Prioritize connection errors
|
|
96
|
+
for exc in exceptions:
|
|
97
|
+
if isinstance(exc, (httpx.ConnectError, ConnectionError)):
|
|
98
|
+
return exc
|
|
99
|
+
|
|
100
|
+
# Then timeout errors
|
|
101
|
+
for exc in exceptions:
|
|
102
|
+
if isinstance(exc, httpx.TimeoutException):
|
|
103
|
+
return exc
|
|
104
|
+
|
|
105
|
+
# Then SSL errors
|
|
106
|
+
for exc in exceptions:
|
|
107
|
+
if isinstance(exc, ssl.SSLError):
|
|
108
|
+
return exc
|
|
109
|
+
|
|
110
|
+
# Fall back to first exception
|
|
111
|
+
return exceptions[0]
|
|
112
|
+
|
|
113
|
+
|
|
114
|
+
def convert_to_mcp_error(exception: Exception, url: str) -> MCPError:
|
|
115
|
+
"""Convert single exception to appropriate MCPError.
|
|
116
|
+
|
|
117
|
+
Args:
|
|
118
|
+
exception (Exception): Single exception to convert
|
|
119
|
+
url (str): MCP server URL for context
|
|
120
|
+
|
|
121
|
+
Returns:
|
|
122
|
+
MCPError: Appropriate MCPError subclass
|
|
123
|
+
"""
|
|
124
|
+
match exception:
|
|
125
|
+
case httpx.ConnectError() | ConnectionError():
|
|
126
|
+
return MCPConnectionError(url, exception)
|
|
127
|
+
case httpx.TimeoutException():
|
|
128
|
+
return MCPTimeoutError(url, exception)
|
|
129
|
+
case ssl.SSLError():
|
|
130
|
+
return MCPSSLError(url, exception)
|
|
131
|
+
case httpx.RequestError():
|
|
132
|
+
return MCPRequestError(url, exception)
|
|
133
|
+
case ValueError() if "Tool" in str(exception) and "not available" in str(exception):
|
|
134
|
+
# Extract tool name from error message if possible
|
|
135
|
+
tool_name = str(exception).split("Tool ")[1].split(" not available")[0] if "Tool " in str(
|
|
136
|
+
exception) else "unknown"
|
|
137
|
+
return MCPToolNotFoundError(tool_name, url, exception)
|
|
138
|
+
case _:
|
|
139
|
+
# Handle TaskGroup error message specifically
|
|
140
|
+
if "unhandled errors in a TaskGroup" in str(exception):
|
|
141
|
+
return MCPProtocolError(url, "Failed to connect to MCP server", exception)
|
|
142
|
+
if "unauthorized" in str(exception).lower() or "forbidden" in str(exception).lower():
|
|
143
|
+
return MCPAuthenticationError(url, exception)
|
|
144
|
+
return MCPError(f"Unexpected error: {exception}", url, original_exception=exception)
|
|
145
|
+
|
|
146
|
+
|
|
147
|
+
def handle_mcp_exceptions(url_param: str = "url") -> Callable[..., Any]:
|
|
148
|
+
"""Decorator that handles exceptions and converts them to MCPErrors.
|
|
149
|
+
|
|
150
|
+
This decorator wraps MCP client methods and converts low-level exceptions
|
|
151
|
+
to structured MCPError instances with helpful user guidance.
|
|
152
|
+
|
|
153
|
+
Args:
|
|
154
|
+
url_param (str): Name of the parameter or attribute containing the MCP server URL
|
|
155
|
+
|
|
156
|
+
Returns:
|
|
157
|
+
Callable[..., Any]: Decorated function
|
|
158
|
+
|
|
159
|
+
Example:
|
|
160
|
+
.. code-block:: python
|
|
161
|
+
|
|
162
|
+
@handle_mcp_exceptions("url")
|
|
163
|
+
async def get_tools(self, url: str):
|
|
164
|
+
# Method implementation
|
|
165
|
+
pass
|
|
166
|
+
|
|
167
|
+
@handle_mcp_exceptions("url") # Uses self.url
|
|
168
|
+
async def get_tool(self):
|
|
169
|
+
# Method implementation
|
|
170
|
+
pass
|
|
171
|
+
"""
|
|
172
|
+
|
|
173
|
+
def decorator(func: Callable[..., Any]) -> Callable[..., Any]:
|
|
174
|
+
|
|
175
|
+
@wraps(func)
|
|
176
|
+
async def wrapper(*args, **kwargs):
|
|
177
|
+
try:
|
|
178
|
+
return await func(*args, **kwargs)
|
|
179
|
+
except MCPError:
|
|
180
|
+
# Re-raise MCPErrors as-is
|
|
181
|
+
raise
|
|
182
|
+
except Exception as e:
|
|
183
|
+
url = _extract_url(args, kwargs, url_param, func.__name__)
|
|
184
|
+
|
|
185
|
+
# Handle ExceptionGroup by extracting most relevant exception
|
|
186
|
+
if isinstance(e, ExceptionGroup): # noqa: F821
|
|
187
|
+
primary_exception = extract_primary_exception(list(e.exceptions))
|
|
188
|
+
mcp_error = convert_to_mcp_error(primary_exception, url)
|
|
189
|
+
else:
|
|
190
|
+
mcp_error = convert_to_mcp_error(e, url)
|
|
191
|
+
|
|
192
|
+
raise mcp_error from e
|
|
193
|
+
|
|
194
|
+
return wrapper
|
|
195
|
+
|
|
196
|
+
return decorator
|
|
197
|
+
|
|
198
|
+
|
|
199
|
+
def mcp_exception_handler(func: Callable[..., Any]) -> Callable[..., Any]:
|
|
200
|
+
"""Simplified decorator for methods that have self.url attribute.
|
|
201
|
+
|
|
202
|
+
This is a convenience decorator that assumes the URL is available as self.url.
|
|
203
|
+
Follows the same pattern as schema_exception_handler in this directory.
|
|
204
|
+
|
|
205
|
+
Args:
|
|
206
|
+
func (Callable[..., Any]): The function to decorate
|
|
207
|
+
|
|
208
|
+
Returns:
|
|
209
|
+
Callable[..., Any]: Decorated function
|
|
210
|
+
"""
|
|
211
|
+
return handle_mcp_exceptions("url")(func)
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2024-2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import logging
|
|
17
|
+
|
|
18
|
+
import yaml
|
|
19
|
+
from pydantic import ValidationError
|
|
20
|
+
|
|
21
|
+
logger = logging.getLogger(__name__)
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
def schema_exception_handler(func, **kwargs): # pylint: disable=unused-argument
|
|
25
|
+
"""
|
|
26
|
+
A decorator that handles `ValidationError` exceptions for schema validation functions.
|
|
27
|
+
|
|
28
|
+
This decorator wraps a function that performs schema validation using Pydantic.
|
|
29
|
+
If a `ValidationError` is raised, it logs detailed error messages and raises a `ValueError` with the combined error
|
|
30
|
+
messages.
|
|
31
|
+
|
|
32
|
+
Parameters
|
|
33
|
+
----------
|
|
34
|
+
func : callable
|
|
35
|
+
The function to be decorated. This function is expected to perform schema validation.
|
|
36
|
+
|
|
37
|
+
kwargs : dict
|
|
38
|
+
Additional keyword arguments to be passed to the function.
|
|
39
|
+
|
|
40
|
+
Returns
|
|
41
|
+
-------
|
|
42
|
+
callable
|
|
43
|
+
The wrapped function that executes `func` with exception handling.
|
|
44
|
+
|
|
45
|
+
Raises
|
|
46
|
+
------
|
|
47
|
+
ValueError
|
|
48
|
+
If a `ValidationError` is caught, this decorator logs the error details and raises a `ValueError` with the
|
|
49
|
+
combined error messages.
|
|
50
|
+
|
|
51
|
+
Notes
|
|
52
|
+
-----
|
|
53
|
+
This decorator is particularly useful for functions that validate configurations or data models,
|
|
54
|
+
ensuring that any validation errors are logged and communicated clearly.
|
|
55
|
+
|
|
56
|
+
Examples
|
|
57
|
+
--------
|
|
58
|
+
>>> @schema_exception_handler
|
|
59
|
+
... def validate_config(config_data):
|
|
60
|
+
... schema = MySchema(**config_data)
|
|
61
|
+
... return schema
|
|
62
|
+
...
|
|
63
|
+
>>> try:
|
|
64
|
+
... validate_config(invalid_config)
|
|
65
|
+
... except ValueError as e:
|
|
66
|
+
... logger.error("Caught error: %s", e)
|
|
67
|
+
Caught error: Invalid configuration: field1: value is not a valid integer; field2: field required
|
|
68
|
+
"""
|
|
69
|
+
|
|
70
|
+
def inner_function(*args, **kwargs):
|
|
71
|
+
try:
|
|
72
|
+
return func(*args, **kwargs)
|
|
73
|
+
except ValidationError as e:
|
|
74
|
+
error_messages = "; ".join([f"{error['loc'][0]}: {error['msg']}" for error in e.errors()])
|
|
75
|
+
log_error_message = f"Invalid configuration: {error_messages}"
|
|
76
|
+
logger.error(log_error_message)
|
|
77
|
+
raise ValueError(log_error_message) from e
|
|
78
|
+
|
|
79
|
+
return inner_function
|
|
80
|
+
|
|
81
|
+
|
|
82
|
+
def yaml_exception_handler(func):
|
|
83
|
+
"""
|
|
84
|
+
A decorator that handles YAML parsing exceptions.
|
|
85
|
+
|
|
86
|
+
This decorator wraps a function that performs YAML file operations.
|
|
87
|
+
If a YAML-related error occurs, it logs the error and raises a ValueError
|
|
88
|
+
with a clear error message.
|
|
89
|
+
|
|
90
|
+
Returns
|
|
91
|
+
-------
|
|
92
|
+
callable
|
|
93
|
+
The wrapped function that executes `func` with YAML exception handling.
|
|
94
|
+
|
|
95
|
+
Raises
|
|
96
|
+
------
|
|
97
|
+
ValueError
|
|
98
|
+
If a YAML error is caught, with details about the parsing failure.
|
|
99
|
+
"""
|
|
100
|
+
|
|
101
|
+
def inner_function(*args, **kwargs):
|
|
102
|
+
try:
|
|
103
|
+
return func(*args, **kwargs)
|
|
104
|
+
except yaml.YAMLError as e:
|
|
105
|
+
log_error_message = f"Invalid YAML configuration: {str(e)}"
|
|
106
|
+
logger.error(log_error_message)
|
|
107
|
+
raise ValueError(log_error_message) from e
|
|
108
|
+
|
|
109
|
+
except Exception as e:
|
|
110
|
+
log_error_message = f"Error reading YAML file: {str(e)}"
|
|
111
|
+
logger.error(log_error_message)
|
|
112
|
+
raise ValueError(log_error_message) from e
|
|
113
|
+
|
|
114
|
+
return inner_function
|
nat/utils/io/__init__.py
ADDED
|
File without changes
|
|
@@ -0,0 +1,28 @@
|
|
|
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 re
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def remove_r1_think_tags(text: str):
|
|
20
|
+
pattern = r'(<think>)?.*?</think>\s*(.*)'
|
|
21
|
+
|
|
22
|
+
# Add re.DOTALL flag to make . match newlines
|
|
23
|
+
match = re.match(pattern, text, re.DOTALL)
|
|
24
|
+
|
|
25
|
+
if match:
|
|
26
|
+
return match.group(2)
|
|
27
|
+
|
|
28
|
+
return text
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2024-2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import io
|
|
17
|
+
import logging
|
|
18
|
+
import typing
|
|
19
|
+
|
|
20
|
+
import expandvars
|
|
21
|
+
import yaml
|
|
22
|
+
|
|
23
|
+
from nat.utils.type_utils import StrPath
|
|
24
|
+
|
|
25
|
+
logger = logging.getLogger(__name__)
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
def _interpolate_variables(value: str | int | float | bool | None) -> str | int | float | bool | None:
|
|
29
|
+
"""
|
|
30
|
+
Interpolate variables in a string with the format ${VAR:-default_value}.
|
|
31
|
+
If the variable is not set, the default value will be used.
|
|
32
|
+
If no default value is provided, an empty string will be used.
|
|
33
|
+
|
|
34
|
+
Args:
|
|
35
|
+
value (str | int | float | bool | None): The value to interpolate variables in.
|
|
36
|
+
|
|
37
|
+
Returns:
|
|
38
|
+
str | int | float | bool | None: The value with variables interpolated.
|
|
39
|
+
"""
|
|
40
|
+
|
|
41
|
+
if not isinstance(value, str):
|
|
42
|
+
return value
|
|
43
|
+
|
|
44
|
+
return expandvars.expandvars(value)
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def yaml_load(config_path: StrPath) -> dict:
|
|
48
|
+
"""
|
|
49
|
+
Load a YAML file and interpolate variables in the format
|
|
50
|
+
${VAR:-default_value}.
|
|
51
|
+
|
|
52
|
+
Args:
|
|
53
|
+
config_path (StrPath): The path to the YAML file to load.
|
|
54
|
+
|
|
55
|
+
Returns:
|
|
56
|
+
dict: The processed configuration dictionary.
|
|
57
|
+
"""
|
|
58
|
+
|
|
59
|
+
# Read YAML file
|
|
60
|
+
with open(config_path, "r", encoding="utf-8") as stream:
|
|
61
|
+
config_str = stream.read()
|
|
62
|
+
|
|
63
|
+
return yaml_loads(config_str)
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
def yaml_loads(config: str) -> dict:
|
|
67
|
+
"""
|
|
68
|
+
Load a YAML string and interpolate variables in the format
|
|
69
|
+
${VAR:-default_value}.
|
|
70
|
+
|
|
71
|
+
Args:
|
|
72
|
+
config (str): The YAML string to load.
|
|
73
|
+
|
|
74
|
+
Returns:
|
|
75
|
+
dict: The processed configuration dictionary.
|
|
76
|
+
"""
|
|
77
|
+
|
|
78
|
+
interpolated_config_str = _interpolate_variables(config)
|
|
79
|
+
assert isinstance(interpolated_config_str, str), "Config must be a string"
|
|
80
|
+
|
|
81
|
+
stream = io.StringIO(interpolated_config_str)
|
|
82
|
+
stream.seek(0)
|
|
83
|
+
|
|
84
|
+
# Load the YAML data
|
|
85
|
+
try:
|
|
86
|
+
config_data = yaml.safe_load(stream)
|
|
87
|
+
except yaml.YAMLError as e:
|
|
88
|
+
logger.error("Error loading YAML: %s", interpolated_config_str, exc_info=True)
|
|
89
|
+
raise ValueError(f"Error loading YAML: {e}") from e
|
|
90
|
+
|
|
91
|
+
assert isinstance(config_data, dict)
|
|
92
|
+
|
|
93
|
+
return config_data
|
|
94
|
+
|
|
95
|
+
|
|
96
|
+
def yaml_dump(config: dict, fp: typing.TextIO) -> None:
|
|
97
|
+
"""
|
|
98
|
+
Dump a configuration dictionary to a YAML file.
|
|
99
|
+
|
|
100
|
+
Args:
|
|
101
|
+
config (dict): The configuration dictionary to dump.
|
|
102
|
+
fp (typing.TextIO): The file pointer to write the YAML to.
|
|
103
|
+
"""
|
|
104
|
+
yaml.dump(config, stream=fp, indent=2, sort_keys=False)
|
|
105
|
+
fp.flush()
|
|
106
|
+
|
|
107
|
+
|
|
108
|
+
def yaml_dumps(config: dict) -> str:
|
|
109
|
+
"""
|
|
110
|
+
Dump a configuration dictionary to a YAML string.
|
|
111
|
+
|
|
112
|
+
Args:
|
|
113
|
+
config (dict): The configuration dictionary to dump.
|
|
114
|
+
|
|
115
|
+
Returns:
|
|
116
|
+
str: The YAML string.
|
|
117
|
+
"""
|
|
118
|
+
|
|
119
|
+
return yaml.dump(config, indent=2)
|
nat/utils/log_utils.py
ADDED
|
@@ -0,0 +1,37 @@
|
|
|
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
|
+
|
|
19
|
+
class LogFilter(logging.Filter):
|
|
20
|
+
"""
|
|
21
|
+
This class is used to filter log records based on a defined set of criteria.
|
|
22
|
+
"""
|
|
23
|
+
|
|
24
|
+
def __init__(self, filter_criteria: list[str]):
|
|
25
|
+
self._filter_criteria = filter_criteria
|
|
26
|
+
super().__init__()
|
|
27
|
+
|
|
28
|
+
def filter(self, record: logging.LogRecord):
|
|
29
|
+
"""
|
|
30
|
+
Evaluates whether a log record should be emitted based on the message content.
|
|
31
|
+
|
|
32
|
+
Returns:
|
|
33
|
+
False if the message content contains any of the filter criteria, True otherwise.
|
|
34
|
+
"""
|
|
35
|
+
if any(match in record.getMessage() for match in self._filter_criteria):
|
|
36
|
+
return False
|
|
37
|
+
return True
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
from pydantic_core import PydanticUndefined
|
|
17
|
+
|
|
18
|
+
from nat.data_models.common import TypedBaseModelT
|
|
19
|
+
from nat.utils.type_utils import DecomposedType
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def generate_config_type_docs(config_type: TypedBaseModelT) -> str:
|
|
23
|
+
"""Generates a docstring from configuration object to facilitate discovery.
|
|
24
|
+
|
|
25
|
+
Args:
|
|
26
|
+
config_type (TypedBaseModelT): A component configuration object.
|
|
27
|
+
|
|
28
|
+
Returns:
|
|
29
|
+
str: An enriched docstring, including model attributes and default values.
|
|
30
|
+
"""
|
|
31
|
+
|
|
32
|
+
# Get the docstring
|
|
33
|
+
description_formatting = []
|
|
34
|
+
# Ensure uniform formatting of docstring
|
|
35
|
+
docstring = (config_type.__doc__ or "").strip().strip(".")
|
|
36
|
+
docstring = docstring + "." if docstring != "" else "Description unavailable."
|
|
37
|
+
description_formatting.append(docstring)
|
|
38
|
+
description_formatting.append("")
|
|
39
|
+
description_formatting.append(" Args:")
|
|
40
|
+
|
|
41
|
+
# Iterate over fields to get their documentation
|
|
42
|
+
for field_name, field_info in config_type.model_fields.items():
|
|
43
|
+
|
|
44
|
+
if (field_name == "type"):
|
|
45
|
+
field_name = "_type"
|
|
46
|
+
|
|
47
|
+
decomponsed_type = DecomposedType(field_info.annotation)
|
|
48
|
+
|
|
49
|
+
if not (decomponsed_type.is_union):
|
|
50
|
+
annotation = field_info.annotation.__name__
|
|
51
|
+
else:
|
|
52
|
+
annotation = field_info.annotation
|
|
53
|
+
|
|
54
|
+
default_string = ""
|
|
55
|
+
if ((field_info.get_default() is not PydanticUndefined) and (field_name != "_type")):
|
|
56
|
+
if issubclass(type(field_info.get_default()), str):
|
|
57
|
+
default_value = f'"{field_info.get_default()}"'
|
|
58
|
+
else:
|
|
59
|
+
default_value = field_info.get_default()
|
|
60
|
+
default_string += f" Defaults to {default_value}."
|
|
61
|
+
|
|
62
|
+
# Ensure uniform formatting of field info
|
|
63
|
+
field_info_description = (field_info.description or "").strip(".")
|
|
64
|
+
if field_info_description != "":
|
|
65
|
+
field_info_description = field_info_description + "."
|
|
66
|
+
else:
|
|
67
|
+
field_info_description = "Description unavailable."
|
|
68
|
+
|
|
69
|
+
parameter_string = f" {field_name} ({annotation}): {field_info_description}{default_string}"
|
|
70
|
+
description_formatting.append(parameter_string)
|
|
71
|
+
|
|
72
|
+
description = "\n".join(description_formatting)
|
|
73
|
+
|
|
74
|
+
return description
|