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