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,142 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2024-2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import importlib
|
|
17
|
+
import logging
|
|
18
|
+
from types import ModuleType
|
|
19
|
+
|
|
20
|
+
logger = logging.getLogger(__name__)
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
class OptionalImportError(Exception):
|
|
24
|
+
"""Raised when an optional import fails."""
|
|
25
|
+
|
|
26
|
+
def __init__(self, module_name: str, additional_message: str = ""):
|
|
27
|
+
super().__init__(f"Optional dependency '{module_name}' is not installed. {additional_message}")
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
class TelemetryOptionalImportError(OptionalImportError):
|
|
31
|
+
"""Raised when an optional import of telemetry dependencies fails."""
|
|
32
|
+
|
|
33
|
+
def __init__(self, module_name: str):
|
|
34
|
+
super().__init__(
|
|
35
|
+
module_name,
|
|
36
|
+
"But the configuration file contains tracing exporters. "
|
|
37
|
+
"If you want to use this feature, please install it with: uv pip install -e '.[telemetry]'",
|
|
38
|
+
)
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
def optional_import(module_name: str) -> ModuleType:
|
|
42
|
+
"""Attempt to import a module, raising OptionalImportError if it fails."""
|
|
43
|
+
try:
|
|
44
|
+
return importlib.import_module(module_name)
|
|
45
|
+
except ImportError as e:
|
|
46
|
+
raise OptionalImportError(module_name) from e
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
def telemetry_optional_import(module_name: str) -> ModuleType:
|
|
50
|
+
"""Attempt to import a module, raising TelemetryOptionalImportError if it fails."""
|
|
51
|
+
try:
|
|
52
|
+
return importlib.import_module(module_name)
|
|
53
|
+
except ImportError as e:
|
|
54
|
+
raise TelemetryOptionalImportError(module_name) from e
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
def try_import_opentelemetry() -> ModuleType:
|
|
58
|
+
"""Get the opentelemetry module if available."""
|
|
59
|
+
return telemetry_optional_import("opentelemetry")
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
def try_import_phoenix() -> ModuleType:
|
|
63
|
+
"""Get the phoenix module if available."""
|
|
64
|
+
return telemetry_optional_import("phoenix")
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
# Dummy OpenTelemetry classes for when the package is not available
|
|
68
|
+
class DummySpan:
|
|
69
|
+
"""Dummy span class that does nothing when OpenTelemetry is not available."""
|
|
70
|
+
|
|
71
|
+
def __init__(self, *args, **kwargs):
|
|
72
|
+
pass
|
|
73
|
+
|
|
74
|
+
def end(self, *args, **kwargs):
|
|
75
|
+
pass
|
|
76
|
+
|
|
77
|
+
def set_attribute(self, *args, **kwargs):
|
|
78
|
+
pass
|
|
79
|
+
|
|
80
|
+
|
|
81
|
+
class DummyTracer:
|
|
82
|
+
"""Dummy tracer class that returns dummy spans."""
|
|
83
|
+
|
|
84
|
+
def start_span(self, *args, **kwargs):
|
|
85
|
+
return DummySpan()
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
class DummyTracerProvider:
|
|
89
|
+
"""Dummy tracer provider that returns dummy tracers."""
|
|
90
|
+
|
|
91
|
+
@staticmethod
|
|
92
|
+
def get_tracer(*args, **kwargs):
|
|
93
|
+
return DummyTracer()
|
|
94
|
+
|
|
95
|
+
@staticmethod
|
|
96
|
+
def add_span_processor(*args, **kwargs):
|
|
97
|
+
pass
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
class DummyTrace:
|
|
101
|
+
"""Dummy trace module that returns dummy tracer providers."""
|
|
102
|
+
|
|
103
|
+
@staticmethod
|
|
104
|
+
def get_tracer_provider():
|
|
105
|
+
return DummyTracerProvider()
|
|
106
|
+
|
|
107
|
+
@staticmethod
|
|
108
|
+
def set_tracer_provider(*args, **kwargs):
|
|
109
|
+
pass
|
|
110
|
+
|
|
111
|
+
@staticmethod
|
|
112
|
+
def get_tracer(*args, **kwargs):
|
|
113
|
+
return DummyTracer()
|
|
114
|
+
|
|
115
|
+
|
|
116
|
+
class DummySpanExporter:
|
|
117
|
+
"""Dummy span exporter that does nothing."""
|
|
118
|
+
|
|
119
|
+
@staticmethod
|
|
120
|
+
def export(*args, **kwargs):
|
|
121
|
+
pass
|
|
122
|
+
|
|
123
|
+
@staticmethod
|
|
124
|
+
def shutdown(*args, **kwargs):
|
|
125
|
+
pass
|
|
126
|
+
|
|
127
|
+
|
|
128
|
+
class DummyBatchSpanProcessor:
|
|
129
|
+
"""Dummy implementation of BatchSpanProcessor for when OpenTelemetry is not available."""
|
|
130
|
+
|
|
131
|
+
def __init__(self, *args, **kwargs):
|
|
132
|
+
pass
|
|
133
|
+
|
|
134
|
+
@staticmethod
|
|
135
|
+
def shutdown(*args, **kwargs):
|
|
136
|
+
pass
|
|
137
|
+
|
|
138
|
+
|
|
139
|
+
# Dummy functions for when OpenTelemetry is not available
|
|
140
|
+
def dummy_set_span_in_context(*args, **kwargs) -> None:
|
|
141
|
+
"""Dummy function that does nothing."""
|
|
142
|
+
return None
|
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import asyncio
|
|
17
|
+
import typing
|
|
18
|
+
|
|
19
|
+
_T = typing.TypeVar("_T")
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
class QueueClosed(Exception):
|
|
23
|
+
'Exception raised when the queue is closed'
|
|
24
|
+
pass
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
class AsyncIOProducerConsumerQueue(asyncio.Queue, typing.Generic[_T]):
|
|
28
|
+
"""
|
|
29
|
+
Custom queue.Queue implementation which supports closing and uses recursive locks
|
|
30
|
+
"""
|
|
31
|
+
|
|
32
|
+
def __init__(self, maxsize=0) -> None:
|
|
33
|
+
super().__init__(maxsize=maxsize)
|
|
34
|
+
|
|
35
|
+
self._closed = asyncio.Event()
|
|
36
|
+
self._is_closed = False
|
|
37
|
+
|
|
38
|
+
async def __aiter__(self):
|
|
39
|
+
try:
|
|
40
|
+
while True:
|
|
41
|
+
yield await self.get()
|
|
42
|
+
except QueueClosed:
|
|
43
|
+
return
|
|
44
|
+
|
|
45
|
+
async def join(self):
|
|
46
|
+
"""Block until all items in the queue have been gotten and processed.
|
|
47
|
+
|
|
48
|
+
The count of unfinished tasks goes up whenever an item is added to the
|
|
49
|
+
queue. The count goes down whenever a consumer calls task_done() to
|
|
50
|
+
indicate that the item was retrieved and all work on it is complete.
|
|
51
|
+
When the count of unfinished tasks drops to zero, join() unblocks.
|
|
52
|
+
"""
|
|
53
|
+
|
|
54
|
+
# First wait for the closed flag to be set
|
|
55
|
+
await self._closed.wait()
|
|
56
|
+
|
|
57
|
+
if self._unfinished_tasks > 0:
|
|
58
|
+
await self._finished.wait()
|
|
59
|
+
|
|
60
|
+
async def put(self, item):
|
|
61
|
+
"""Put an item into the queue.
|
|
62
|
+
|
|
63
|
+
Put an item into the queue. If the queue is full, wait until a free
|
|
64
|
+
slot is available before adding item.
|
|
65
|
+
"""
|
|
66
|
+
while self.full() and not self._is_closed:
|
|
67
|
+
putter = self._get_loop().create_future()
|
|
68
|
+
self._putters.append(putter)
|
|
69
|
+
try:
|
|
70
|
+
await putter
|
|
71
|
+
except Exception:
|
|
72
|
+
putter.cancel() # Just in case putter is not done yet.
|
|
73
|
+
try:
|
|
74
|
+
# Clean self._putters from canceled putters.
|
|
75
|
+
self._putters.remove(putter)
|
|
76
|
+
except ValueError:
|
|
77
|
+
# The putter could be removed from self._putters by a
|
|
78
|
+
# previous get_nowait call.
|
|
79
|
+
pass
|
|
80
|
+
if not self.full() and not putter.cancelled():
|
|
81
|
+
# We were woken up by get_nowait(), but can't take
|
|
82
|
+
# the call. Wake up the next in line.
|
|
83
|
+
self._wakeup_next(self._putters)
|
|
84
|
+
raise
|
|
85
|
+
|
|
86
|
+
if (self._is_closed):
|
|
87
|
+
raise QueueClosed # @IgnoreException
|
|
88
|
+
|
|
89
|
+
return self.put_nowait(item)
|
|
90
|
+
|
|
91
|
+
async def get(self) -> _T:
|
|
92
|
+
"""Remove and return an item from the queue.
|
|
93
|
+
|
|
94
|
+
If queue is empty, wait until an item is available.
|
|
95
|
+
"""
|
|
96
|
+
while self.empty() and not self._is_closed:
|
|
97
|
+
getter = self._get_loop().create_future()
|
|
98
|
+
self._getters.append(getter)
|
|
99
|
+
try:
|
|
100
|
+
await getter
|
|
101
|
+
except Exception:
|
|
102
|
+
getter.cancel() # Just in case getter is not done yet.
|
|
103
|
+
try:
|
|
104
|
+
# Clean self._getters from canceled getters.
|
|
105
|
+
self._getters.remove(getter)
|
|
106
|
+
except ValueError:
|
|
107
|
+
# The getter could be removed from self._getters by a
|
|
108
|
+
# previous put_nowait call.
|
|
109
|
+
pass
|
|
110
|
+
if not self.empty() and not getter.cancelled():
|
|
111
|
+
# We were woken up by put_nowait(), but can't take
|
|
112
|
+
# the call. Wake up the next in line.
|
|
113
|
+
self._wakeup_next(self._getters)
|
|
114
|
+
raise
|
|
115
|
+
|
|
116
|
+
if (self.empty() and self._is_closed):
|
|
117
|
+
raise QueueClosed # @IgnoreException
|
|
118
|
+
|
|
119
|
+
return self.get_nowait()
|
|
120
|
+
|
|
121
|
+
def put_blocking(self, item: _T):
|
|
122
|
+
"""
|
|
123
|
+
Synchronously block until the item can be put.
|
|
124
|
+
This method creates or uses an event loop internally to call the async put().
|
|
125
|
+
If the queue is closed, it raises QueueClosed.
|
|
126
|
+
|
|
127
|
+
NOTE: If you already have an event loop running in this same thread, calling
|
|
128
|
+
`run_until_complete` can cause conflicts or an error. Typically, you only
|
|
129
|
+
want to do this from a pure synchronous environment.
|
|
130
|
+
"""
|
|
131
|
+
|
|
132
|
+
# If the queue is already closed, raise immediately
|
|
133
|
+
if self._is_closed:
|
|
134
|
+
raise QueueClosed("Queue is closed, cannot put more items.")
|
|
135
|
+
|
|
136
|
+
# Quick check: if there's space, just put_nowait() and exit
|
|
137
|
+
# (This covers the trivial case with no blocking)
|
|
138
|
+
if not self.full():
|
|
139
|
+
self.put_nowait(item)
|
|
140
|
+
return None
|
|
141
|
+
|
|
142
|
+
# If we do need to block, we run self.put(...) in an event loop
|
|
143
|
+
# We'll attempt to get the currently running loop if there is one,
|
|
144
|
+
# otherwise create a new one. If there's an existing loop, we might get
|
|
145
|
+
# an error if that loop is in the same thread. Adjust logic as needed.
|
|
146
|
+
|
|
147
|
+
try:
|
|
148
|
+
# If a loop is already running in this thread, get_running_loop() will succeed.
|
|
149
|
+
loop = asyncio.get_running_loop()
|
|
150
|
+
except RuntimeError:
|
|
151
|
+
# Means no running event loop in this thread -> create a new loop
|
|
152
|
+
loop = asyncio.new_event_loop()
|
|
153
|
+
try:
|
|
154
|
+
result = loop.run_until_complete(self.put(item))
|
|
155
|
+
finally:
|
|
156
|
+
loop.close()
|
|
157
|
+
return None
|
|
158
|
+
|
|
159
|
+
# If we got a running loop, but we aren't inside an async function,
|
|
160
|
+
# do a "blocking" wait by scheduling the put and waiting:
|
|
161
|
+
future = asyncio.run_coroutine_threadsafe(self.put(item), loop)
|
|
162
|
+
result = future.result() # blocks until done
|
|
163
|
+
return result
|
|
164
|
+
|
|
165
|
+
async def close(self):
|
|
166
|
+
"""Close the queue."""
|
|
167
|
+
if (not self._is_closed):
|
|
168
|
+
self._is_closed = True
|
|
169
|
+
|
|
170
|
+
# Hit the flag
|
|
171
|
+
self._closed.set()
|
|
172
|
+
|
|
173
|
+
self._wakeup_next(self._putters)
|
|
174
|
+
self._wakeup_next(self._getters)
|
|
175
|
+
|
|
176
|
+
def is_closed(self) -> bool:
|
|
177
|
+
"""Check if the queue is closed."""
|
|
178
|
+
return self._is_closed
|
|
File without changes
|
|
File without changes
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import typing
|
|
17
|
+
from abc import ABC
|
|
18
|
+
from abc import abstractmethod
|
|
19
|
+
from collections.abc import Callable
|
|
20
|
+
from typing import Generic
|
|
21
|
+
from typing import TypeVar
|
|
22
|
+
|
|
23
|
+
from aiq.utils.reactive.base.observer_base import ObserverBase
|
|
24
|
+
from aiq.utils.reactive.subscription import Subscription
|
|
25
|
+
|
|
26
|
+
# Covariant type param: An Observable producing type X can also produce
|
|
27
|
+
# a subtype of X.
|
|
28
|
+
_T_out_co = TypeVar("_T_out_co", covariant=True) # pylint: disable=invalid-name
|
|
29
|
+
_T = TypeVar("_T") # pylint: disable=invalid-name
|
|
30
|
+
|
|
31
|
+
OnNext = Callable[[_T], None]
|
|
32
|
+
OnError = Callable[[Exception], None]
|
|
33
|
+
OnComplete = Callable[[], None]
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
class ObservableBase(Generic[_T_out_co], ABC):
|
|
37
|
+
"""
|
|
38
|
+
Abstract base class for an Observable that can be subscribed to.
|
|
39
|
+
Produces items of type _T_out for its subscribers.
|
|
40
|
+
"""
|
|
41
|
+
|
|
42
|
+
@typing.overload
|
|
43
|
+
def subscribe(self, on_next: ObserverBase[_T_out_co]) -> Subscription:
|
|
44
|
+
...
|
|
45
|
+
|
|
46
|
+
@typing.overload
|
|
47
|
+
def subscribe(self,
|
|
48
|
+
on_next: OnNext[_T_out_co] | None = None,
|
|
49
|
+
on_error: OnError | None = None,
|
|
50
|
+
on_complete: OnComplete | None = None) -> Subscription:
|
|
51
|
+
...
|
|
52
|
+
|
|
53
|
+
@abstractmethod
|
|
54
|
+
def subscribe(self,
|
|
55
|
+
on_next: ObserverBase[_T_out_co] | OnNext[_T_out_co] | None = None,
|
|
56
|
+
on_error: OnError | None = None,
|
|
57
|
+
on_complete: OnComplete | None = None) -> Subscription:
|
|
58
|
+
"""
|
|
59
|
+
Subscribes an Observer or callbacks to this Observable.
|
|
60
|
+
|
|
61
|
+
If an Observer is provided, it will be subscribed to this Observable.
|
|
62
|
+
If callbacks are provided, they will be wrapped into an Observer and
|
|
63
|
+
subscribed to this Observable.
|
|
64
|
+
"""
|
|
65
|
+
pass
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
from abc import ABC
|
|
17
|
+
from abc import abstractmethod
|
|
18
|
+
from typing import Generic
|
|
19
|
+
from typing import TypeVar
|
|
20
|
+
|
|
21
|
+
# Contravariant type param: An Observer that can accept type X can also
|
|
22
|
+
# accept any supertype of X.
|
|
23
|
+
_T_in_contra = TypeVar("_T_in_contra", contravariant=True) # pylint: disable=invalid-name
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
class ObserverBase(Generic[_T_in_contra], ABC):
|
|
27
|
+
"""
|
|
28
|
+
Abstract base class for an Observer that can receive events of type _T_in.
|
|
29
|
+
|
|
30
|
+
Once on_error or on_complete is called, the observer is considered stopped.
|
|
31
|
+
"""
|
|
32
|
+
|
|
33
|
+
@abstractmethod
|
|
34
|
+
def on_next(self, value: _T_in_contra) -> None:
|
|
35
|
+
"""
|
|
36
|
+
Called when a new item is produced. If the observer is stopped,
|
|
37
|
+
this call should be ignored or raise an error.
|
|
38
|
+
"""
|
|
39
|
+
pass
|
|
40
|
+
|
|
41
|
+
@abstractmethod
|
|
42
|
+
def on_error(self, exc: Exception) -> None:
|
|
43
|
+
"""
|
|
44
|
+
Called when the producer signals an unrecoverable error.
|
|
45
|
+
After this call, the observer is stopped.
|
|
46
|
+
"""
|
|
47
|
+
pass
|
|
48
|
+
|
|
49
|
+
@abstractmethod
|
|
50
|
+
def on_complete(self) -> None:
|
|
51
|
+
"""
|
|
52
|
+
Called when the producer signals completion (no more items).
|
|
53
|
+
After this call, the observer is stopped.
|
|
54
|
+
"""
|
|
55
|
+
pass
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import typing
|
|
17
|
+
from abc import abstractmethod
|
|
18
|
+
from collections.abc import Callable
|
|
19
|
+
from typing import TypeVar
|
|
20
|
+
|
|
21
|
+
from .observable_base import ObservableBase
|
|
22
|
+
from .observer_base import ObserverBase
|
|
23
|
+
|
|
24
|
+
if typing.TYPE_CHECKING:
|
|
25
|
+
from aiq.utils.reactive.subscription import Subscription
|
|
26
|
+
|
|
27
|
+
T = TypeVar("T")
|
|
28
|
+
|
|
29
|
+
OnNext = Callable[[T], None]
|
|
30
|
+
OnError = Callable[[Exception], None]
|
|
31
|
+
OnComplete = Callable[[], None]
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
class SubjectBase(ObserverBase[T], ObservableBase[T]):
|
|
35
|
+
"""
|
|
36
|
+
Minimal interface we expect from the Subject for unsubscribing logic.
|
|
37
|
+
"""
|
|
38
|
+
|
|
39
|
+
@abstractmethod
|
|
40
|
+
def _unsubscribe_observer(self, observer: object) -> None:
|
|
41
|
+
pass
|
|
42
|
+
|
|
43
|
+
@abstractmethod
|
|
44
|
+
def subscribe(self,
|
|
45
|
+
on_next: ObserverBase[T] | OnNext[T] | None = None,
|
|
46
|
+
on_error: OnError | None = None,
|
|
47
|
+
on_complete: OnComplete | None = None) -> "Subscription":
|
|
48
|
+
"""
|
|
49
|
+
Subscribes an Observer or callbacks to this Observable.
|
|
50
|
+
|
|
51
|
+
If an Observer is provided, it will be subscribed to this Observable.
|
|
52
|
+
If callbacks are provided, they will be wrapped into an Observer and
|
|
53
|
+
subscribed to this Observable.
|
|
54
|
+
"""
|
|
55
|
+
pass
|
|
56
|
+
|
|
57
|
+
@abstractmethod
|
|
58
|
+
def on_next(self, value: T) -> None:
|
|
59
|
+
"""
|
|
60
|
+
Called when a new item is produced. If the observer is stopped,
|
|
61
|
+
this call should be ignored or raise an error.
|
|
62
|
+
"""
|
|
63
|
+
pass
|
|
64
|
+
|
|
65
|
+
@abstractmethod
|
|
66
|
+
def on_error(self, exc: Exception) -> None:
|
|
67
|
+
"""
|
|
68
|
+
Called when the producer signals an unrecoverable error.
|
|
69
|
+
After this call, the observer is stopped.
|
|
70
|
+
"""
|
|
71
|
+
pass
|
|
72
|
+
|
|
73
|
+
@abstractmethod
|
|
74
|
+
def on_complete(self) -> None:
|
|
75
|
+
"""
|
|
76
|
+
Called when the producer signals completion (no more items).
|
|
77
|
+
After this call, the observer is stopped.
|
|
78
|
+
"""
|
|
79
|
+
pass
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
from collections.abc import Callable
|
|
17
|
+
from typing import TypeVar
|
|
18
|
+
|
|
19
|
+
from aiq.utils.reactive.base.observable_base import ObservableBase
|
|
20
|
+
from aiq.utils.reactive.base.observer_base import ObserverBase
|
|
21
|
+
from aiq.utils.reactive.observer import Observer
|
|
22
|
+
from aiq.utils.reactive.subscription import Subscription
|
|
23
|
+
from aiq.utils.type_utils import override
|
|
24
|
+
|
|
25
|
+
# Covariant type param: An Observable producing type X can also produce
|
|
26
|
+
# a subtype of X.
|
|
27
|
+
_T_out_co = TypeVar("_T_out_co", covariant=True) # pylint: disable=invalid-name
|
|
28
|
+
_T = TypeVar("_T") # pylint: disable=invalid-name
|
|
29
|
+
|
|
30
|
+
OnNext = Callable[[_T], None]
|
|
31
|
+
OnError = Callable[[Exception], None]
|
|
32
|
+
OnComplete = Callable[[], None]
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
class Observable(ObservableBase[_T_out_co]):
|
|
36
|
+
"""
|
|
37
|
+
Concrete base Observable that implements subscribe, deferring actual hooking
|
|
38
|
+
logic to _subscribe_core.
|
|
39
|
+
"""
|
|
40
|
+
|
|
41
|
+
__slots__ = ()
|
|
42
|
+
|
|
43
|
+
def _subscribe_core(self, observer: ObserverBase) -> Subscription:
|
|
44
|
+
"""
|
|
45
|
+
By default, does nothing. Subclasses should override this to
|
|
46
|
+
attach the observer to their emission logic.
|
|
47
|
+
"""
|
|
48
|
+
raise NotImplementedError("Observable._subscribe_core must be implemented by subclasses")
|
|
49
|
+
|
|
50
|
+
@override
|
|
51
|
+
def subscribe(self,
|
|
52
|
+
on_next: ObserverBase[_T_out_co] | OnNext[_T_out_co] | None = None,
|
|
53
|
+
on_error: OnError | None = None,
|
|
54
|
+
on_complete: OnComplete | None = None) -> "Subscription":
|
|
55
|
+
|
|
56
|
+
if isinstance(on_next, ObserverBase):
|
|
57
|
+
return self._subscribe_core(on_next)
|
|
58
|
+
|
|
59
|
+
return self._subscribe_core(Observer(on_next, on_error, on_complete))
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
# SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
2
|
+
# SPDX-License-Identifier: Apache-2.0
|
|
3
|
+
#
|
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
# you may not use this file except in compliance with the License.
|
|
6
|
+
# You may obtain a copy of the License at
|
|
7
|
+
#
|
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
#
|
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
# See the License for the specific language governing permissions and
|
|
14
|
+
# limitations under the License.
|
|
15
|
+
|
|
16
|
+
import logging
|
|
17
|
+
from collections.abc import Callable
|
|
18
|
+
from typing import TypeVar
|
|
19
|
+
|
|
20
|
+
from aiq.utils.reactive.base.observer_base import ObserverBase
|
|
21
|
+
|
|
22
|
+
logger = logging.getLogger(__name__)
|
|
23
|
+
|
|
24
|
+
# Contravariant type param: An Observer that can accept type X can also
|
|
25
|
+
# accept any supertype of X.
|
|
26
|
+
_T_in_contra = TypeVar("_T_in_contra", contravariant=True) # pylint: disable=invalid-name
|
|
27
|
+
_T = TypeVar("_T") # pylint: disable=invalid-name
|
|
28
|
+
|
|
29
|
+
OnNext = Callable[[_T], None]
|
|
30
|
+
OnError = Callable[[Exception], None]
|
|
31
|
+
OnComplete = Callable[[], None]
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
class Observer(ObserverBase[_T_in_contra]):
|
|
35
|
+
"""
|
|
36
|
+
Concrete Observer that wraps user-provided callbacks into an ObserverBase.
|
|
37
|
+
"""
|
|
38
|
+
|
|
39
|
+
def __init__(
|
|
40
|
+
self,
|
|
41
|
+
on_next: OnNext | None = None,
|
|
42
|
+
on_error: OnError | None = None,
|
|
43
|
+
on_complete: OnComplete | None = None,
|
|
44
|
+
) -> None:
|
|
45
|
+
self._on_next = on_next
|
|
46
|
+
self._on_error = on_error
|
|
47
|
+
self._on_complete = on_complete
|
|
48
|
+
self._stopped = False
|
|
49
|
+
|
|
50
|
+
def on_next(self, value: _T) -> None:
|
|
51
|
+
if self._stopped:
|
|
52
|
+
return
|
|
53
|
+
if self._on_next is None:
|
|
54
|
+
return
|
|
55
|
+
try:
|
|
56
|
+
self._on_next(value)
|
|
57
|
+
except Exception as exc:
|
|
58
|
+
# If the callback itself raises, treat that as an error
|
|
59
|
+
self.on_error(exc)
|
|
60
|
+
|
|
61
|
+
def on_error(self, exc: Exception) -> None:
|
|
62
|
+
if not self._stopped:
|
|
63
|
+
if self._on_error:
|
|
64
|
+
try:
|
|
65
|
+
self._on_error(exc)
|
|
66
|
+
except Exception as e:
|
|
67
|
+
logger.exception("Error in on_error callback: %s", e, exc_info=True)
|
|
68
|
+
|
|
69
|
+
def on_complete(self) -> None:
|
|
70
|
+
if not self._stopped:
|
|
71
|
+
self._stopped = True
|
|
72
|
+
if self._on_complete:
|
|
73
|
+
try:
|
|
74
|
+
self._on_complete()
|
|
75
|
+
except Exception as e:
|
|
76
|
+
logger.exception("Error in on_complete callback: %s", e, exc_info=True)
|