nvidia-nat 1.2.0rc5__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- aiq/agent/__init__.py +0 -0
- aiq/agent/base.py +239 -0
- aiq/agent/dual_node.py +67 -0
- aiq/agent/react_agent/__init__.py +0 -0
- aiq/agent/react_agent/agent.py +355 -0
- aiq/agent/react_agent/output_parser.py +104 -0
- aiq/agent/react_agent/prompt.py +41 -0
- aiq/agent/react_agent/register.py +149 -0
- aiq/agent/reasoning_agent/__init__.py +0 -0
- aiq/agent/reasoning_agent/reasoning_agent.py +225 -0
- aiq/agent/register.py +23 -0
- aiq/agent/rewoo_agent/__init__.py +0 -0
- aiq/agent/rewoo_agent/agent.py +411 -0
- aiq/agent/rewoo_agent/prompt.py +108 -0
- aiq/agent/rewoo_agent/register.py +158 -0
- aiq/agent/tool_calling_agent/__init__.py +0 -0
- aiq/agent/tool_calling_agent/agent.py +119 -0
- aiq/agent/tool_calling_agent/register.py +106 -0
- aiq/authentication/__init__.py +14 -0
- aiq/authentication/api_key/__init__.py +14 -0
- aiq/authentication/api_key/api_key_auth_provider.py +96 -0
- aiq/authentication/api_key/api_key_auth_provider_config.py +124 -0
- aiq/authentication/api_key/register.py +26 -0
- aiq/authentication/exceptions/__init__.py +14 -0
- aiq/authentication/exceptions/api_key_exceptions.py +38 -0
- aiq/authentication/http_basic_auth/__init__.py +0 -0
- aiq/authentication/http_basic_auth/http_basic_auth_provider.py +81 -0
- aiq/authentication/http_basic_auth/register.py +30 -0
- aiq/authentication/interfaces.py +93 -0
- aiq/authentication/oauth2/__init__.py +14 -0
- aiq/authentication/oauth2/oauth2_auth_code_flow_provider.py +107 -0
- aiq/authentication/oauth2/oauth2_auth_code_flow_provider_config.py +39 -0
- aiq/authentication/oauth2/register.py +25 -0
- aiq/authentication/register.py +21 -0
- aiq/builder/__init__.py +0 -0
- aiq/builder/builder.py +285 -0
- aiq/builder/component_utils.py +316 -0
- aiq/builder/context.py +264 -0
- aiq/builder/embedder.py +24 -0
- aiq/builder/eval_builder.py +161 -0
- aiq/builder/evaluator.py +29 -0
- aiq/builder/framework_enum.py +24 -0
- aiq/builder/front_end.py +73 -0
- aiq/builder/function.py +344 -0
- aiq/builder/function_base.py +380 -0
- aiq/builder/function_info.py +627 -0
- aiq/builder/intermediate_step_manager.py +174 -0
- aiq/builder/llm.py +25 -0
- aiq/builder/retriever.py +25 -0
- aiq/builder/user_interaction_manager.py +74 -0
- aiq/builder/workflow.py +148 -0
- aiq/builder/workflow_builder.py +1117 -0
- aiq/cli/__init__.py +14 -0
- aiq/cli/cli_utils/__init__.py +0 -0
- aiq/cli/cli_utils/config_override.py +231 -0
- aiq/cli/cli_utils/validation.py +37 -0
- aiq/cli/commands/__init__.py +0 -0
- aiq/cli/commands/configure/__init__.py +0 -0
- aiq/cli/commands/configure/channel/__init__.py +0 -0
- aiq/cli/commands/configure/channel/add.py +28 -0
- aiq/cli/commands/configure/channel/channel.py +36 -0
- aiq/cli/commands/configure/channel/remove.py +30 -0
- aiq/cli/commands/configure/channel/update.py +30 -0
- aiq/cli/commands/configure/configure.py +33 -0
- aiq/cli/commands/evaluate.py +139 -0
- aiq/cli/commands/info/__init__.py +14 -0
- aiq/cli/commands/info/info.py +39 -0
- aiq/cli/commands/info/list_channels.py +32 -0
- aiq/cli/commands/info/list_components.py +129 -0
- aiq/cli/commands/info/list_mcp.py +213 -0
- aiq/cli/commands/registry/__init__.py +14 -0
- aiq/cli/commands/registry/publish.py +88 -0
- aiq/cli/commands/registry/pull.py +118 -0
- aiq/cli/commands/registry/registry.py +38 -0
- aiq/cli/commands/registry/remove.py +108 -0
- aiq/cli/commands/registry/search.py +155 -0
- aiq/cli/commands/sizing/__init__.py +14 -0
- aiq/cli/commands/sizing/calc.py +297 -0
- aiq/cli/commands/sizing/sizing.py +27 -0
- aiq/cli/commands/start.py +246 -0
- aiq/cli/commands/uninstall.py +81 -0
- aiq/cli/commands/validate.py +47 -0
- aiq/cli/commands/workflow/__init__.py +14 -0
- aiq/cli/commands/workflow/templates/__init__.py.j2 +0 -0
- aiq/cli/commands/workflow/templates/config.yml.j2 +16 -0
- aiq/cli/commands/workflow/templates/pyproject.toml.j2 +22 -0
- aiq/cli/commands/workflow/templates/register.py.j2 +5 -0
- aiq/cli/commands/workflow/templates/workflow.py.j2 +36 -0
- aiq/cli/commands/workflow/workflow.py +37 -0
- aiq/cli/commands/workflow/workflow_commands.py +313 -0
- aiq/cli/entrypoint.py +135 -0
- aiq/cli/main.py +44 -0
- aiq/cli/register_workflow.py +488 -0
- aiq/cli/type_registry.py +1000 -0
- aiq/data_models/__init__.py +14 -0
- aiq/data_models/api_server.py +694 -0
- aiq/data_models/authentication.py +231 -0
- aiq/data_models/common.py +171 -0
- aiq/data_models/component.py +54 -0
- aiq/data_models/component_ref.py +168 -0
- aiq/data_models/config.py +406 -0
- aiq/data_models/dataset_handler.py +123 -0
- aiq/data_models/discovery_metadata.py +335 -0
- aiq/data_models/embedder.py +27 -0
- aiq/data_models/evaluate.py +127 -0
- aiq/data_models/evaluator.py +26 -0
- aiq/data_models/front_end.py +26 -0
- aiq/data_models/function.py +30 -0
- aiq/data_models/function_dependencies.py +72 -0
- aiq/data_models/interactive.py +246 -0
- aiq/data_models/intermediate_step.py +302 -0
- aiq/data_models/invocation_node.py +38 -0
- aiq/data_models/llm.py +27 -0
- aiq/data_models/logging.py +26 -0
- aiq/data_models/memory.py +27 -0
- aiq/data_models/object_store.py +44 -0
- aiq/data_models/profiler.py +54 -0
- aiq/data_models/registry_handler.py +26 -0
- aiq/data_models/retriever.py +30 -0
- aiq/data_models/retry_mixin.py +35 -0
- aiq/data_models/span.py +187 -0
- aiq/data_models/step_adaptor.py +64 -0
- aiq/data_models/streaming.py +33 -0
- aiq/data_models/swe_bench_model.py +54 -0
- aiq/data_models/telemetry_exporter.py +26 -0
- aiq/data_models/ttc_strategy.py +30 -0
- aiq/embedder/__init__.py +0 -0
- aiq/embedder/langchain_client.py +41 -0
- aiq/embedder/nim_embedder.py +59 -0
- aiq/embedder/openai_embedder.py +43 -0
- aiq/embedder/register.py +24 -0
- aiq/eval/__init__.py +14 -0
- aiq/eval/config.py +60 -0
- aiq/eval/dataset_handler/__init__.py +0 -0
- aiq/eval/dataset_handler/dataset_downloader.py +106 -0
- aiq/eval/dataset_handler/dataset_filter.py +52 -0
- aiq/eval/dataset_handler/dataset_handler.py +254 -0
- aiq/eval/evaluate.py +506 -0
- aiq/eval/evaluator/__init__.py +14 -0
- aiq/eval/evaluator/base_evaluator.py +73 -0
- aiq/eval/evaluator/evaluator_model.py +45 -0
- aiq/eval/intermediate_step_adapter.py +99 -0
- aiq/eval/rag_evaluator/__init__.py +0 -0
- aiq/eval/rag_evaluator/evaluate.py +178 -0
- aiq/eval/rag_evaluator/register.py +143 -0
- aiq/eval/register.py +23 -0
- aiq/eval/remote_workflow.py +133 -0
- aiq/eval/runners/__init__.py +14 -0
- aiq/eval/runners/config.py +39 -0
- aiq/eval/runners/multi_eval_runner.py +54 -0
- aiq/eval/runtime_event_subscriber.py +52 -0
- aiq/eval/swe_bench_evaluator/__init__.py +0 -0
- aiq/eval/swe_bench_evaluator/evaluate.py +215 -0
- aiq/eval/swe_bench_evaluator/register.py +36 -0
- aiq/eval/trajectory_evaluator/__init__.py +0 -0
- aiq/eval/trajectory_evaluator/evaluate.py +75 -0
- aiq/eval/trajectory_evaluator/register.py +40 -0
- aiq/eval/tunable_rag_evaluator/__init__.py +0 -0
- aiq/eval/tunable_rag_evaluator/evaluate.py +245 -0
- aiq/eval/tunable_rag_evaluator/register.py +52 -0
- aiq/eval/usage_stats.py +41 -0
- aiq/eval/utils/__init__.py +0 -0
- aiq/eval/utils/output_uploader.py +140 -0
- aiq/eval/utils/tqdm_position_registry.py +40 -0
- aiq/eval/utils/weave_eval.py +184 -0
- aiq/experimental/__init__.py +0 -0
- aiq/experimental/decorators/__init__.py +0 -0
- aiq/experimental/decorators/experimental_warning_decorator.py +130 -0
- aiq/experimental/test_time_compute/__init__.py +0 -0
- aiq/experimental/test_time_compute/editing/__init__.py +0 -0
- aiq/experimental/test_time_compute/editing/iterative_plan_refinement_editor.py +147 -0
- aiq/experimental/test_time_compute/editing/llm_as_a_judge_editor.py +204 -0
- aiq/experimental/test_time_compute/editing/motivation_aware_summarization.py +107 -0
- aiq/experimental/test_time_compute/functions/__init__.py +0 -0
- aiq/experimental/test_time_compute/functions/execute_score_select_function.py +105 -0
- aiq/experimental/test_time_compute/functions/its_tool_orchestration_function.py +205 -0
- aiq/experimental/test_time_compute/functions/its_tool_wrapper_function.py +146 -0
- aiq/experimental/test_time_compute/functions/plan_select_execute_function.py +224 -0
- aiq/experimental/test_time_compute/models/__init__.py +0 -0
- aiq/experimental/test_time_compute/models/editor_config.py +132 -0
- aiq/experimental/test_time_compute/models/scoring_config.py +112 -0
- aiq/experimental/test_time_compute/models/search_config.py +120 -0
- aiq/experimental/test_time_compute/models/selection_config.py +154 -0
- aiq/experimental/test_time_compute/models/stage_enums.py +43 -0
- aiq/experimental/test_time_compute/models/strategy_base.py +66 -0
- aiq/experimental/test_time_compute/models/tool_use_config.py +41 -0
- aiq/experimental/test_time_compute/models/ttc_item.py +48 -0
- aiq/experimental/test_time_compute/register.py +36 -0
- aiq/experimental/test_time_compute/scoring/__init__.py +0 -0
- aiq/experimental/test_time_compute/scoring/llm_based_agent_scorer.py +168 -0
- aiq/experimental/test_time_compute/scoring/llm_based_plan_scorer.py +168 -0
- aiq/experimental/test_time_compute/scoring/motivation_aware_scorer.py +111 -0
- aiq/experimental/test_time_compute/search/__init__.py +0 -0
- aiq/experimental/test_time_compute/search/multi_llm_planner.py +128 -0
- aiq/experimental/test_time_compute/search/multi_query_retrieval_search.py +122 -0
- aiq/experimental/test_time_compute/search/single_shot_multi_plan_planner.py +128 -0
- aiq/experimental/test_time_compute/selection/__init__.py +0 -0
- aiq/experimental/test_time_compute/selection/best_of_n_selector.py +63 -0
- aiq/experimental/test_time_compute/selection/llm_based_agent_output_selector.py +131 -0
- aiq/experimental/test_time_compute/selection/llm_based_output_merging_selector.py +159 -0
- aiq/experimental/test_time_compute/selection/llm_based_plan_selector.py +128 -0
- aiq/experimental/test_time_compute/selection/threshold_selector.py +58 -0
- aiq/front_ends/__init__.py +14 -0
- aiq/front_ends/console/__init__.py +14 -0
- aiq/front_ends/console/authentication_flow_handler.py +233 -0
- aiq/front_ends/console/console_front_end_config.py +32 -0
- aiq/front_ends/console/console_front_end_plugin.py +96 -0
- aiq/front_ends/console/register.py +25 -0
- aiq/front_ends/cron/__init__.py +14 -0
- aiq/front_ends/fastapi/__init__.py +14 -0
- aiq/front_ends/fastapi/auth_flow_handlers/__init__.py +0 -0
- aiq/front_ends/fastapi/auth_flow_handlers/http_flow_handler.py +27 -0
- aiq/front_ends/fastapi/auth_flow_handlers/websocket_flow_handler.py +107 -0
- aiq/front_ends/fastapi/fastapi_front_end_config.py +234 -0
- aiq/front_ends/fastapi/fastapi_front_end_controller.py +68 -0
- aiq/front_ends/fastapi/fastapi_front_end_plugin.py +116 -0
- aiq/front_ends/fastapi/fastapi_front_end_plugin_worker.py +1092 -0
- aiq/front_ends/fastapi/html_snippets/__init__.py +14 -0
- aiq/front_ends/fastapi/html_snippets/auth_code_grant_success.py +35 -0
- aiq/front_ends/fastapi/intermediate_steps_subscriber.py +80 -0
- aiq/front_ends/fastapi/job_store.py +183 -0
- aiq/front_ends/fastapi/main.py +72 -0
- aiq/front_ends/fastapi/message_handler.py +298 -0
- aiq/front_ends/fastapi/message_validator.py +345 -0
- aiq/front_ends/fastapi/register.py +25 -0
- aiq/front_ends/fastapi/response_helpers.py +195 -0
- aiq/front_ends/fastapi/step_adaptor.py +321 -0
- aiq/front_ends/mcp/__init__.py +14 -0
- aiq/front_ends/mcp/mcp_front_end_config.py +32 -0
- aiq/front_ends/mcp/mcp_front_end_plugin.py +93 -0
- aiq/front_ends/mcp/register.py +27 -0
- aiq/front_ends/mcp/tool_converter.py +242 -0
- aiq/front_ends/register.py +22 -0
- aiq/front_ends/simple_base/__init__.py +14 -0
- aiq/front_ends/simple_base/simple_front_end_plugin_base.py +54 -0
- aiq/llm/__init__.py +0 -0
- aiq/llm/aws_bedrock_llm.py +57 -0
- aiq/llm/nim_llm.py +46 -0
- aiq/llm/openai_llm.py +46 -0
- aiq/llm/register.py +23 -0
- aiq/llm/utils/__init__.py +14 -0
- aiq/llm/utils/env_config_value.py +94 -0
- aiq/llm/utils/error.py +17 -0
- aiq/memory/__init__.py +20 -0
- aiq/memory/interfaces.py +183 -0
- aiq/memory/models.py +112 -0
- aiq/meta/module_to_distro.json +3 -0
- aiq/meta/pypi.md +58 -0
- aiq/object_store/__init__.py +20 -0
- aiq/object_store/in_memory_object_store.py +76 -0
- aiq/object_store/interfaces.py +84 -0
- aiq/object_store/models.py +36 -0
- aiq/object_store/register.py +20 -0
- aiq/observability/__init__.py +14 -0
- aiq/observability/exporter/__init__.py +14 -0
- aiq/observability/exporter/base_exporter.py +449 -0
- aiq/observability/exporter/exporter.py +78 -0
- aiq/observability/exporter/file_exporter.py +33 -0
- aiq/observability/exporter/processing_exporter.py +322 -0
- aiq/observability/exporter/raw_exporter.py +52 -0
- aiq/observability/exporter/span_exporter.py +265 -0
- aiq/observability/exporter_manager.py +335 -0
- aiq/observability/mixin/__init__.py +14 -0
- aiq/observability/mixin/batch_config_mixin.py +26 -0
- aiq/observability/mixin/collector_config_mixin.py +23 -0
- aiq/observability/mixin/file_mixin.py +288 -0
- aiq/observability/mixin/file_mode.py +23 -0
- aiq/observability/mixin/resource_conflict_mixin.py +134 -0
- aiq/observability/mixin/serialize_mixin.py +61 -0
- aiq/observability/mixin/type_introspection_mixin.py +183 -0
- aiq/observability/processor/__init__.py +14 -0
- aiq/observability/processor/batching_processor.py +310 -0
- aiq/observability/processor/callback_processor.py +42 -0
- aiq/observability/processor/intermediate_step_serializer.py +28 -0
- aiq/observability/processor/processor.py +71 -0
- aiq/observability/register.py +96 -0
- aiq/observability/utils/__init__.py +14 -0
- aiq/observability/utils/dict_utils.py +236 -0
- aiq/observability/utils/time_utils.py +31 -0
- aiq/plugins/.namespace +1 -0
- aiq/profiler/__init__.py +0 -0
- aiq/profiler/calc/__init__.py +14 -0
- aiq/profiler/calc/calc_runner.py +627 -0
- aiq/profiler/calc/calculations.py +288 -0
- aiq/profiler/calc/data_models.py +188 -0
- aiq/profiler/calc/plot.py +345 -0
- aiq/profiler/callbacks/__init__.py +0 -0
- aiq/profiler/callbacks/agno_callback_handler.py +295 -0
- aiq/profiler/callbacks/base_callback_class.py +20 -0
- aiq/profiler/callbacks/langchain_callback_handler.py +290 -0
- aiq/profiler/callbacks/llama_index_callback_handler.py +205 -0
- aiq/profiler/callbacks/semantic_kernel_callback_handler.py +238 -0
- aiq/profiler/callbacks/token_usage_base_model.py +27 -0
- aiq/profiler/data_frame_row.py +51 -0
- aiq/profiler/data_models.py +24 -0
- aiq/profiler/decorators/__init__.py +0 -0
- aiq/profiler/decorators/framework_wrapper.py +131 -0
- aiq/profiler/decorators/function_tracking.py +254 -0
- aiq/profiler/forecasting/__init__.py +0 -0
- aiq/profiler/forecasting/config.py +18 -0
- aiq/profiler/forecasting/model_trainer.py +75 -0
- aiq/profiler/forecasting/models/__init__.py +22 -0
- aiq/profiler/forecasting/models/forecasting_base_model.py +40 -0
- aiq/profiler/forecasting/models/linear_model.py +196 -0
- aiq/profiler/forecasting/models/random_forest_regressor.py +268 -0
- aiq/profiler/inference_metrics_model.py +28 -0
- aiq/profiler/inference_optimization/__init__.py +0 -0
- aiq/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
- aiq/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +460 -0
- aiq/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +258 -0
- aiq/profiler/inference_optimization/data_models.py +386 -0
- aiq/profiler/inference_optimization/experimental/__init__.py +0 -0
- aiq/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +468 -0
- aiq/profiler/inference_optimization/experimental/prefix_span_analysis.py +405 -0
- aiq/profiler/inference_optimization/llm_metrics.py +212 -0
- aiq/profiler/inference_optimization/prompt_caching.py +163 -0
- aiq/profiler/inference_optimization/token_uniqueness.py +107 -0
- aiq/profiler/inference_optimization/workflow_runtimes.py +72 -0
- aiq/profiler/intermediate_property_adapter.py +102 -0
- aiq/profiler/profile_runner.py +473 -0
- aiq/profiler/utils.py +184 -0
- aiq/registry_handlers/__init__.py +0 -0
- aiq/registry_handlers/local/__init__.py +0 -0
- aiq/registry_handlers/local/local_handler.py +176 -0
- aiq/registry_handlers/local/register_local.py +37 -0
- aiq/registry_handlers/metadata_factory.py +60 -0
- aiq/registry_handlers/package_utils.py +567 -0
- aiq/registry_handlers/pypi/__init__.py +0 -0
- aiq/registry_handlers/pypi/pypi_handler.py +251 -0
- aiq/registry_handlers/pypi/register_pypi.py +40 -0
- aiq/registry_handlers/register.py +21 -0
- aiq/registry_handlers/registry_handler_base.py +157 -0
- aiq/registry_handlers/rest/__init__.py +0 -0
- aiq/registry_handlers/rest/register_rest.py +56 -0
- aiq/registry_handlers/rest/rest_handler.py +237 -0
- aiq/registry_handlers/schemas/__init__.py +0 -0
- aiq/registry_handlers/schemas/headers.py +42 -0
- aiq/registry_handlers/schemas/package.py +68 -0
- aiq/registry_handlers/schemas/publish.py +63 -0
- aiq/registry_handlers/schemas/pull.py +82 -0
- aiq/registry_handlers/schemas/remove.py +36 -0
- aiq/registry_handlers/schemas/search.py +91 -0
- aiq/registry_handlers/schemas/status.py +47 -0
- aiq/retriever/__init__.py +0 -0
- aiq/retriever/interface.py +37 -0
- aiq/retriever/milvus/__init__.py +14 -0
- aiq/retriever/milvus/register.py +81 -0
- aiq/retriever/milvus/retriever.py +228 -0
- aiq/retriever/models.py +74 -0
- aiq/retriever/nemo_retriever/__init__.py +14 -0
- aiq/retriever/nemo_retriever/register.py +60 -0
- aiq/retriever/nemo_retriever/retriever.py +190 -0
- aiq/retriever/register.py +22 -0
- aiq/runtime/__init__.py +14 -0
- aiq/runtime/loader.py +215 -0
- aiq/runtime/runner.py +190 -0
- aiq/runtime/session.py +158 -0
- aiq/runtime/user_metadata.py +130 -0
- aiq/settings/__init__.py +0 -0
- aiq/settings/global_settings.py +318 -0
- aiq/test/.namespace +1 -0
- aiq/tool/__init__.py +0 -0
- aiq/tool/chat_completion.py +74 -0
- aiq/tool/code_execution/README.md +151 -0
- aiq/tool/code_execution/__init__.py +0 -0
- aiq/tool/code_execution/code_sandbox.py +267 -0
- aiq/tool/code_execution/local_sandbox/.gitignore +1 -0
- aiq/tool/code_execution/local_sandbox/Dockerfile.sandbox +60 -0
- aiq/tool/code_execution/local_sandbox/__init__.py +13 -0
- aiq/tool/code_execution/local_sandbox/local_sandbox_server.py +198 -0
- aiq/tool/code_execution/local_sandbox/sandbox.requirements.txt +6 -0
- aiq/tool/code_execution/local_sandbox/start_local_sandbox.sh +50 -0
- aiq/tool/code_execution/register.py +74 -0
- aiq/tool/code_execution/test_code_execution_sandbox.py +414 -0
- aiq/tool/code_execution/utils.py +100 -0
- aiq/tool/datetime_tools.py +42 -0
- aiq/tool/document_search.py +141 -0
- aiq/tool/github_tools/__init__.py +0 -0
- aiq/tool/github_tools/create_github_commit.py +133 -0
- aiq/tool/github_tools/create_github_issue.py +87 -0
- aiq/tool/github_tools/create_github_pr.py +106 -0
- aiq/tool/github_tools/get_github_file.py +106 -0
- aiq/tool/github_tools/get_github_issue.py +166 -0
- aiq/tool/github_tools/get_github_pr.py +256 -0
- aiq/tool/github_tools/update_github_issue.py +100 -0
- aiq/tool/mcp/__init__.py +14 -0
- aiq/tool/mcp/exceptions.py +142 -0
- aiq/tool/mcp/mcp_client.py +255 -0
- aiq/tool/mcp/mcp_tool.py +96 -0
- aiq/tool/memory_tools/__init__.py +0 -0
- aiq/tool/memory_tools/add_memory_tool.py +79 -0
- aiq/tool/memory_tools/delete_memory_tool.py +67 -0
- aiq/tool/memory_tools/get_memory_tool.py +72 -0
- aiq/tool/nvidia_rag.py +95 -0
- aiq/tool/register.py +38 -0
- aiq/tool/retriever.py +89 -0
- aiq/tool/server_tools.py +66 -0
- aiq/utils/__init__.py +0 -0
- aiq/utils/data_models/__init__.py +0 -0
- aiq/utils/data_models/schema_validator.py +58 -0
- aiq/utils/debugging_utils.py +43 -0
- aiq/utils/dump_distro_mapping.py +32 -0
- aiq/utils/exception_handlers/__init__.py +0 -0
- aiq/utils/exception_handlers/automatic_retries.py +289 -0
- aiq/utils/exception_handlers/mcp.py +211 -0
- aiq/utils/exception_handlers/schemas.py +114 -0
- aiq/utils/io/__init__.py +0 -0
- aiq/utils/io/model_processing.py +28 -0
- aiq/utils/io/yaml_tools.py +119 -0
- aiq/utils/log_utils.py +37 -0
- aiq/utils/metadata_utils.py +74 -0
- aiq/utils/optional_imports.py +142 -0
- aiq/utils/producer_consumer_queue.py +178 -0
- aiq/utils/reactive/__init__.py +0 -0
- aiq/utils/reactive/base/__init__.py +0 -0
- aiq/utils/reactive/base/observable_base.py +65 -0
- aiq/utils/reactive/base/observer_base.py +55 -0
- aiq/utils/reactive/base/subject_base.py +79 -0
- aiq/utils/reactive/observable.py +59 -0
- aiq/utils/reactive/observer.py +76 -0
- aiq/utils/reactive/subject.py +131 -0
- aiq/utils/reactive/subscription.py +49 -0
- aiq/utils/settings/__init__.py +0 -0
- aiq/utils/settings/global_settings.py +197 -0
- aiq/utils/string_utils.py +38 -0
- aiq/utils/type_converter.py +290 -0
- aiq/utils/type_utils.py +484 -0
- aiq/utils/url_utils.py +27 -0
- nvidia_nat-1.2.0rc5.dist-info/METADATA +363 -0
- nvidia_nat-1.2.0rc5.dist-info/RECORD +435 -0
- nvidia_nat-1.2.0rc5.dist-info/WHEEL +5 -0
- nvidia_nat-1.2.0rc5.dist-info/entry_points.txt +20 -0
- nvidia_nat-1.2.0rc5.dist-info/licenses/LICENSE-3rd-party.txt +3686 -0
- nvidia_nat-1.2.0rc5.dist-info/licenses/LICENSE.md +201 -0
- nvidia_nat-1.2.0rc5.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,449 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import asyncio
|
|
17
|
+
import copy
|
|
18
|
+
import logging
|
|
19
|
+
import weakref
|
|
20
|
+
from abc import abstractmethod
|
|
21
|
+
from collections.abc import AsyncGenerator
|
|
22
|
+
from collections.abc import Callable
|
|
23
|
+
from contextlib import asynccontextmanager
|
|
24
|
+
from typing import Any
|
|
25
|
+
from typing import Generic
|
|
26
|
+
from typing import TypeVar
|
|
27
|
+
from typing import overload
|
|
28
|
+
|
|
29
|
+
from aiq.builder.context import AIQContextState
|
|
30
|
+
from aiq.data_models.intermediate_step import IntermediateStep
|
|
31
|
+
from aiq.observability.exporter.exporter import Exporter
|
|
32
|
+
from aiq.utils.reactive.subject import Subject
|
|
33
|
+
from aiq.utils.type_utils import override
|
|
34
|
+
|
|
35
|
+
logger = logging.getLogger(__name__)
|
|
36
|
+
|
|
37
|
+
IsolatedAttributeT = TypeVar('IsolatedAttributeT')
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
class IsolatedAttribute(Generic[IsolatedAttributeT]):
|
|
41
|
+
"""Descriptor for copy-on-write isolation.
|
|
42
|
+
|
|
43
|
+
This descriptor uses Python's descriptor protocol to automatically manage
|
|
44
|
+
attribute isolation during object copying. It enables efficient concurrent
|
|
45
|
+
execution by sharing expensive resources while isolating mutable state.
|
|
46
|
+
|
|
47
|
+
Performance Note: This pattern shares expensive resources (HTTP clients,
|
|
48
|
+
auth headers) while isolating cheap mutable state (task sets, events).
|
|
49
|
+
Tasks are tracked for monitoring but don't block shutdown - they complete
|
|
50
|
+
asynchronously in the event loop. Critical for high-throughput concurrent execution.
|
|
51
|
+
|
|
52
|
+
Implementation Note: Uses Python descriptor protocol (__get__, __set__, __set_name__)
|
|
53
|
+
for automatic attribute isolation on object copying.
|
|
54
|
+
|
|
55
|
+
Example:
|
|
56
|
+
class MyExporter(BaseExporter):
|
|
57
|
+
# Expensive HTTP client shared across instances
|
|
58
|
+
_client = expensive_http_client
|
|
59
|
+
|
|
60
|
+
# Cheap mutable state isolated per instance
|
|
61
|
+
_tasks: IsolatedAttribute[set] = IsolatedAttribute(set)
|
|
62
|
+
|
|
63
|
+
exporter1 = MyExporter(endpoint="https://api.service.com")
|
|
64
|
+
exporter2 = exporter1.create_isolated_instance(context)
|
|
65
|
+
# exporter2 shares _client but has isolated _tasks tracking
|
|
66
|
+
"""
|
|
67
|
+
|
|
68
|
+
def __init__(self, factory: Callable[[], IsolatedAttributeT]):
|
|
69
|
+
self.factory = factory
|
|
70
|
+
self.name: str | None = None
|
|
71
|
+
self._private_name: str
|
|
72
|
+
|
|
73
|
+
def __set_name__(self, owner, name):
|
|
74
|
+
self.name = name
|
|
75
|
+
self._private_name = f"__{name}_isolated"
|
|
76
|
+
|
|
77
|
+
@overload
|
|
78
|
+
def __get__(self, obj: None, objtype: type[Any] | None = None) -> "IsolatedAttribute[IsolatedAttributeT]":
|
|
79
|
+
...
|
|
80
|
+
|
|
81
|
+
@overload
|
|
82
|
+
def __get__(self, obj: Any, objtype: type[Any] | None = None) -> IsolatedAttributeT:
|
|
83
|
+
...
|
|
84
|
+
|
|
85
|
+
def __get__(self, obj, objtype=None):
|
|
86
|
+
if obj is None:
|
|
87
|
+
return self
|
|
88
|
+
|
|
89
|
+
if not hasattr(obj, self._private_name):
|
|
90
|
+
setattr(obj, self._private_name, self.factory())
|
|
91
|
+
|
|
92
|
+
return getattr(obj, self._private_name)
|
|
93
|
+
|
|
94
|
+
def __set__(self, obj, value: IsolatedAttributeT):
|
|
95
|
+
setattr(obj, self._private_name, value)
|
|
96
|
+
|
|
97
|
+
def reset_for_copy(self, obj):
|
|
98
|
+
"""Reset the attribute for a copied object."""
|
|
99
|
+
if hasattr(obj, self._private_name):
|
|
100
|
+
delattr(obj, self._private_name)
|
|
101
|
+
|
|
102
|
+
|
|
103
|
+
class BaseExporter(Exporter):
|
|
104
|
+
"""Abstract base class for event exporters with isolated copy support.
|
|
105
|
+
|
|
106
|
+
This class provides the foundation for creating event exporters that can handle
|
|
107
|
+
concurrent execution through copy-on-write isolation. It manages the lifecycle
|
|
108
|
+
of event subscriptions and provides hooks for processing events.
|
|
109
|
+
|
|
110
|
+
The class supports isolation for concurrent execution by automatically resetting
|
|
111
|
+
mutable state when creating isolated copies using descriptors.
|
|
112
|
+
|
|
113
|
+
Performance Design:
|
|
114
|
+
- Export tasks run asynchronously in the event loop background
|
|
115
|
+
- stop() method does not wait for background tasks to complete
|
|
116
|
+
- Tasks are tracked for monitoring but cleaned up automatically
|
|
117
|
+
- This keeps observability "off the hot path" for optimal performance
|
|
118
|
+
|
|
119
|
+
Args:
|
|
120
|
+
context_state (AIQContextState, optional): The context state to use for the exporter. Defaults to None.
|
|
121
|
+
"""
|
|
122
|
+
|
|
123
|
+
# Class-level tracking for debugging and monitoring
|
|
124
|
+
_instance_count: int = 0
|
|
125
|
+
_active_instances: set[weakref.ref] = set()
|
|
126
|
+
_isolated_instances: set[weakref.ref] = set()
|
|
127
|
+
|
|
128
|
+
# Use descriptors for automatic isolation with proper generic typing
|
|
129
|
+
_tasks: IsolatedAttribute[set[asyncio.Task]] = IsolatedAttribute(set)
|
|
130
|
+
_ready_event: IsolatedAttribute[asyncio.Event] = IsolatedAttribute(asyncio.Event)
|
|
131
|
+
_shutdown_event: IsolatedAttribute[asyncio.Event] = IsolatedAttribute(asyncio.Event)
|
|
132
|
+
|
|
133
|
+
def __init__(self, context_state: AIQContextState | None = None):
|
|
134
|
+
"""Initialize the BaseExporter."""
|
|
135
|
+
if context_state is None:
|
|
136
|
+
context_state = AIQContextState.get()
|
|
137
|
+
|
|
138
|
+
self._context_state = context_state
|
|
139
|
+
self._subscription = None
|
|
140
|
+
self._running = False
|
|
141
|
+
# Get the event loop (set to None if not available, will be set later)
|
|
142
|
+
self._loop = None
|
|
143
|
+
self._is_isolated_instance = False
|
|
144
|
+
|
|
145
|
+
# Track instance creation
|
|
146
|
+
BaseExporter._instance_count += 1
|
|
147
|
+
BaseExporter._active_instances.add(weakref.ref(self, self._cleanup_instance_tracking))
|
|
148
|
+
|
|
149
|
+
# Note: _tasks, _ready_event, _shutdown_event are descriptors
|
|
150
|
+
|
|
151
|
+
@classmethod
|
|
152
|
+
def _cleanup_instance_tracking(cls, ref):
|
|
153
|
+
"""Cleanup callback for weakref when instance is garbage collected."""
|
|
154
|
+
cls._active_instances.discard(ref)
|
|
155
|
+
cls._isolated_instances.discard(ref)
|
|
156
|
+
|
|
157
|
+
@classmethod
|
|
158
|
+
def get_active_instance_count(cls) -> int:
|
|
159
|
+
"""Get the number of active BaseExporter instances.
|
|
160
|
+
|
|
161
|
+
Returns:
|
|
162
|
+
int: Number of active instances (cleaned up automatically via weakref)
|
|
163
|
+
"""
|
|
164
|
+
# Clean up dead references automatically via weakref callback
|
|
165
|
+
return len(cls._active_instances)
|
|
166
|
+
|
|
167
|
+
@classmethod
|
|
168
|
+
def get_isolated_instance_count(cls) -> int:
|
|
169
|
+
"""Get the number of active isolated BaseExporter instances.
|
|
170
|
+
|
|
171
|
+
Returns:
|
|
172
|
+
int: Number of active isolated instances
|
|
173
|
+
"""
|
|
174
|
+
return len(cls._isolated_instances)
|
|
175
|
+
|
|
176
|
+
@classmethod
|
|
177
|
+
def log_instance_stats(cls) -> None:
|
|
178
|
+
"""Log current instance statistics for debugging."""
|
|
179
|
+
total = cls.get_active_instance_count()
|
|
180
|
+
isolated = cls.get_isolated_instance_count()
|
|
181
|
+
original = total - isolated
|
|
182
|
+
|
|
183
|
+
logger.info("BaseExporter instances - Total: %d, Original: %d, Isolated: %d", total, original, isolated)
|
|
184
|
+
|
|
185
|
+
if isolated > 50: # Warn if we have many isolated instances
|
|
186
|
+
warning_msg = (f"High number of isolated BaseExporter instances ({isolated}). "
|
|
187
|
+
"Check for potential memory leaks.")
|
|
188
|
+
logger.warning(warning_msg)
|
|
189
|
+
|
|
190
|
+
def __del__(self):
|
|
191
|
+
"""Destructor with memory leak warnings.
|
|
192
|
+
|
|
193
|
+
Warns if the exporter is being garbage collected while still running,
|
|
194
|
+
which indicates stop() was never called. Task tracking is used for
|
|
195
|
+
diagnostics but stop() doesn't wait for tasks to complete.
|
|
196
|
+
|
|
197
|
+
This method is defensive against partial initialization - if the object
|
|
198
|
+
failed to initialize completely, some attributes may not exist.
|
|
199
|
+
"""
|
|
200
|
+
try:
|
|
201
|
+
# Check if object was fully initialized before checking for active resources
|
|
202
|
+
is_running = getattr(self, '_running', False)
|
|
203
|
+
has_tasks = hasattr(self, '__tasks_isolated') and bool(getattr(self, '_tasks', None))
|
|
204
|
+
|
|
205
|
+
if is_running or has_tasks:
|
|
206
|
+
# Safely get name and task count
|
|
207
|
+
try:
|
|
208
|
+
name = self.name
|
|
209
|
+
except (AttributeError, TypeError):
|
|
210
|
+
# Fallback if name property fails due to missing attributes
|
|
211
|
+
name = f"{self.__class__.__name__} (partially initialized)"
|
|
212
|
+
|
|
213
|
+
task_count = len(self._tasks) if has_tasks else 0
|
|
214
|
+
|
|
215
|
+
logger.warning(
|
|
216
|
+
"%s: Exporter being garbage collected with active resources. "
|
|
217
|
+
"Running: %s, Tasks: %s. "
|
|
218
|
+
"Call stop() explicitly to avoid memory leaks.",
|
|
219
|
+
name,
|
|
220
|
+
is_running,
|
|
221
|
+
task_count)
|
|
222
|
+
|
|
223
|
+
except Exception as e:
|
|
224
|
+
# Last resort: log that cleanup had issues but don't raise
|
|
225
|
+
# This prevents exceptions during garbage collection
|
|
226
|
+
try:
|
|
227
|
+
class_name = self.__class__.__name__
|
|
228
|
+
logger.debug("Exception during %s cleanup: %s", class_name, e)
|
|
229
|
+
except Exception:
|
|
230
|
+
# If even logging fails, silently ignore to prevent GC issues
|
|
231
|
+
pass
|
|
232
|
+
|
|
233
|
+
@property
|
|
234
|
+
def name(self) -> str:
|
|
235
|
+
"""Get the name of the exporter.
|
|
236
|
+
|
|
237
|
+
Returns:
|
|
238
|
+
str: The unique name of the exporter.
|
|
239
|
+
"""
|
|
240
|
+
try:
|
|
241
|
+
suffix = " (isolated)" if getattr(self, '_is_isolated_instance', False) else ""
|
|
242
|
+
return f"{self.__class__.__name__}{suffix}"
|
|
243
|
+
except AttributeError:
|
|
244
|
+
# Fallback for partially initialized objects
|
|
245
|
+
return f"{self.__class__.__name__} (partial)"
|
|
246
|
+
|
|
247
|
+
@property
|
|
248
|
+
def is_isolated_instance(self) -> bool:
|
|
249
|
+
"""Check if this is an isolated instance.
|
|
250
|
+
|
|
251
|
+
Returns:
|
|
252
|
+
bool: True if this is an isolated instance, False otherwise
|
|
253
|
+
"""
|
|
254
|
+
return self._is_isolated_instance
|
|
255
|
+
|
|
256
|
+
@abstractmethod
|
|
257
|
+
def export(self, event: IntermediateStep) -> None:
|
|
258
|
+
"""This method is called on each event from the event stream to initiate the trace export.
|
|
259
|
+
|
|
260
|
+
This is the base implementation that can be overridden by subclasses.
|
|
261
|
+
By default, it does nothing - subclasses should implement their specific export logic.
|
|
262
|
+
|
|
263
|
+
Args:
|
|
264
|
+
event (IntermediateStep): The event to be exported.
|
|
265
|
+
"""
|
|
266
|
+
pass
|
|
267
|
+
|
|
268
|
+
@override
|
|
269
|
+
def on_error(self, exc: Exception) -> None:
|
|
270
|
+
"""Handle an error in the event subscription.
|
|
271
|
+
|
|
272
|
+
Args:
|
|
273
|
+
exc (Exception): The error to handle.
|
|
274
|
+
"""
|
|
275
|
+
logger.error("Error in event subscription: %s", exc, exc_info=True)
|
|
276
|
+
|
|
277
|
+
@override
|
|
278
|
+
def on_complete(self) -> None:
|
|
279
|
+
"""Handle the completion of the event stream.
|
|
280
|
+
|
|
281
|
+
This method is called when the event stream is complete.
|
|
282
|
+
"""
|
|
283
|
+
logger.info("Event stream completed. No more events will arrive.")
|
|
284
|
+
|
|
285
|
+
def _start(self) -> Subject | None:
|
|
286
|
+
"""Start the exporter.
|
|
287
|
+
|
|
288
|
+
Returns:
|
|
289
|
+
Subject | None: The subject to subscribe to.
|
|
290
|
+
"""
|
|
291
|
+
subject = self._context_state.event_stream.get()
|
|
292
|
+
if subject is None:
|
|
293
|
+
return None
|
|
294
|
+
|
|
295
|
+
if not hasattr(subject, 'subscribe'):
|
|
296
|
+
logger.error("Event stream subject does not support subscription")
|
|
297
|
+
return None
|
|
298
|
+
|
|
299
|
+
def on_next_wrapper(event: IntermediateStep) -> None:
|
|
300
|
+
self.export(event)
|
|
301
|
+
|
|
302
|
+
self._subscription = subject.subscribe(
|
|
303
|
+
on_next=on_next_wrapper,
|
|
304
|
+
on_error=self.on_error,
|
|
305
|
+
on_complete=self.on_complete,
|
|
306
|
+
)
|
|
307
|
+
|
|
308
|
+
self._running = True
|
|
309
|
+
self._ready_event.set()
|
|
310
|
+
return subject
|
|
311
|
+
|
|
312
|
+
async def _pre_start(self):
|
|
313
|
+
"""Called before the exporter starts."""
|
|
314
|
+
pass
|
|
315
|
+
|
|
316
|
+
@override
|
|
317
|
+
@asynccontextmanager
|
|
318
|
+
async def start(self) -> AsyncGenerator[None]:
|
|
319
|
+
"""Start the exporter and yield control to the caller."""
|
|
320
|
+
try:
|
|
321
|
+
await self._pre_start()
|
|
322
|
+
|
|
323
|
+
if self._running:
|
|
324
|
+
logger.debug("Listener already running.")
|
|
325
|
+
yield
|
|
326
|
+
return
|
|
327
|
+
|
|
328
|
+
subject = self._start()
|
|
329
|
+
if subject is None:
|
|
330
|
+
logger.warning("No event stream available.")
|
|
331
|
+
yield
|
|
332
|
+
return
|
|
333
|
+
|
|
334
|
+
yield # let the caller do their workflow
|
|
335
|
+
|
|
336
|
+
finally:
|
|
337
|
+
await self.stop()
|
|
338
|
+
|
|
339
|
+
async def _cleanup(self):
|
|
340
|
+
"""Clean up any resources."""
|
|
341
|
+
pass
|
|
342
|
+
|
|
343
|
+
async def _cancel_tasks(self):
|
|
344
|
+
"""Cancel all scheduled tasks.
|
|
345
|
+
|
|
346
|
+
Note: This method is NOT called during normal stop() operation for performance.
|
|
347
|
+
It's available for special cases where explicit task completion is needed.
|
|
348
|
+
"""
|
|
349
|
+
tasks_to_cancel = set(self._tasks)
|
|
350
|
+
for task in tasks_to_cancel:
|
|
351
|
+
if not task.done():
|
|
352
|
+
task.cancel()
|
|
353
|
+
try:
|
|
354
|
+
await task
|
|
355
|
+
except asyncio.CancelledError:
|
|
356
|
+
pass
|
|
357
|
+
except Exception as e:
|
|
358
|
+
logger.warning("Error while canceling task %s: %s", task.get_name(), e)
|
|
359
|
+
|
|
360
|
+
async def _wait_for_tasks(self, timeout: float = 5.0):
|
|
361
|
+
"""Wait for all tracked tasks to complete with a timeout.
|
|
362
|
+
|
|
363
|
+
Note: This method is NOT called during normal stop() operation for performance.
|
|
364
|
+
It's available for special cases where explicit task completion is needed.
|
|
365
|
+
|
|
366
|
+
Args:
|
|
367
|
+
timeout (float, optional): The timeout in seconds. Defaults to 5.0.
|
|
368
|
+
"""
|
|
369
|
+
if not self._tasks:
|
|
370
|
+
return
|
|
371
|
+
|
|
372
|
+
try:
|
|
373
|
+
# Wait for all tasks to complete with a timeout
|
|
374
|
+
await asyncio.wait_for(asyncio.gather(*self._tasks, return_exceptions=True), timeout=timeout)
|
|
375
|
+
except asyncio.TimeoutError:
|
|
376
|
+
logger.warning("%s: Some tasks did not complete within %s seconds", self.name, timeout)
|
|
377
|
+
except Exception as e:
|
|
378
|
+
logger.error("%s: Error while waiting for tasks: %s", self.name, e, exc_info=True)
|
|
379
|
+
|
|
380
|
+
@override
|
|
381
|
+
async def stop(self):
|
|
382
|
+
"""Stop the exporter immediately without waiting for background tasks.
|
|
383
|
+
|
|
384
|
+
This method performs fast shutdown by:
|
|
385
|
+
1. Setting running=False to prevent new export tasks
|
|
386
|
+
2. Signaling shutdown to waiting code
|
|
387
|
+
3. Cleaning up subscriptions and resources
|
|
388
|
+
4. Clearing task tracking (tasks continue in event loop)
|
|
389
|
+
|
|
390
|
+
Performance: Does not block waiting for background export tasks to complete.
|
|
391
|
+
Background tasks will finish asynchronously and clean themselves up.
|
|
392
|
+
|
|
393
|
+
Note: This method is called when the exporter is no longer needed.
|
|
394
|
+
"""
|
|
395
|
+
if not self._running:
|
|
396
|
+
return
|
|
397
|
+
|
|
398
|
+
self._running = False
|
|
399
|
+
self._shutdown_event.set()
|
|
400
|
+
|
|
401
|
+
await self._cleanup()
|
|
402
|
+
|
|
403
|
+
if self._subscription:
|
|
404
|
+
self._subscription.unsubscribe()
|
|
405
|
+
self._subscription = None
|
|
406
|
+
|
|
407
|
+
self._tasks.clear()
|
|
408
|
+
|
|
409
|
+
async def wait_ready(self):
|
|
410
|
+
"""Wait for the exporter to be ready.
|
|
411
|
+
|
|
412
|
+
This method is called when the exporter is ready to export events.
|
|
413
|
+
"""
|
|
414
|
+
await self._ready_event.wait()
|
|
415
|
+
|
|
416
|
+
def create_isolated_instance(self, context_state: AIQContextState) -> "BaseExporter":
|
|
417
|
+
"""Create an isolated copy with automatic descriptor-based state reset.
|
|
418
|
+
|
|
419
|
+
This method creates a shallow copy that shares expensive resources
|
|
420
|
+
(HTTP clients, auth headers) while isolating mutable state through
|
|
421
|
+
the IsolatedAttribute descriptor pattern.
|
|
422
|
+
|
|
423
|
+
Args:
|
|
424
|
+
context_state: The isolated context state for the new instance
|
|
425
|
+
|
|
426
|
+
Returns:
|
|
427
|
+
BaseExporter: Isolated instance sharing expensive resources
|
|
428
|
+
"""
|
|
429
|
+
# Create shallow copy
|
|
430
|
+
isolated_instance = copy.copy(self)
|
|
431
|
+
|
|
432
|
+
# Reset context state
|
|
433
|
+
isolated_instance._context_state = context_state
|
|
434
|
+
|
|
435
|
+
# Mark as isolated instance and track it
|
|
436
|
+
isolated_instance._is_isolated_instance = True
|
|
437
|
+
BaseExporter._isolated_instances.add(weakref.ref(isolated_instance, self._cleanup_instance_tracking))
|
|
438
|
+
|
|
439
|
+
# Reset IsolatedAttribute descriptors automatically
|
|
440
|
+
for attr_name in dir(type(self)):
|
|
441
|
+
attr_value = getattr(type(self), attr_name, None)
|
|
442
|
+
if isinstance(attr_value, IsolatedAttribute):
|
|
443
|
+
attr_value.reset_for_copy(isolated_instance)
|
|
444
|
+
|
|
445
|
+
# Reset basic attributes that aren't descriptors but need isolation
|
|
446
|
+
isolated_instance._subscription = None
|
|
447
|
+
isolated_instance._running = False
|
|
448
|
+
|
|
449
|
+
return isolated_instance
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import logging
|
|
17
|
+
from abc import ABC
|
|
18
|
+
from abc import abstractmethod
|
|
19
|
+
from collections.abc import AsyncGenerator
|
|
20
|
+
|
|
21
|
+
from aiq.data_models.intermediate_step import IntermediateStep
|
|
22
|
+
|
|
23
|
+
logger = logging.getLogger(__name__)
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
class Exporter(ABC):
|
|
27
|
+
|
|
28
|
+
@abstractmethod
|
|
29
|
+
async def start(self) -> AsyncGenerator[None]:
|
|
30
|
+
"""Subscribes to event stream and starts the exporter.
|
|
31
|
+
|
|
32
|
+
This is an async context manager that should be used with 'async with'.
|
|
33
|
+
The exporter is automatically stopped when exiting the context.
|
|
34
|
+
|
|
35
|
+
Usage::
|
|
36
|
+
|
|
37
|
+
.. code-block:: python
|
|
38
|
+
|
|
39
|
+
async with exporter.start():
|
|
40
|
+
# Exporter is now running and subscribed to events
|
|
41
|
+
# Your workflow code here
|
|
42
|
+
pass
|
|
43
|
+
|
|
44
|
+
Note:
|
|
45
|
+
Implementations should use the @asynccontextmanager decorator.
|
|
46
|
+
"""
|
|
47
|
+
pass
|
|
48
|
+
|
|
49
|
+
@abstractmethod
|
|
50
|
+
async def stop(self) -> None:
|
|
51
|
+
"""Unsubscribes to the event stream and stops the exporter."""
|
|
52
|
+
pass
|
|
53
|
+
|
|
54
|
+
@abstractmethod
|
|
55
|
+
def export(self, event: IntermediateStep) -> None:
|
|
56
|
+
"""This method is called on each event from the event stream to initiate the trace export.
|
|
57
|
+
|
|
58
|
+
Args:
|
|
59
|
+
event (IntermediateStep): The event to be exported.
|
|
60
|
+
"""
|
|
61
|
+
pass
|
|
62
|
+
|
|
63
|
+
@abstractmethod
|
|
64
|
+
def on_error(self, exc: Exception) -> None:
|
|
65
|
+
"""Handle an error in the event subscription.
|
|
66
|
+
|
|
67
|
+
Args:
|
|
68
|
+
exc (Exception): The error to handle.
|
|
69
|
+
"""
|
|
70
|
+
pass
|
|
71
|
+
|
|
72
|
+
@abstractmethod
|
|
73
|
+
def on_complete(self) -> None:
|
|
74
|
+
"""Handle the completion of the event stream.
|
|
75
|
+
|
|
76
|
+
This method is called when the event stream is complete.
|
|
77
|
+
"""
|
|
78
|
+
pass
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import logging
|
|
17
|
+
|
|
18
|
+
from aiq.builder.context import AIQContextState
|
|
19
|
+
from aiq.data_models.intermediate_step import IntermediateStep
|
|
20
|
+
from aiq.observability.exporter.raw_exporter import RawExporter
|
|
21
|
+
from aiq.observability.mixin.file_mixin import FileExportMixin
|
|
22
|
+
from aiq.observability.processor.intermediate_step_serializer import IntermediateStepSerializer
|
|
23
|
+
|
|
24
|
+
logger = logging.getLogger(__name__)
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
class FileExporter(FileExportMixin, RawExporter[IntermediateStep, str]): # pylint: disable=R0901
|
|
28
|
+
"""A File exporter that exports telemetry traces to a local file."""
|
|
29
|
+
|
|
30
|
+
def __init__(self, context_state: AIQContextState | None = None, **file_kwargs):
|
|
31
|
+
super().__init__(context_state=context_state, **file_kwargs)
|
|
32
|
+
self._processor = IntermediateStepSerializer()
|
|
33
|
+
self.add_processor(self._processor)
|