nvidia-nat 1.2.1__py3-none-any.whl → 1.3.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (257) hide show
  1. aiq/__init__.py +2 -2
  2. nat/agent/base.py +27 -18
  3. nat/agent/dual_node.py +9 -4
  4. nat/agent/prompt_optimizer/prompt.py +68 -0
  5. nat/agent/prompt_optimizer/register.py +149 -0
  6. nat/agent/react_agent/agent.py +81 -50
  7. nat/agent/react_agent/register.py +59 -40
  8. nat/agent/reasoning_agent/reasoning_agent.py +17 -15
  9. nat/agent/register.py +1 -1
  10. nat/agent/rewoo_agent/agent.py +327 -149
  11. nat/agent/rewoo_agent/prompt.py +19 -22
  12. nat/agent/rewoo_agent/register.py +64 -46
  13. nat/agent/tool_calling_agent/agent.py +152 -29
  14. nat/agent/tool_calling_agent/register.py +61 -38
  15. nat/authentication/api_key/api_key_auth_provider.py +2 -2
  16. nat/authentication/credential_validator/bearer_token_validator.py +557 -0
  17. nat/authentication/http_basic_auth/http_basic_auth_provider.py +1 -1
  18. nat/authentication/interfaces.py +5 -2
  19. nat/authentication/oauth2/oauth2_auth_code_flow_provider.py +69 -36
  20. nat/authentication/oauth2/oauth2_resource_server_config.py +124 -0
  21. nat/authentication/register.py +0 -1
  22. nat/builder/builder.py +56 -24
  23. nat/builder/component_utils.py +10 -6
  24. nat/builder/context.py +70 -18
  25. nat/builder/eval_builder.py +16 -11
  26. nat/builder/framework_enum.py +1 -0
  27. nat/builder/front_end.py +1 -1
  28. nat/builder/function.py +378 -8
  29. nat/builder/function_base.py +3 -3
  30. nat/builder/function_info.py +6 -8
  31. nat/builder/intermediate_step_manager.py +6 -2
  32. nat/builder/user_interaction_manager.py +2 -2
  33. nat/builder/workflow.py +13 -1
  34. nat/builder/workflow_builder.py +327 -79
  35. nat/cli/cli_utils/config_override.py +2 -2
  36. nat/cli/commands/evaluate.py +1 -1
  37. nat/cli/commands/info/info.py +16 -6
  38. nat/cli/commands/info/list_channels.py +1 -1
  39. nat/cli/commands/info/list_components.py +7 -8
  40. nat/cli/commands/mcp/__init__.py +14 -0
  41. nat/cli/commands/mcp/mcp.py +986 -0
  42. nat/cli/commands/object_store/__init__.py +14 -0
  43. nat/cli/commands/object_store/object_store.py +227 -0
  44. nat/cli/commands/optimize.py +90 -0
  45. nat/cli/commands/registry/publish.py +2 -2
  46. nat/cli/commands/registry/pull.py +2 -2
  47. nat/cli/commands/registry/remove.py +2 -2
  48. nat/cli/commands/registry/search.py +15 -17
  49. nat/cli/commands/start.py +16 -5
  50. nat/cli/commands/uninstall.py +1 -1
  51. nat/cli/commands/workflow/templates/config.yml.j2 +14 -13
  52. nat/cli/commands/workflow/templates/pyproject.toml.j2 +5 -2
  53. nat/cli/commands/workflow/templates/register.py.j2 +2 -3
  54. nat/cli/commands/workflow/templates/workflow.py.j2 +35 -21
  55. nat/cli/commands/workflow/workflow_commands.py +105 -19
  56. nat/cli/entrypoint.py +17 -11
  57. nat/cli/main.py +3 -0
  58. nat/cli/register_workflow.py +38 -4
  59. nat/cli/type_registry.py +79 -10
  60. nat/control_flow/__init__.py +0 -0
  61. nat/control_flow/register.py +20 -0
  62. nat/control_flow/router_agent/__init__.py +0 -0
  63. nat/control_flow/router_agent/agent.py +329 -0
  64. nat/control_flow/router_agent/prompt.py +48 -0
  65. nat/control_flow/router_agent/register.py +91 -0
  66. nat/control_flow/sequential_executor.py +166 -0
  67. nat/data_models/agent.py +34 -0
  68. nat/data_models/api_server.py +196 -67
  69. nat/data_models/authentication.py +23 -9
  70. nat/data_models/common.py +1 -1
  71. nat/data_models/component.py +2 -0
  72. nat/data_models/component_ref.py +11 -0
  73. nat/data_models/config.py +42 -18
  74. nat/data_models/dataset_handler.py +1 -1
  75. nat/data_models/discovery_metadata.py +4 -4
  76. nat/data_models/evaluate.py +4 -1
  77. nat/data_models/function.py +34 -0
  78. nat/data_models/function_dependencies.py +14 -6
  79. nat/data_models/gated_field_mixin.py +242 -0
  80. nat/data_models/intermediate_step.py +3 -3
  81. nat/data_models/optimizable.py +119 -0
  82. nat/data_models/optimizer.py +149 -0
  83. nat/data_models/span.py +41 -3
  84. nat/data_models/swe_bench_model.py +1 -1
  85. nat/data_models/temperature_mixin.py +44 -0
  86. nat/data_models/thinking_mixin.py +86 -0
  87. nat/data_models/top_p_mixin.py +44 -0
  88. nat/embedder/azure_openai_embedder.py +46 -0
  89. nat/embedder/nim_embedder.py +1 -1
  90. nat/embedder/openai_embedder.py +2 -3
  91. nat/embedder/register.py +1 -1
  92. nat/eval/config.py +3 -1
  93. nat/eval/dataset_handler/dataset_handler.py +71 -7
  94. nat/eval/evaluate.py +86 -31
  95. nat/eval/evaluator/base_evaluator.py +1 -1
  96. nat/eval/evaluator/evaluator_model.py +13 -0
  97. nat/eval/intermediate_step_adapter.py +1 -1
  98. nat/eval/rag_evaluator/evaluate.py +9 -6
  99. nat/eval/rag_evaluator/register.py +3 -3
  100. nat/eval/register.py +4 -1
  101. nat/eval/remote_workflow.py +3 -3
  102. nat/eval/runtime_evaluator/__init__.py +14 -0
  103. nat/eval/runtime_evaluator/evaluate.py +123 -0
  104. nat/eval/runtime_evaluator/register.py +100 -0
  105. nat/eval/swe_bench_evaluator/evaluate.py +6 -6
  106. nat/eval/trajectory_evaluator/evaluate.py +1 -1
  107. nat/eval/trajectory_evaluator/register.py +1 -1
  108. nat/eval/tunable_rag_evaluator/evaluate.py +4 -7
  109. nat/eval/utils/eval_trace_ctx.py +89 -0
  110. nat/eval/utils/weave_eval.py +18 -9
  111. nat/experimental/decorators/experimental_warning_decorator.py +27 -7
  112. nat/experimental/test_time_compute/functions/execute_score_select_function.py +1 -1
  113. nat/experimental/test_time_compute/functions/plan_select_execute_function.py +7 -3
  114. nat/experimental/test_time_compute/functions/ttc_tool_orchestration_function.py +3 -3
  115. nat/experimental/test_time_compute/functions/ttc_tool_wrapper_function.py +3 -3
  116. nat/experimental/test_time_compute/models/strategy_base.py +5 -4
  117. nat/experimental/test_time_compute/register.py +0 -1
  118. nat/experimental/test_time_compute/selection/llm_based_output_merging_selector.py +1 -3
  119. nat/front_ends/console/authentication_flow_handler.py +82 -30
  120. nat/front_ends/console/console_front_end_plugin.py +19 -7
  121. nat/front_ends/fastapi/auth_flow_handlers/http_flow_handler.py +1 -1
  122. nat/front_ends/fastapi/auth_flow_handlers/websocket_flow_handler.py +52 -17
  123. nat/front_ends/fastapi/dask_client_mixin.py +65 -0
  124. nat/front_ends/fastapi/fastapi_front_end_config.py +36 -5
  125. nat/front_ends/fastapi/fastapi_front_end_controller.py +4 -4
  126. nat/front_ends/fastapi/fastapi_front_end_plugin.py +135 -4
  127. nat/front_ends/fastapi/fastapi_front_end_plugin_worker.py +455 -282
  128. nat/front_ends/fastapi/job_store.py +518 -99
  129. nat/front_ends/fastapi/main.py +11 -19
  130. nat/front_ends/fastapi/message_handler.py +74 -50
  131. nat/front_ends/fastapi/message_validator.py +20 -21
  132. nat/front_ends/fastapi/response_helpers.py +4 -4
  133. nat/front_ends/fastapi/step_adaptor.py +2 -2
  134. nat/front_ends/fastapi/utils.py +57 -0
  135. nat/front_ends/mcp/introspection_token_verifier.py +73 -0
  136. nat/front_ends/mcp/mcp_front_end_config.py +47 -3
  137. nat/front_ends/mcp/mcp_front_end_plugin.py +48 -13
  138. nat/front_ends/mcp/mcp_front_end_plugin_worker.py +120 -8
  139. nat/front_ends/mcp/tool_converter.py +44 -14
  140. nat/front_ends/register.py +0 -1
  141. nat/front_ends/simple_base/simple_front_end_plugin_base.py +3 -1
  142. nat/llm/aws_bedrock_llm.py +24 -12
  143. nat/llm/azure_openai_llm.py +57 -0
  144. nat/llm/litellm_llm.py +69 -0
  145. nat/llm/nim_llm.py +20 -8
  146. nat/llm/openai_llm.py +14 -6
  147. nat/llm/register.py +5 -1
  148. nat/llm/utils/env_config_value.py +2 -3
  149. nat/llm/utils/thinking.py +215 -0
  150. nat/meta/pypi.md +9 -9
  151. nat/object_store/register.py +0 -1
  152. nat/observability/exporter/base_exporter.py +3 -3
  153. nat/observability/exporter/file_exporter.py +1 -1
  154. nat/observability/exporter/processing_exporter.py +309 -81
  155. nat/observability/exporter/span_exporter.py +35 -15
  156. nat/observability/exporter_manager.py +7 -7
  157. nat/observability/mixin/file_mixin.py +7 -7
  158. nat/observability/mixin/redaction_config_mixin.py +42 -0
  159. nat/observability/mixin/tagging_config_mixin.py +62 -0
  160. nat/observability/mixin/type_introspection_mixin.py +420 -107
  161. nat/observability/processor/batching_processor.py +5 -7
  162. nat/observability/processor/falsy_batch_filter_processor.py +55 -0
  163. nat/observability/processor/processor.py +3 -0
  164. nat/observability/processor/processor_factory.py +70 -0
  165. nat/observability/processor/redaction/__init__.py +24 -0
  166. nat/observability/processor/redaction/contextual_redaction_processor.py +125 -0
  167. nat/observability/processor/redaction/contextual_span_redaction_processor.py +66 -0
  168. nat/observability/processor/redaction/redaction_processor.py +177 -0
  169. nat/observability/processor/redaction/span_header_redaction_processor.py +92 -0
  170. nat/observability/processor/span_tagging_processor.py +68 -0
  171. nat/observability/register.py +22 -4
  172. nat/profiler/calc/calc_runner.py +3 -4
  173. nat/profiler/callbacks/agno_callback_handler.py +1 -1
  174. nat/profiler/callbacks/langchain_callback_handler.py +14 -7
  175. nat/profiler/callbacks/llama_index_callback_handler.py +3 -3
  176. nat/profiler/callbacks/semantic_kernel_callback_handler.py +3 -3
  177. nat/profiler/data_frame_row.py +1 -1
  178. nat/profiler/decorators/framework_wrapper.py +62 -13
  179. nat/profiler/decorators/function_tracking.py +160 -3
  180. nat/profiler/forecasting/models/forecasting_base_model.py +3 -1
  181. nat/profiler/forecasting/models/linear_model.py +1 -1
  182. nat/profiler/forecasting/models/random_forest_regressor.py +1 -1
  183. nat/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +1 -1
  184. nat/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +1 -1
  185. nat/profiler/inference_optimization/data_models.py +3 -3
  186. nat/profiler/inference_optimization/experimental/prefix_span_analysis.py +8 -9
  187. nat/profiler/inference_optimization/token_uniqueness.py +1 -1
  188. nat/profiler/parameter_optimization/__init__.py +0 -0
  189. nat/profiler/parameter_optimization/optimizable_utils.py +93 -0
  190. nat/profiler/parameter_optimization/optimizer_runtime.py +67 -0
  191. nat/profiler/parameter_optimization/parameter_optimizer.py +164 -0
  192. nat/profiler/parameter_optimization/parameter_selection.py +107 -0
  193. nat/profiler/parameter_optimization/pareto_visualizer.py +395 -0
  194. nat/profiler/parameter_optimization/prompt_optimizer.py +384 -0
  195. nat/profiler/parameter_optimization/update_helpers.py +66 -0
  196. nat/profiler/profile_runner.py +14 -9
  197. nat/profiler/utils.py +4 -2
  198. nat/registry_handlers/local/local_handler.py +2 -2
  199. nat/registry_handlers/package_utils.py +1 -2
  200. nat/registry_handlers/pypi/pypi_handler.py +23 -26
  201. nat/registry_handlers/register.py +3 -4
  202. nat/registry_handlers/rest/rest_handler.py +12 -13
  203. nat/retriever/milvus/retriever.py +2 -2
  204. nat/retriever/nemo_retriever/retriever.py +1 -1
  205. nat/retriever/register.py +0 -1
  206. nat/runtime/loader.py +2 -2
  207. nat/runtime/runner.py +105 -8
  208. nat/runtime/session.py +69 -8
  209. nat/settings/global_settings.py +16 -5
  210. nat/tool/chat_completion.py +5 -2
  211. nat/tool/code_execution/local_sandbox/local_sandbox_server.py +3 -3
  212. nat/tool/datetime_tools.py +49 -9
  213. nat/tool/document_search.py +2 -2
  214. nat/tool/github_tools.py +450 -0
  215. nat/tool/memory_tools/add_memory_tool.py +3 -3
  216. nat/tool/memory_tools/delete_memory_tool.py +3 -4
  217. nat/tool/memory_tools/get_memory_tool.py +4 -4
  218. nat/tool/nvidia_rag.py +1 -1
  219. nat/tool/register.py +2 -9
  220. nat/tool/retriever.py +3 -2
  221. nat/utils/callable_utils.py +70 -0
  222. nat/utils/data_models/schema_validator.py +3 -3
  223. nat/utils/decorators.py +210 -0
  224. nat/utils/exception_handlers/automatic_retries.py +104 -51
  225. nat/utils/exception_handlers/schemas.py +1 -1
  226. nat/utils/io/yaml_tools.py +2 -2
  227. nat/utils/log_levels.py +25 -0
  228. nat/utils/reactive/base/observable_base.py +2 -2
  229. nat/utils/reactive/base/observer_base.py +1 -1
  230. nat/utils/reactive/observable.py +2 -2
  231. nat/utils/reactive/observer.py +4 -4
  232. nat/utils/reactive/subscription.py +1 -1
  233. nat/utils/settings/global_settings.py +6 -8
  234. nat/utils/type_converter.py +12 -3
  235. nat/utils/type_utils.py +9 -5
  236. nvidia_nat-1.3.0.dist-info/METADATA +195 -0
  237. {nvidia_nat-1.2.1.dist-info → nvidia_nat-1.3.0.dist-info}/RECORD +244 -200
  238. {nvidia_nat-1.2.1.dist-info → nvidia_nat-1.3.0.dist-info}/entry_points.txt +1 -0
  239. nat/cli/commands/info/list_mcp.py +0 -304
  240. nat/tool/github_tools/create_github_commit.py +0 -133
  241. nat/tool/github_tools/create_github_issue.py +0 -87
  242. nat/tool/github_tools/create_github_pr.py +0 -106
  243. nat/tool/github_tools/get_github_file.py +0 -106
  244. nat/tool/github_tools/get_github_issue.py +0 -166
  245. nat/tool/github_tools/get_github_pr.py +0 -256
  246. nat/tool/github_tools/update_github_issue.py +0 -100
  247. nat/tool/mcp/exceptions.py +0 -142
  248. nat/tool/mcp/mcp_client.py +0 -255
  249. nat/tool/mcp/mcp_tool.py +0 -96
  250. nat/utils/exception_handlers/mcp.py +0 -211
  251. nvidia_nat-1.2.1.dist-info/METADATA +0 -365
  252. /nat/{tool/github_tools → agent/prompt_optimizer}/__init__.py +0 -0
  253. /nat/{tool/mcp → authentication/credential_validator}/__init__.py +0 -0
  254. {nvidia_nat-1.2.1.dist-info → nvidia_nat-1.3.0.dist-info}/WHEEL +0 -0
  255. {nvidia_nat-1.2.1.dist-info → nvidia_nat-1.3.0.dist-info}/licenses/LICENSE-3rd-party.txt +0 -0
  256. {nvidia_nat-1.2.1.dist-info → nvidia_nat-1.3.0.dist-info}/licenses/LICENSE.md +0 -0
  257. {nvidia_nat-1.2.1.dist-info → nvidia_nat-1.3.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,119 @@
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 Sequence
17
+ from typing import Any
18
+ from typing import Generic
19
+ from typing import TypeVar
20
+
21
+ from optuna import Trial
22
+ from pydantic import BaseModel
23
+ from pydantic import ConfigDict
24
+ from pydantic import Field
25
+ from pydantic import model_validator
26
+
27
+ T = TypeVar("T", int, float, bool, str)
28
+
29
+
30
+ # --------------------------------------------------------------------- #
31
+ # 1. Hyper‑parameter metadata container #
32
+ # --------------------------------------------------------------------- #
33
+ class SearchSpace(BaseModel, Generic[T]):
34
+ values: Sequence[T] | None = None
35
+ low: T | None = None
36
+ high: T | None = None
37
+ log: bool = False # log scale
38
+ step: float | None = None
39
+ is_prompt: bool = False
40
+ prompt: str | None = None # prompt to optimize
41
+ prompt_purpose: str | None = None # purpose of the prompt
42
+
43
+ model_config = ConfigDict(protected_namespaces=(), extra="forbid")
44
+
45
+ @model_validator(mode="after")
46
+ def validate_search_space_parameters(self):
47
+ """Validate that either values is provided, or both high and low."""
48
+ if self.values is not None:
49
+ # If values is provided, we don't need high/low
50
+ if self.high is not None or self.low is not None:
51
+ raise ValueError("SearchSpace 'values' is mutually exclusive with 'high' and 'low'")
52
+ return self
53
+
54
+ return self
55
+
56
+ # Helper for Optuna Trials
57
+ def suggest(self, trial: Trial, name: str):
58
+ if self.is_prompt:
59
+ raise ValueError("Prompt optimization not currently supported using Optuna. "
60
+ "Use the genetic algorithm implementation instead.")
61
+ if self.values is not None:
62
+ return trial.suggest_categorical(name, self.values)
63
+ if isinstance(self.low, int):
64
+ return trial.suggest_int(name, self.low, self.high, log=self.log, step=self.step)
65
+ return trial.suggest_float(name, self.low, self.high, log=self.log, step=self.step)
66
+
67
+
68
+ def OptimizableField(
69
+ default: Any,
70
+ *,
71
+ space: SearchSpace | None = None,
72
+ merge_conflict: str = "overwrite",
73
+ **fld_kw,
74
+ ):
75
+ # 1. Pull out any user‑supplied extras (must be a dict)
76
+ user_extra = fld_kw.pop("json_schema_extra", None) or {}
77
+ if not isinstance(user_extra, dict):
78
+ raise TypeError("`json_schema_extra` must be a mapping.")
79
+
80
+ # 2. If the space is a prompt, ensure a concrete base prompt exists
81
+ if space is not None and getattr(space, "is_prompt", False):
82
+ if getattr(space, "prompt", None) is None:
83
+ if default is None:
84
+ raise ValueError("Prompt-optimized fields require a base prompt: provide a "
85
+ "non-None field default or set space.prompt.")
86
+ # Default prompt not provided in space; fall back to the field's default
87
+ space.prompt = default
88
+
89
+ # 3. Prepare our own metadata
90
+ ours = {"optimizable": True}
91
+ if space is not None:
92
+ ours["search_space"] = space
93
+
94
+ # 4. Merge with user extras according to merge_conflict policy
95
+ intersect = ours.keys() & user_extra.keys()
96
+ if intersect:
97
+ if merge_conflict == "error":
98
+ raise ValueError("`json_schema_extra` already contains reserved key(s): "
99
+ f"{', '.join(intersect)}")
100
+ if merge_conflict == "keep":
101
+ # remove the ones the user already set so we don't overwrite them
102
+ ours = {k: v for k, v in ours.items() if k not in intersect}
103
+
104
+ merged_extra = {**user_extra, **ours} # ours wins if 'overwrite'
105
+
106
+ # 5. Return a normal Pydantic Field with merged extras
107
+ return Field(default, json_schema_extra=merged_extra, **fld_kw)
108
+
109
+
110
+ class OptimizableMixin(BaseModel):
111
+ optimizable_params: list[str] = Field(default_factory=list,
112
+ description="List of parameters that can be optimized.",
113
+ exclude=True)
114
+
115
+ search_space: dict[str, SearchSpace] = Field(
116
+ default_factory=dict,
117
+ description="Optional search space overrides for optimizable parameters.",
118
+ exclude=True,
119
+ )
@@ -0,0 +1,149 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2021-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 pathlib import Path
17
+
18
+ from pydantic import BaseModel
19
+ from pydantic import Field
20
+
21
+
22
+ class OptimizerMetric(BaseModel):
23
+ """
24
+ Parameters used by the workflow optimizer to define a metric to optimize.
25
+ """
26
+ evaluator_name: str = Field(description="Name of the metric to optimize.")
27
+ direction: str = Field(description="Direction of the optimization. Can be 'maximize' or 'minimize'.")
28
+ weight: float = Field(description="Weight of the metric in the optimization process.", default=1.0)
29
+
30
+
31
+ class NumericOptimizationConfig(BaseModel):
32
+ """
33
+ Configuration for numeric/enum optimization (Optuna).
34
+ """
35
+ enabled: bool = Field(default=True, description="Enable numeric optimization")
36
+ n_trials: int = Field(description="Number of trials for numeric optimization.", default=20)
37
+
38
+
39
+ class PromptGAOptimizationConfig(BaseModel):
40
+ """
41
+ Configuration for prompt optimization using a Genetic Algorithm.
42
+ """
43
+ enabled: bool = Field(default=False, description="Enable GA-based prompt optimization")
44
+
45
+ # Prompt optimization function hooks
46
+ prompt_population_init_function: str | None = Field(
47
+ default=None,
48
+ description="Optional function name to initialize/mutate candidate prompts.",
49
+ )
50
+ prompt_recombination_function: str | None = Field(
51
+ default=None,
52
+ description="Optional function name to recombine two parent prompts into a child.",
53
+ )
54
+
55
+ # Genetic algorithm configuration
56
+ ga_population_size: int = Field(
57
+ description="Population size for genetic algorithm prompt optimization.",
58
+ default=24,
59
+ )
60
+ ga_generations: int = Field(
61
+ description="Number of generations to evolve in GA prompt optimization.",
62
+ default=15,
63
+ )
64
+ ga_offspring_size: int | None = Field(
65
+ description="Number of offspring to produce per generation. Defaults to population_size - elitism.",
66
+ default=None,
67
+ )
68
+ ga_crossover_rate: float = Field(
69
+ description="Probability of applying crossover during reproduction.",
70
+ default=0.8,
71
+ ge=0.0,
72
+ le=1.0,
73
+ )
74
+ ga_mutation_rate: float = Field(
75
+ description="Probability of mutating a child after crossover.",
76
+ default=0.3,
77
+ ge=0.0,
78
+ le=1.0,
79
+ )
80
+ ga_elitism: int = Field(
81
+ description="Number of top individuals carried over unchanged each generation.",
82
+ default=2,
83
+ )
84
+ ga_selection_method: str = Field(
85
+ description="Parent selection strategy: 'tournament' or 'roulette'.",
86
+ default="tournament",
87
+ )
88
+ ga_tournament_size: int = Field(
89
+ description="Tournament size when using tournament selection.",
90
+ default=3,
91
+ )
92
+ ga_parallel_evaluations: int = Field(
93
+ description="Max number of individuals to evaluate concurrently per generation.",
94
+ default=8,
95
+ )
96
+ ga_diversity_lambda: float = Field(
97
+ description="Strength of diversity penalty (0 disables). Penalizes identical/near-identical prompts.",
98
+ default=0.0,
99
+ ge=0.0,
100
+ )
101
+
102
+
103
+ class OptimizerConfig(BaseModel):
104
+ """
105
+ Parameters used by the workflow optimizer.
106
+ """
107
+ output_path: Path | None = Field(
108
+ default=None,
109
+ description="Path to the output directory where the results will be saved.",
110
+ )
111
+
112
+ eval_metrics: dict[str, OptimizerMetric] | None = Field(
113
+ description="List of evaluation metrics to optimize.",
114
+ default=None,
115
+ )
116
+
117
+ reps_per_param_set: int = Field(
118
+ default=3,
119
+ description="Number of repetitions per parameter set for the optimization.",
120
+ )
121
+
122
+ target: float | None = Field(
123
+ description=(
124
+ "Target value for the optimization. If set, the optimization will stop when this value is reached."),
125
+ default=None,
126
+ )
127
+
128
+ multi_objective_combination_mode: str = Field(
129
+ description="Method to combine multiple objectives into a single score.",
130
+ default="harmonic",
131
+ )
132
+
133
+ # Nested configs
134
+ numeric: NumericOptimizationConfig = NumericOptimizationConfig()
135
+ prompt: PromptGAOptimizationConfig = PromptGAOptimizationConfig()
136
+
137
+
138
+ class OptimizerRunConfig(BaseModel):
139
+ """
140
+ Parameters used for an Optimizer R=run
141
+ """
142
+ # Eval parameters
143
+
144
+ config_file: Path | BaseModel # allow for instantiated configs to be passed in
145
+ dataset: str | Path | None # dataset file path can be specified in the config file
146
+ result_json_path: str = "$"
147
+ endpoint: str | None = None # only used when running the workflow remotely
148
+ endpoint_timeout: int = 300
149
+ override: tuple[tuple[str, str], ...] = ()
nat/data_models/span.py CHANGED
@@ -128,10 +128,48 @@ class SpanStatus(BaseModel):
128
128
  message: str | None = Field(default=None, description="The status message of the span.")
129
129
 
130
130
 
131
+ def _generate_nonzero_trace_id() -> int:
132
+ """Generate a non-zero 128-bit trace ID."""
133
+ return uuid.uuid4().int
134
+
135
+
136
+ def _generate_nonzero_span_id() -> int:
137
+ """Generate a non-zero 64-bit span ID."""
138
+ return uuid.uuid4().int >> 64
139
+
140
+
131
141
  class SpanContext(BaseModel):
132
- trace_id: int = Field(default_factory=lambda: uuid.uuid4().int, description="The 128-bit trace ID of the span.")
133
- span_id: int = Field(default_factory=lambda: uuid.uuid4().int & ((1 << 64) - 1),
134
- description="The 64-bit span ID of the span.")
142
+ trace_id: int = Field(default_factory=_generate_nonzero_trace_id,
143
+ description="The OTel-syle 128-bit trace ID of the span.")
144
+ span_id: int = Field(default_factory=_generate_nonzero_span_id,
145
+ description="The OTel-syle 64-bit span ID of the span.")
146
+
147
+ @field_validator("trace_id", mode="before")
148
+ @classmethod
149
+ def _validate_trace_id(cls, v: int | str | None) -> int:
150
+ """Regenerate if trace_id is None; raise an exception if trace_id is invalid;"""
151
+ if isinstance(v, str):
152
+ v = uuid.UUID(v).int
153
+ if isinstance(v, type(None)):
154
+ v = _generate_nonzero_trace_id()
155
+ if v <= 0 or v >> 128:
156
+ raise ValueError(f"Invalid trace_id: must be a non-zero 128-bit integer, got {v}")
157
+ return v
158
+
159
+ @field_validator("span_id", mode="before")
160
+ @classmethod
161
+ def _validate_span_id(cls, v: int | str | None) -> int:
162
+ """Regenerate if span_id is None; raise an exception if span_id is invalid;"""
163
+ if isinstance(v, str):
164
+ try:
165
+ v = int(v, 16)
166
+ except ValueError:
167
+ raise ValueError(f"span_id unable to be parsed: {v}")
168
+ if isinstance(v, type(None)):
169
+ v = _generate_nonzero_span_id()
170
+ if v <= 0 or v >> 64:
171
+ raise ValueError(f"Invalid span_id: must be a non-zero 64-bit integer, got {v}")
172
+ return v
135
173
 
136
174
 
137
175
  class Span(BaseModel):
@@ -39,7 +39,7 @@ class SWEBenchInput(BaseModel):
39
39
 
40
40
  # Handle improperly formatted JSON strings for list fields
41
41
  @field_validator("FAIL_TO_PASS", "PASS_TO_PASS", mode="before")
42
- def parse_list_fields(cls, value): # pylint: disable=no-self-argument
42
+ def parse_list_fields(cls, value):
43
43
  if isinstance(value, str):
44
44
  # Attempt to parse the string as a list
45
45
  return json.loads(value)
@@ -0,0 +1,44 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ import re
17
+
18
+ from pydantic import BaseModel
19
+
20
+ from nat.data_models.gated_field_mixin import GatedFieldMixin
21
+ from nat.data_models.optimizable import OptimizableField
22
+ from nat.data_models.optimizable import SearchSpace
23
+
24
+
25
+ class TemperatureMixin(
26
+ BaseModel,
27
+ GatedFieldMixin,
28
+ field_name="temperature",
29
+ default_if_supported=0.0,
30
+ keys=("model_name", "model", "azure_deployment"),
31
+ unsupported=(re.compile(r"gpt-?5", re.IGNORECASE), ),
32
+ ):
33
+ """
34
+ Mixin class for temperature configuration. Unsupported on models like gpt-5.
35
+
36
+ Attributes:
37
+ temperature: Sampling temperature in [0, 1]. Defaults to 0.0 when supported on the model.
38
+ """
39
+ temperature: float | None = OptimizableField(
40
+ default=None,
41
+ ge=0.0,
42
+ le=1.0,
43
+ description="Sampling temperature in [0, 1]. Defaults to 0.0 when supported on the model.",
44
+ space=SearchSpace(high=0.9, low=0.1, step=0.2))
@@ -0,0 +1,86 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ import re
17
+
18
+ from pydantic import BaseModel
19
+ from pydantic import Field
20
+
21
+ from nat.data_models.gated_field_mixin import GatedFieldMixin
22
+
23
+ # Currently the control logic for thinking is only implemented for Nemotron models
24
+ _NEMOTRON_REGEX = re.compile(r"^nvidia/(llama|nvidia).*nemotron", re.IGNORECASE)
25
+ # The keys are the fields that are used to determine if the model supports thinking
26
+ _MODEL_KEYS = ("model_name", "model", "azure_deployment")
27
+
28
+
29
+ class ThinkingMixin(
30
+ BaseModel,
31
+ GatedFieldMixin,
32
+ field_name="thinking",
33
+ default_if_supported=None,
34
+ keys=_MODEL_KEYS,
35
+ supported=(_NEMOTRON_REGEX, ),
36
+ ):
37
+ """
38
+ Mixin class for thinking configuration. Only supported on Nemotron models.
39
+
40
+ Attributes:
41
+ thinking: Whether to enable thinking. Defaults to None when supported on the model.
42
+ """
43
+ thinking: bool | None = Field(
44
+ default=None,
45
+ description="Whether to enable thinking. Defaults to None when supported on the model.",
46
+ )
47
+
48
+ @property
49
+ def thinking_system_prompt(self) -> str | None:
50
+ """
51
+ Returns the system prompt to use for thinking.
52
+ For NVIDIA Nemotron, returns "/think" if enabled, else "/no_think".
53
+ For Llama Nemotron v1.5, returns "/think" if enabled, else "/no_think".
54
+ For Llama Nemotron v1.0 or v1.1, returns "detailed thinking on" if enabled, else "detailed thinking off".
55
+ If thinking is not supported on the model, returns None.
56
+
57
+ Returns:
58
+ str | None: The system prompt to use for thinking.
59
+ """
60
+ if self.thinking is None:
61
+ return None
62
+
63
+ for key in _MODEL_KEYS:
64
+ model = getattr(self, key, None)
65
+ if not isinstance(model, str) or model is None:
66
+ continue
67
+
68
+ # Normalize name to reduce checks
69
+ model = model.lower().translate(str.maketrans("_.", "--"))
70
+
71
+ if model.startswith("nvidia/nvidia"):
72
+ return "/think" if self.thinking else "/no_think"
73
+
74
+ if model.startswith("nvidia/llama"):
75
+ if "v1-0" in model or "v1-1" in model or model.endswith("v1"):
76
+ return f"detailed thinking {'on' if self.thinking else 'off'}"
77
+
78
+ if "v1-5" in model:
79
+ # v1.5 models are updated to use the /think and /no_think system prompts
80
+ return "/think" if self.thinking else "/no_think"
81
+
82
+ # Assume any other model is a newer model that uses the /think and /no_think system prompts
83
+ return "/think" if self.thinking else "/no_think"
84
+
85
+ # Unknown model
86
+ return None
@@ -0,0 +1,44 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ import re
17
+
18
+ from pydantic import BaseModel
19
+
20
+ from nat.data_models.gated_field_mixin import GatedFieldMixin
21
+ from nat.data_models.optimizable import OptimizableField
22
+ from nat.data_models.optimizable import SearchSpace
23
+
24
+
25
+ class TopPMixin(
26
+ BaseModel,
27
+ GatedFieldMixin,
28
+ field_name="top_p",
29
+ default_if_supported=1.0,
30
+ keys=("model_name", "model", "azure_deployment"),
31
+ unsupported=(re.compile(r"gpt-?5", re.IGNORECASE), ),
32
+ ):
33
+ """
34
+ Mixin class for top-p configuration. Unsupported on models like gpt-5.
35
+
36
+ Attributes:
37
+ top_p: Top-p for distribution sampling. Defaults to 1.0 when supported on the model.
38
+ """
39
+ top_p: float | None = OptimizableField(
40
+ default=None,
41
+ ge=0.0,
42
+ le=1.0,
43
+ description="Top-p for distribution sampling. Defaults to 1.0 when supported on the model.",
44
+ space=SearchSpace(high=1.0, low=0.5, step=0.1))
@@ -0,0 +1,46 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ from pydantic import AliasChoices
17
+ from pydantic import ConfigDict
18
+ from pydantic import Field
19
+
20
+ from nat.builder.builder import Builder
21
+ from nat.builder.embedder import EmbedderProviderInfo
22
+ from nat.cli.register_workflow import register_embedder_provider
23
+ from nat.data_models.embedder import EmbedderBaseConfig
24
+ from nat.data_models.retry_mixin import RetryMixin
25
+
26
+
27
+ class AzureOpenAIEmbedderModelConfig(EmbedderBaseConfig, RetryMixin, name="azure_openai"):
28
+ """An Azure OpenAI embedder provider to be used with an embedder client."""
29
+
30
+ model_config = ConfigDict(protected_namespaces=(), extra="allow")
31
+
32
+ api_key: str | None = Field(default=None, description="Azure OpenAI API key to interact with hosted model.")
33
+ api_version: str = Field(default="2025-04-01-preview", description="Azure OpenAI API version.")
34
+ azure_endpoint: str | None = Field(validation_alias=AliasChoices("azure_endpoint", "base_url"),
35
+ serialization_alias="azure_endpoint",
36
+ default=None,
37
+ description="Base URL for the hosted model.")
38
+ azure_deployment: str = Field(validation_alias=AliasChoices("azure_deployment", "model_name", "model"),
39
+ serialization_alias="azure_deployment",
40
+ description="The Azure OpenAI hosted model/deployment name.")
41
+
42
+
43
+ @register_embedder_provider(config_type=AzureOpenAIEmbedderModelConfig)
44
+ async def azure_openai_embedder_model(config: AzureOpenAIEmbedderModelConfig, _builder: Builder):
45
+
46
+ yield EmbedderProviderInfo(config=config, description="An Azure OpenAI model for use with an Embedder client.")
@@ -50,7 +50,7 @@ class NIMEmbedderModelConfig(EmbedderBaseConfig, RetryMixin, name="nim"):
50
50
  description=("The truncation strategy if the input on the "
51
51
  "server side if it's too large."))
52
52
 
53
- model_config = ConfigDict(protected_namespaces=())
53
+ model_config = ConfigDict(protected_namespaces=(), extra="allow")
54
54
 
55
55
 
56
56
  @register_embedder_provider(config_type=NIMEmbedderModelConfig)
@@ -27,17 +27,16 @@ from nat.data_models.retry_mixin import RetryMixin
27
27
  class OpenAIEmbedderModelConfig(EmbedderBaseConfig, RetryMixin, name="openai"):
28
28
  """An OpenAI LLM provider to be used with an LLM client."""
29
29
 
30
- model_config = ConfigDict(protected_namespaces=())
30
+ model_config = ConfigDict(protected_namespaces=(), extra="allow")
31
31
 
32
32
  api_key: str | None = Field(default=None, description="OpenAI API key to interact with hosted model.")
33
33
  base_url: str | None = Field(default=None, description="Base url to the hosted model.")
34
34
  model_name: str = Field(validation_alias=AliasChoices("model_name", "model"),
35
35
  serialization_alias="model",
36
36
  description="The OpenAI hosted model name.")
37
- max_retries: int = Field(default=2, description="The max number of retries for the request.")
38
37
 
39
38
 
40
39
  @register_embedder_provider(config_type=OpenAIEmbedderModelConfig)
41
- async def openai_llm(config: OpenAIEmbedderModelConfig, builder: Builder):
40
+ async def openai_embedder_model(config: OpenAIEmbedderModelConfig, _builder: Builder):
42
41
 
43
42
  yield EmbedderProviderInfo(config=config, description="An OpenAI model for use with an Embedder client.")
nat/embedder/register.py CHANGED
@@ -13,10 +13,10 @@
13
13
  # See the License for the specific language governing permissions and
14
14
  # limitations under the License.
15
15
 
16
- # pylint: disable=unused-import
17
16
  # flake8: noqa
18
17
  # isort:skip_file
19
18
 
20
19
  # Import any providers which need to be automatically registered here
20
+ from . import azure_openai_embedder
21
21
  from . import nim_embedder
22
22
  from . import openai_embedder
nat/eval/config.py CHANGED
@@ -27,7 +27,7 @@ class EvaluationRunConfig(BaseModel):
27
27
  """
28
28
  Parameters used for a single evaluation run.
29
29
  """
30
- config_file: Path
30
+ config_file: Path | BaseModel
31
31
  dataset: str | None = None # dataset file path can be specified in the config file
32
32
  result_json_path: str = "$"
33
33
  skip_workflow: bool = False
@@ -44,6 +44,8 @@ class EvaluationRunConfig(BaseModel):
44
44
  # number of passes at each concurrency, if 0 the dataset is adjusted to a multiple of the
45
45
  # concurrency. The is only used if adjust_dataset_size is true
46
46
  num_passes: int = 0
47
+ # timeout for waiting for trace export tasks to complete
48
+ export_timeout: float = 60.0
47
49
 
48
50
 
49
51
  class EvaluationRunOutput(BaseModel):