aiqtoolkit 1.1.0a20250503__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.

Potentially problematic release.


This version of aiqtoolkit might be problematic. Click here for more details.

Files changed (314) hide show
  1. aiq/agent/__init__.py +0 -0
  2. aiq/agent/base.py +76 -0
  3. aiq/agent/dual_node.py +67 -0
  4. aiq/agent/react_agent/__init__.py +0 -0
  5. aiq/agent/react_agent/agent.py +322 -0
  6. aiq/agent/react_agent/output_parser.py +104 -0
  7. aiq/agent/react_agent/prompt.py +46 -0
  8. aiq/agent/react_agent/register.py +148 -0
  9. aiq/agent/reasoning_agent/__init__.py +0 -0
  10. aiq/agent/reasoning_agent/reasoning_agent.py +224 -0
  11. aiq/agent/register.py +23 -0
  12. aiq/agent/rewoo_agent/__init__.py +0 -0
  13. aiq/agent/rewoo_agent/agent.py +410 -0
  14. aiq/agent/rewoo_agent/prompt.py +108 -0
  15. aiq/agent/rewoo_agent/register.py +158 -0
  16. aiq/agent/tool_calling_agent/__init__.py +0 -0
  17. aiq/agent/tool_calling_agent/agent.py +123 -0
  18. aiq/agent/tool_calling_agent/register.py +105 -0
  19. aiq/builder/__init__.py +0 -0
  20. aiq/builder/builder.py +223 -0
  21. aiq/builder/component_utils.py +303 -0
  22. aiq/builder/context.py +212 -0
  23. aiq/builder/embedder.py +24 -0
  24. aiq/builder/eval_builder.py +116 -0
  25. aiq/builder/evaluator.py +29 -0
  26. aiq/builder/framework_enum.py +24 -0
  27. aiq/builder/front_end.py +73 -0
  28. aiq/builder/function.py +297 -0
  29. aiq/builder/function_base.py +376 -0
  30. aiq/builder/function_info.py +627 -0
  31. aiq/builder/intermediate_step_manager.py +127 -0
  32. aiq/builder/llm.py +25 -0
  33. aiq/builder/retriever.py +25 -0
  34. aiq/builder/user_interaction_manager.py +71 -0
  35. aiq/builder/workflow.py +143 -0
  36. aiq/builder/workflow_builder.py +749 -0
  37. aiq/cli/__init__.py +14 -0
  38. aiq/cli/cli_utils/__init__.py +0 -0
  39. aiq/cli/cli_utils/config_override.py +233 -0
  40. aiq/cli/cli_utils/validation.py +37 -0
  41. aiq/cli/commands/__init__.py +0 -0
  42. aiq/cli/commands/configure/__init__.py +0 -0
  43. aiq/cli/commands/configure/channel/__init__.py +0 -0
  44. aiq/cli/commands/configure/channel/add.py +28 -0
  45. aiq/cli/commands/configure/channel/channel.py +36 -0
  46. aiq/cli/commands/configure/channel/remove.py +30 -0
  47. aiq/cli/commands/configure/channel/update.py +30 -0
  48. aiq/cli/commands/configure/configure.py +33 -0
  49. aiq/cli/commands/evaluate.py +139 -0
  50. aiq/cli/commands/info/__init__.py +14 -0
  51. aiq/cli/commands/info/info.py +37 -0
  52. aiq/cli/commands/info/list_channels.py +32 -0
  53. aiq/cli/commands/info/list_components.py +129 -0
  54. aiq/cli/commands/registry/__init__.py +14 -0
  55. aiq/cli/commands/registry/publish.py +88 -0
  56. aiq/cli/commands/registry/pull.py +118 -0
  57. aiq/cli/commands/registry/registry.py +38 -0
  58. aiq/cli/commands/registry/remove.py +108 -0
  59. aiq/cli/commands/registry/search.py +155 -0
  60. aiq/cli/commands/start.py +250 -0
  61. aiq/cli/commands/uninstall.py +83 -0
  62. aiq/cli/commands/validate.py +47 -0
  63. aiq/cli/commands/workflow/__init__.py +14 -0
  64. aiq/cli/commands/workflow/templates/__init__.py.j2 +0 -0
  65. aiq/cli/commands/workflow/templates/config.yml.j2 +16 -0
  66. aiq/cli/commands/workflow/templates/pyproject.toml.j2 +22 -0
  67. aiq/cli/commands/workflow/templates/register.py.j2 +5 -0
  68. aiq/cli/commands/workflow/templates/workflow.py.j2 +36 -0
  69. aiq/cli/commands/workflow/workflow.py +37 -0
  70. aiq/cli/commands/workflow/workflow_commands.py +313 -0
  71. aiq/cli/entrypoint.py +133 -0
  72. aiq/cli/main.py +44 -0
  73. aiq/cli/register_workflow.py +408 -0
  74. aiq/cli/type_registry.py +879 -0
  75. aiq/data_models/__init__.py +14 -0
  76. aiq/data_models/api_server.py +588 -0
  77. aiq/data_models/common.py +143 -0
  78. aiq/data_models/component.py +46 -0
  79. aiq/data_models/component_ref.py +135 -0
  80. aiq/data_models/config.py +349 -0
  81. aiq/data_models/dataset_handler.py +122 -0
  82. aiq/data_models/discovery_metadata.py +269 -0
  83. aiq/data_models/embedder.py +26 -0
  84. aiq/data_models/evaluate.py +104 -0
  85. aiq/data_models/evaluator.py +26 -0
  86. aiq/data_models/front_end.py +26 -0
  87. aiq/data_models/function.py +30 -0
  88. aiq/data_models/function_dependencies.py +64 -0
  89. aiq/data_models/interactive.py +237 -0
  90. aiq/data_models/intermediate_step.py +269 -0
  91. aiq/data_models/invocation_node.py +38 -0
  92. aiq/data_models/llm.py +26 -0
  93. aiq/data_models/logging.py +26 -0
  94. aiq/data_models/memory.py +26 -0
  95. aiq/data_models/profiler.py +53 -0
  96. aiq/data_models/registry_handler.py +26 -0
  97. aiq/data_models/retriever.py +30 -0
  98. aiq/data_models/step_adaptor.py +64 -0
  99. aiq/data_models/streaming.py +33 -0
  100. aiq/data_models/swe_bench_model.py +54 -0
  101. aiq/data_models/telemetry_exporter.py +26 -0
  102. aiq/embedder/__init__.py +0 -0
  103. aiq/embedder/langchain_client.py +41 -0
  104. aiq/embedder/nim_embedder.py +58 -0
  105. aiq/embedder/openai_embedder.py +42 -0
  106. aiq/embedder/register.py +24 -0
  107. aiq/eval/__init__.py +14 -0
  108. aiq/eval/config.py +42 -0
  109. aiq/eval/dataset_handler/__init__.py +0 -0
  110. aiq/eval/dataset_handler/dataset_downloader.py +106 -0
  111. aiq/eval/dataset_handler/dataset_filter.py +52 -0
  112. aiq/eval/dataset_handler/dataset_handler.py +169 -0
  113. aiq/eval/evaluate.py +323 -0
  114. aiq/eval/evaluator/__init__.py +14 -0
  115. aiq/eval/evaluator/evaluator_model.py +44 -0
  116. aiq/eval/intermediate_step_adapter.py +93 -0
  117. aiq/eval/rag_evaluator/__init__.py +0 -0
  118. aiq/eval/rag_evaluator/evaluate.py +138 -0
  119. aiq/eval/rag_evaluator/register.py +138 -0
  120. aiq/eval/register.py +23 -0
  121. aiq/eval/remote_workflow.py +128 -0
  122. aiq/eval/runtime_event_subscriber.py +52 -0
  123. aiq/eval/swe_bench_evaluator/__init__.py +0 -0
  124. aiq/eval/swe_bench_evaluator/evaluate.py +215 -0
  125. aiq/eval/swe_bench_evaluator/register.py +36 -0
  126. aiq/eval/trajectory_evaluator/__init__.py +0 -0
  127. aiq/eval/trajectory_evaluator/evaluate.py +118 -0
  128. aiq/eval/trajectory_evaluator/register.py +40 -0
  129. aiq/eval/tunable_rag_evaluator/__init__.py +0 -0
  130. aiq/eval/tunable_rag_evaluator/evaluate.py +263 -0
  131. aiq/eval/tunable_rag_evaluator/register.py +50 -0
  132. aiq/eval/utils/__init__.py +0 -0
  133. aiq/eval/utils/output_uploader.py +131 -0
  134. aiq/eval/utils/tqdm_position_registry.py +40 -0
  135. aiq/front_ends/__init__.py +14 -0
  136. aiq/front_ends/console/__init__.py +14 -0
  137. aiq/front_ends/console/console_front_end_config.py +32 -0
  138. aiq/front_ends/console/console_front_end_plugin.py +107 -0
  139. aiq/front_ends/console/register.py +25 -0
  140. aiq/front_ends/cron/__init__.py +14 -0
  141. aiq/front_ends/fastapi/__init__.py +14 -0
  142. aiq/front_ends/fastapi/fastapi_front_end_config.py +150 -0
  143. aiq/front_ends/fastapi/fastapi_front_end_plugin.py +103 -0
  144. aiq/front_ends/fastapi/fastapi_front_end_plugin_worker.py +607 -0
  145. aiq/front_ends/fastapi/intermediate_steps_subscriber.py +80 -0
  146. aiq/front_ends/fastapi/job_store.py +161 -0
  147. aiq/front_ends/fastapi/main.py +70 -0
  148. aiq/front_ends/fastapi/message_handler.py +279 -0
  149. aiq/front_ends/fastapi/message_validator.py +345 -0
  150. aiq/front_ends/fastapi/register.py +25 -0
  151. aiq/front_ends/fastapi/response_helpers.py +195 -0
  152. aiq/front_ends/fastapi/step_adaptor.py +315 -0
  153. aiq/front_ends/fastapi/websocket.py +148 -0
  154. aiq/front_ends/mcp/__init__.py +14 -0
  155. aiq/front_ends/mcp/mcp_front_end_config.py +32 -0
  156. aiq/front_ends/mcp/mcp_front_end_plugin.py +93 -0
  157. aiq/front_ends/mcp/register.py +27 -0
  158. aiq/front_ends/mcp/tool_converter.py +242 -0
  159. aiq/front_ends/register.py +22 -0
  160. aiq/front_ends/simple_base/__init__.py +14 -0
  161. aiq/front_ends/simple_base/simple_front_end_plugin_base.py +52 -0
  162. aiq/llm/__init__.py +0 -0
  163. aiq/llm/nim_llm.py +45 -0
  164. aiq/llm/openai_llm.py +45 -0
  165. aiq/llm/register.py +22 -0
  166. aiq/llm/utils/__init__.py +14 -0
  167. aiq/llm/utils/env_config_value.py +94 -0
  168. aiq/llm/utils/error.py +17 -0
  169. aiq/memory/__init__.py +20 -0
  170. aiq/memory/interfaces.py +183 -0
  171. aiq/memory/models.py +102 -0
  172. aiq/meta/module_to_distro.json +3 -0
  173. aiq/meta/pypi.md +59 -0
  174. aiq/observability/__init__.py +0 -0
  175. aiq/observability/async_otel_listener.py +433 -0
  176. aiq/observability/register.py +99 -0
  177. aiq/plugins/.namespace +1 -0
  178. aiq/profiler/__init__.py +0 -0
  179. aiq/profiler/callbacks/__init__.py +0 -0
  180. aiq/profiler/callbacks/agno_callback_handler.py +295 -0
  181. aiq/profiler/callbacks/base_callback_class.py +20 -0
  182. aiq/profiler/callbacks/langchain_callback_handler.py +278 -0
  183. aiq/profiler/callbacks/llama_index_callback_handler.py +205 -0
  184. aiq/profiler/callbacks/semantic_kernel_callback_handler.py +238 -0
  185. aiq/profiler/callbacks/token_usage_base_model.py +27 -0
  186. aiq/profiler/data_frame_row.py +51 -0
  187. aiq/profiler/decorators/__init__.py +0 -0
  188. aiq/profiler/decorators/framework_wrapper.py +131 -0
  189. aiq/profiler/decorators/function_tracking.py +254 -0
  190. aiq/profiler/forecasting/__init__.py +0 -0
  191. aiq/profiler/forecasting/config.py +18 -0
  192. aiq/profiler/forecasting/model_trainer.py +75 -0
  193. aiq/profiler/forecasting/models/__init__.py +22 -0
  194. aiq/profiler/forecasting/models/forecasting_base_model.py +40 -0
  195. aiq/profiler/forecasting/models/linear_model.py +196 -0
  196. aiq/profiler/forecasting/models/random_forest_regressor.py +268 -0
  197. aiq/profiler/inference_metrics_model.py +25 -0
  198. aiq/profiler/inference_optimization/__init__.py +0 -0
  199. aiq/profiler/inference_optimization/bottleneck_analysis/__init__.py +0 -0
  200. aiq/profiler/inference_optimization/bottleneck_analysis/nested_stack_analysis.py +452 -0
  201. aiq/profiler/inference_optimization/bottleneck_analysis/simple_stack_analysis.py +258 -0
  202. aiq/profiler/inference_optimization/data_models.py +386 -0
  203. aiq/profiler/inference_optimization/experimental/__init__.py +0 -0
  204. aiq/profiler/inference_optimization/experimental/concurrency_spike_analysis.py +468 -0
  205. aiq/profiler/inference_optimization/experimental/prefix_span_analysis.py +405 -0
  206. aiq/profiler/inference_optimization/llm_metrics.py +212 -0
  207. aiq/profiler/inference_optimization/prompt_caching.py +163 -0
  208. aiq/profiler/inference_optimization/token_uniqueness.py +107 -0
  209. aiq/profiler/inference_optimization/workflow_runtimes.py +72 -0
  210. aiq/profiler/intermediate_property_adapter.py +102 -0
  211. aiq/profiler/profile_runner.py +433 -0
  212. aiq/profiler/utils.py +184 -0
  213. aiq/registry_handlers/__init__.py +0 -0
  214. aiq/registry_handlers/local/__init__.py +0 -0
  215. aiq/registry_handlers/local/local_handler.py +176 -0
  216. aiq/registry_handlers/local/register_local.py +37 -0
  217. aiq/registry_handlers/metadata_factory.py +60 -0
  218. aiq/registry_handlers/package_utils.py +198 -0
  219. aiq/registry_handlers/pypi/__init__.py +0 -0
  220. aiq/registry_handlers/pypi/pypi_handler.py +251 -0
  221. aiq/registry_handlers/pypi/register_pypi.py +40 -0
  222. aiq/registry_handlers/register.py +21 -0
  223. aiq/registry_handlers/registry_handler_base.py +157 -0
  224. aiq/registry_handlers/rest/__init__.py +0 -0
  225. aiq/registry_handlers/rest/register_rest.py +56 -0
  226. aiq/registry_handlers/rest/rest_handler.py +237 -0
  227. aiq/registry_handlers/schemas/__init__.py +0 -0
  228. aiq/registry_handlers/schemas/headers.py +42 -0
  229. aiq/registry_handlers/schemas/package.py +68 -0
  230. aiq/registry_handlers/schemas/publish.py +63 -0
  231. aiq/registry_handlers/schemas/pull.py +82 -0
  232. aiq/registry_handlers/schemas/remove.py +36 -0
  233. aiq/registry_handlers/schemas/search.py +91 -0
  234. aiq/registry_handlers/schemas/status.py +47 -0
  235. aiq/retriever/__init__.py +0 -0
  236. aiq/retriever/interface.py +37 -0
  237. aiq/retriever/milvus/__init__.py +14 -0
  238. aiq/retriever/milvus/register.py +81 -0
  239. aiq/retriever/milvus/retriever.py +228 -0
  240. aiq/retriever/models.py +74 -0
  241. aiq/retriever/nemo_retriever/__init__.py +14 -0
  242. aiq/retriever/nemo_retriever/register.py +60 -0
  243. aiq/retriever/nemo_retriever/retriever.py +190 -0
  244. aiq/retriever/register.py +22 -0
  245. aiq/runtime/__init__.py +14 -0
  246. aiq/runtime/loader.py +188 -0
  247. aiq/runtime/runner.py +176 -0
  248. aiq/runtime/session.py +136 -0
  249. aiq/runtime/user_metadata.py +131 -0
  250. aiq/settings/__init__.py +0 -0
  251. aiq/settings/global_settings.py +318 -0
  252. aiq/test/.namespace +1 -0
  253. aiq/tool/__init__.py +0 -0
  254. aiq/tool/code_execution/__init__.py +0 -0
  255. aiq/tool/code_execution/code_sandbox.py +188 -0
  256. aiq/tool/code_execution/local_sandbox/Dockerfile.sandbox +60 -0
  257. aiq/tool/code_execution/local_sandbox/__init__.py +13 -0
  258. aiq/tool/code_execution/local_sandbox/local_sandbox_server.py +79 -0
  259. aiq/tool/code_execution/local_sandbox/sandbox.requirements.txt +4 -0
  260. aiq/tool/code_execution/local_sandbox/start_local_sandbox.sh +25 -0
  261. aiq/tool/code_execution/register.py +70 -0
  262. aiq/tool/code_execution/utils.py +100 -0
  263. aiq/tool/datetime_tools.py +42 -0
  264. aiq/tool/document_search.py +141 -0
  265. aiq/tool/github_tools/__init__.py +0 -0
  266. aiq/tool/github_tools/create_github_commit.py +133 -0
  267. aiq/tool/github_tools/create_github_issue.py +87 -0
  268. aiq/tool/github_tools/create_github_pr.py +106 -0
  269. aiq/tool/github_tools/get_github_file.py +106 -0
  270. aiq/tool/github_tools/get_github_issue.py +166 -0
  271. aiq/tool/github_tools/get_github_pr.py +256 -0
  272. aiq/tool/github_tools/update_github_issue.py +100 -0
  273. aiq/tool/mcp/__init__.py +14 -0
  274. aiq/tool/mcp/mcp_client.py +220 -0
  275. aiq/tool/mcp/mcp_tool.py +76 -0
  276. aiq/tool/memory_tools/__init__.py +0 -0
  277. aiq/tool/memory_tools/add_memory_tool.py +67 -0
  278. aiq/tool/memory_tools/delete_memory_tool.py +67 -0
  279. aiq/tool/memory_tools/get_memory_tool.py +72 -0
  280. aiq/tool/nvidia_rag.py +95 -0
  281. aiq/tool/register.py +36 -0
  282. aiq/tool/retriever.py +89 -0
  283. aiq/utils/__init__.py +0 -0
  284. aiq/utils/data_models/__init__.py +0 -0
  285. aiq/utils/data_models/schema_validator.py +58 -0
  286. aiq/utils/debugging_utils.py +43 -0
  287. aiq/utils/exception_handlers/__init__.py +0 -0
  288. aiq/utils/exception_handlers/schemas.py +114 -0
  289. aiq/utils/io/__init__.py +0 -0
  290. aiq/utils/io/yaml_tools.py +119 -0
  291. aiq/utils/metadata_utils.py +74 -0
  292. aiq/utils/optional_imports.py +142 -0
  293. aiq/utils/producer_consumer_queue.py +178 -0
  294. aiq/utils/reactive/__init__.py +0 -0
  295. aiq/utils/reactive/base/__init__.py +0 -0
  296. aiq/utils/reactive/base/observable_base.py +65 -0
  297. aiq/utils/reactive/base/observer_base.py +55 -0
  298. aiq/utils/reactive/base/subject_base.py +79 -0
  299. aiq/utils/reactive/observable.py +59 -0
  300. aiq/utils/reactive/observer.py +76 -0
  301. aiq/utils/reactive/subject.py +131 -0
  302. aiq/utils/reactive/subscription.py +49 -0
  303. aiq/utils/settings/__init__.py +0 -0
  304. aiq/utils/settings/global_settings.py +197 -0
  305. aiq/utils/type_converter.py +232 -0
  306. aiq/utils/type_utils.py +397 -0
  307. aiq/utils/url_utils.py +27 -0
  308. aiqtoolkit-1.1.0a20250503.dist-info/METADATA +330 -0
  309. aiqtoolkit-1.1.0a20250503.dist-info/RECORD +314 -0
  310. aiqtoolkit-1.1.0a20250503.dist-info/WHEEL +5 -0
  311. aiqtoolkit-1.1.0a20250503.dist-info/entry_points.txt +17 -0
  312. aiqtoolkit-1.1.0a20250503.dist-info/licenses/LICENSE-3rd-party.txt +3686 -0
  313. aiqtoolkit-1.1.0a20250503.dist-info/licenses/LICENSE.md +201 -0
  314. aiqtoolkit-1.1.0a20250503.dist-info/top_level.txt +1 -0
@@ -0,0 +1,176 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ import logging
17
+ import subprocess
18
+ from collections.abc import AsyncGenerator
19
+ from contextlib import asynccontextmanager
20
+
21
+ from aiq.registry_handlers.package_utils import build_package_metadata
22
+ from aiq.registry_handlers.registry_handler_base import AbstractRegistryHandler
23
+ from aiq.registry_handlers.schemas.package import PackageNameVersionList
24
+ from aiq.registry_handlers.schemas.publish import AIQArtifact
25
+ from aiq.registry_handlers.schemas.publish import PublishResponse
26
+ from aiq.registry_handlers.schemas.pull import PullRequestPackages
27
+ from aiq.registry_handlers.schemas.pull import PullResponse
28
+ from aiq.registry_handlers.schemas.remove import RemoveResponse
29
+ from aiq.registry_handlers.schemas.search import SearchFields
30
+ from aiq.registry_handlers.schemas.search import SearchQuery
31
+ from aiq.registry_handlers.schemas.search import SearchResponse
32
+ from aiq.registry_handlers.schemas.status import ActionEnum
33
+ from aiq.registry_handlers.schemas.status import StatusEnum
34
+
35
+ logger = logging.getLogger(__name__)
36
+
37
+
38
+ class LocalRegistryHandler(AbstractRegistryHandler):
39
+ """A registry handler for interactions with a local Python environment."""
40
+
41
+ search_fields: list[SearchFields] = [field for field in SearchFields if field != SearchFields.ALL]
42
+
43
+ @asynccontextmanager
44
+ async def publish(self, artifact: AIQArtifact) -> AsyncGenerator[PublishResponse]:
45
+ """Publishes an AIQ Toolkit artifact to a local registry.
46
+
47
+ Args:
48
+ artifact (AIQArtifact): An artifact that contain AIQ Toolkit plugin wheel and it's corrosponding discovery
49
+ metadata.
50
+
51
+ Yields:
52
+ Iterator[AsyncGenerator[PublishResponse]]: A response message that includes a completion status message.
53
+ """
54
+
55
+ try:
56
+ validated_remove_response = RemoveResponse(status={
57
+ "status": StatusEnum.ERROR, "message": "Local publish not supported.", "action": ActionEnum.PUBLISH
58
+ })
59
+ yield validated_remove_response
60
+ finally:
61
+ logger.warning(validated_remove_response.status.message)
62
+
63
+ @asynccontextmanager
64
+ async def pull(self, packages: PullRequestPackages) -> AsyncGenerator[PullResponse]:
65
+ """Download and install AIQ Toolkit artifacts from a local registry.
66
+
67
+ Args:
68
+ packages (PullRequestPackages): Parameters used to pull the AIQ Toolkit artifact.
69
+
70
+ Yields:
71
+ Iterator[AsyncGenerator[PullResponse]]: A response message that includes a the pulled packages and a
72
+ completion status message.
73
+ """
74
+
75
+ try:
76
+ validated_remove_response = RemoveResponse(status={
77
+ "status": StatusEnum.ERROR, "message": "Local pull not supported.", "action": ActionEnum.PULL
78
+ })
79
+
80
+ yield validated_remove_response
81
+ finally:
82
+ logger.warning(validated_remove_response.status.message)
83
+
84
+ @asynccontextmanager
85
+ async def search(self, query: SearchQuery) -> AsyncGenerator[SearchResponse]:
86
+ """Searches the local aiq registry for relevant AIQ Toolkit components.
87
+
88
+ Args:
89
+ query (SearchQuery): Parameters of the search to be performed.
90
+
91
+ Yields:
92
+ Iterator[AsyncGenerator[SearchResponse]]: A response message that includes search
93
+ parameters and a completion status message.
94
+ """
95
+
96
+ try:
97
+ results_dict = build_package_metadata(wheel_data=None)
98
+ component_results = []
99
+ query_component_types = [component_type.value for component_type in query.component_types]
100
+ for component_type, components in results_dict.items():
101
+ if component_type in query_component_types:
102
+ component_results.extend(components)
103
+
104
+ if (SearchFields.ALL in query.fields):
105
+ query.fields = self.search_fields
106
+
107
+ matched_results = []
108
+ for component_result in component_results:
109
+ for search_field in query.fields:
110
+ if (query.query in component_result.get(search_field.value, "")):
111
+ matched_results.append(component_result)
112
+ break
113
+
114
+ if query.top_k > 0:
115
+ top_k = query.top_k
116
+ else:
117
+ top_k = len(matched_results)
118
+
119
+ validated_search_response = SearchResponse(results=matched_results[:top_k],
120
+ params=query,
121
+ status={
122
+ "status": StatusEnum.SUCCESS,
123
+ "message": "",
124
+ "action": ActionEnum.SEARCH
125
+ })
126
+ yield validated_search_response
127
+
128
+ except Exception as e:
129
+ msg = f"Error searching for artifacts: {e}"
130
+ validated_search_response = SearchResponse(params=query,
131
+ status={
132
+ "status": StatusEnum.SUCCESS,
133
+ "message": msg,
134
+ "action": ActionEnum.SEARCH
135
+ })
136
+ logger.exception(validated_search_response.status.message, exc_info=True)
137
+
138
+ yield validated_search_response
139
+
140
+ finally:
141
+ logger.info("Execution complete.")
142
+
143
+ @asynccontextmanager
144
+ async def remove(self, packages: PackageNameVersionList) -> AsyncGenerator[RemoveResponse]:
145
+ """Uninstall packages from the local Python environment.
146
+
147
+ Args:
148
+ packages (PackageNameVersionList): The list of packages to remove.
149
+
150
+ Yields:
151
+ Iterator[AsyncGenerator[RemoveResponse]]: A response message that includes the packages and a
152
+ completion status message.
153
+ """
154
+
155
+ try:
156
+ for package_name in packages.packages:
157
+ result = subprocess.run(["uv", "pip", "uninstall", package_name.name], check=True)
158
+ result.check_returncode()
159
+
160
+ validated_remove_response = RemoveResponse(status={
161
+ "status": StatusEnum.SUCCESS, "message": "", "action": ActionEnum.REMOVE
162
+ }) # type: ignore
163
+
164
+ yield validated_remove_response
165
+
166
+ except Exception as e:
167
+ msg = f"Error uninstalling artifacts: {e}"
168
+ validated_remove_response = RemoveResponse(status={
169
+ "status": StatusEnum.ERROR, "message": msg, "action": ActionEnum.REMOVE
170
+ }) # type: ignore
171
+ logger.exception(validated_remove_response.status.message, exc_info=True)
172
+
173
+ yield validated_remove_response
174
+
175
+ finally:
176
+ logger.info("Execution complete.")
@@ -0,0 +1,37 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
+ # SPDX-License-Identifier: Apache-2.0
3
+ #
4
+ # Licensed under the Apache License, Version 2.0 (the "License");
5
+ # you may not use this file except in compliance with the License.
6
+ # You may obtain a copy of the License at
7
+ #
8
+ # http://www.apache.org/licenses/LICENSE-2.0
9
+ #
10
+ # Unless required by applicable law or agreed to in writing, software
11
+ # distributed under the License is distributed on an "AS IS" BASIS,
12
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ # See the License for the specific language governing permissions and
14
+ # limitations under the License.
15
+
16
+ import logging
17
+
18
+ from aiq.cli.register_workflow import register_registry_handler
19
+ from aiq.data_models.registry_handler import RegistryHandlerBaseConfig
20
+
21
+ logger = logging.getLogger(__name__)
22
+
23
+
24
+ class LocalRegistryHandlerConfig(RegistryHandlerBaseConfig, name="local"):
25
+ """Interact with the local AIQ Toolkit environment to search and uninstall AIQ Toolkit components."""
26
+
27
+ pass
28
+
29
+
30
+ @register_registry_handler(config_type=LocalRegistryHandlerConfig)
31
+ async def local_registry_handler(config: LocalRegistryHandlerConfig):
32
+
33
+ from aiq.registry_handlers.local.local_handler import LocalRegistryHandler
34
+
35
+ registry_handler = LocalRegistryHandler()
36
+
37
+ yield registry_handler
@@ -0,0 +1,60 @@
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.data_models.component import AIQComponentEnum
19
+ from aiq.data_models.discovery_metadata import DiscoveryMetadata
20
+ from aiq.data_models.discovery_metadata import DiscoveryStatusEnum
21
+ from aiq.registry_handlers.schemas.package import WheelData
22
+
23
+ logger = logging.getLogger(__name__)
24
+
25
+
26
+ class ComponentDiscoveryMetadata:
27
+
28
+ def __init__(self, component_type: AIQComponentEnum, wheel_data: WheelData | None = None):
29
+ self._component_type = component_type
30
+ self._metadata_items: list[dict | DiscoveryMetadata] = []
31
+ self._wheel_data: WheelData = wheel_data
32
+
33
+ def load_metadata(self):
34
+
35
+ from aiq.cli.type_registry import GlobalTypeRegistry
36
+
37
+ registry = GlobalTypeRegistry.get()
38
+
39
+ for _, registered_component_info in registry.get_infos_by_type(component_type=self._component_type).items():
40
+
41
+ if ((registered_component_info.discovery_metadata.status == DiscoveryStatusEnum.SUCCESS) and
42
+ ((self._wheel_data is None) or
43
+ (registered_component_info.discovery_metadata.package in self._wheel_data.union_dependencies))):
44
+
45
+ if ((self._wheel_data is not None)
46
+ and (registered_component_info.discovery_metadata.package == self._wheel_data.package_name)):
47
+ discovery_metadata_copy = registered_component_info.discovery_metadata.model_copy(deep=True)
48
+ discovery_metadata_copy.version = self._wheel_data.whl_version
49
+ self._metadata_items.append(discovery_metadata_copy.model_dump())
50
+ continue
51
+
52
+ self._metadata_items.append(registered_component_info.discovery_metadata.model_dump())
53
+
54
+ def get_metadata_items(self) -> list[dict | DiscoveryMetadata]:
55
+ return self._metadata_items
56
+
57
+ @staticmethod
58
+ def from_package_component_type(component_type: AIQComponentEnum,
59
+ wheel_data: WheelData | None = None) -> "ComponentDiscoveryMetadata":
60
+ return ComponentDiscoveryMetadata(component_type=component_type, wheel_data=wheel_data)
@@ -0,0 +1,198 @@
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 base64
17
+ import importlib.metadata
18
+ import logging
19
+ import os
20
+ import subprocess
21
+ from functools import lru_cache
22
+
23
+ from aiq.data_models.component import AIQComponentEnum
24
+ from aiq.data_models.discovery_metadata import DiscoveryMetadata
25
+ from aiq.registry_handlers.schemas.package import WheelData
26
+ from aiq.registry_handlers.schemas.publish import AIQArtifact
27
+ from aiq.runtime.loader import PluginTypes
28
+ from aiq.runtime.loader import discover_entrypoints
29
+
30
+ # pylint: disable=redefined-outer-name
31
+ logger = logging.getLogger(__name__)
32
+
33
+
34
+ @lru_cache
35
+ def get_module_name_from_distribution(distro_name: str) -> str | None:
36
+ """Return the first top-level module name for a given distribution name."""
37
+ if not distro_name:
38
+ return None
39
+
40
+ try:
41
+ # Read 'top_level.txt' which contains the module(s) provided by the package
42
+ dist = importlib.metadata.distribution(distro_name)
43
+ # will reading a file set of vun scan?
44
+ top_level = dist.read_text('top_level.txt')
45
+
46
+ if top_level:
47
+ module_names = top_level.strip().split()
48
+ # return firs module name
49
+ return module_names[0]
50
+ except importlib.metadata.PackageNotFoundError:
51
+ # Distribution not found
52
+ return None
53
+ except FileNotFoundError:
54
+ # 'top_level.txt' might be missing
55
+ return None
56
+
57
+ return None
58
+
59
+
60
+ def build_wheel(package_root: str) -> WheelData:
61
+ """Builds a Python .whl for the specified package and saves to disk, sets self._whl_path, and returned as bytes.
62
+
63
+ Args:
64
+ package_root (str): Path to the local package repository.
65
+
66
+ Returns:
67
+ WheelData: Data model containing a built python wheel and its corresponding metadata.
68
+ """
69
+
70
+ import importlib.util
71
+ import re
72
+ import tomllib
73
+
74
+ from pkginfo import Wheel
75
+
76
+ pyproject_toml_path = os.path.join(package_root, "pyproject.toml")
77
+
78
+ if not os.path.exists(pyproject_toml_path):
79
+ raise ValueError("Invalid package path, does not contain a pyproject.toml file.")
80
+
81
+ with open(pyproject_toml_path, "rb") as f:
82
+ data = tomllib.load(f)
83
+
84
+ toml_project: dict = data.get("project", {})
85
+ toml_project_name = toml_project.get("name", None)
86
+
87
+ assert toml_project_name is not None, f"Package name '{toml_project_name}' not found in pyproject.toml"
88
+ # replace "aiqtoolkit" substring with "aiq" to get the import name
89
+ module_name = get_module_name_from_distribution(toml_project_name)
90
+ assert module_name is not None, f"No modules found for package name '{toml_project_name}'"
91
+
92
+ assert importlib.util.find_spec(module_name) is not None, (f"Package {module_name} not "
93
+ "installed, cannot discover components.")
94
+
95
+ toml_packages = set(i for i in data.get("project", {}).get("entry-points", {}).get("aiq.plugins", {}))
96
+ toml_dependencies = set(
97
+ re.search(r"[a-zA-Z][a-zA-Z\d_-]*", package_name).group(0)
98
+ for package_name in toml_project.get("dependencies", []))
99
+
100
+ union_dependencies = toml_dependencies.union(toml_packages)
101
+ union_dependencies.add(toml_project_name)
102
+
103
+ working_dir = os.getcwd()
104
+ os.chdir(package_root)
105
+
106
+ result = subprocess.run(["uv", "build", "--wheel"], check=True)
107
+ result.check_returncode()
108
+
109
+ whl_file = sorted(os.listdir("dist"), reverse=True)[0]
110
+ whl_file_path = os.path.join("dist", whl_file)
111
+
112
+ with open(whl_file_path, "rb") as whl:
113
+ whl_bytes = whl.read()
114
+ whl_base64 = base64.b64encode(whl_bytes).decode("utf-8")
115
+
116
+ whl_path = os.path.join(os.getcwd(), whl_file_path)
117
+
118
+ os.chdir(working_dir)
119
+
120
+ whl_version = Wheel(whl_path).version
121
+
122
+ return WheelData(
123
+ package_root=package_root,
124
+ package_name=module_name, # should it be module name or distro name here
125
+ toml_project=toml_project,
126
+ toml_dependencies=toml_dependencies,
127
+ toml_aiq_packages=toml_packages,
128
+ union_dependencies=union_dependencies,
129
+ whl_path=whl_path,
130
+ whl_base64=whl_base64,
131
+ whl_version=whl_version)
132
+
133
+
134
+ def build_package_metadata(wheel_data: WheelData | None) -> dict[AIQComponentEnum, list[dict | DiscoveryMetadata]]:
135
+ """Loads discovery metadata for all registered AIQ Toolkit components included in this Python package.
136
+
137
+ Args:
138
+ wheel_data (WheelData): Data model containing a built python wheel and its corresponding metadata.
139
+
140
+ Returns:
141
+ dict[AIQComponentEnum, list[typing.Union[dict, DiscoveryMetadata]]]: List containing each components discovery
142
+ metadata.
143
+ """
144
+
145
+ from aiq.cli.type_registry import GlobalTypeRegistry
146
+ from aiq.registry_handlers.metadata_factory import ComponentDiscoveryMetadata
147
+ from aiq.runtime.loader import discover_and_register_plugins
148
+
149
+ discover_and_register_plugins(PluginTypes.ALL)
150
+
151
+ registry = GlobalTypeRegistry.get()
152
+
153
+ aiq_plugins = discover_entrypoints(PluginTypes.ALL)
154
+
155
+ if (wheel_data is not None):
156
+ registry.register_package(package_name=wheel_data.package_name, package_version=wheel_data.whl_version)
157
+ for entry_point in aiq_plugins:
158
+ package_name = entry_point.module.split('.')[0]
159
+ if (package_name == wheel_data.package_name):
160
+ continue
161
+ if (package_name in wheel_data.union_dependencies):
162
+ registry.register_package(package_name=package_name)
163
+
164
+ else:
165
+ for entry_point in aiq_plugins:
166
+ package_name = entry_point.module.split('.')[0]
167
+ registry.register_package(package_name=package_name)
168
+
169
+ discovery_metadata = {}
170
+ for component_type in AIQComponentEnum:
171
+
172
+ if (component_type == AIQComponentEnum.UNDEFINED):
173
+ continue
174
+ component_metadata = ComponentDiscoveryMetadata.from_package_component_type(wheel_data=wheel_data,
175
+ component_type=component_type)
176
+ component_metadata.load_metadata()
177
+ discovery_metadata[component_type] = component_metadata.get_metadata_items()
178
+
179
+ return discovery_metadata
180
+
181
+
182
+ def build_aiq_artifact(package_root: str) -> AIQArtifact:
183
+ """Builds a complete AIQ Toolkit Artifact that can be published for discovery and reuse.
184
+
185
+ Args:
186
+ package_root (str): Path to root of python package
187
+
188
+ Returns:
189
+ AIQArtifact: An publishabla AIQArtifact containing package wheel and discovery metadata.
190
+ """
191
+
192
+ from aiq.registry_handlers.schemas.publish import BuiltAIQArtifact
193
+
194
+ wheel_data = build_wheel(package_root=package_root)
195
+ metadata = build_package_metadata(wheel_data=wheel_data)
196
+ built_artifact = BuiltAIQArtifact(whl=wheel_data.whl_base64, metadata=metadata)
197
+
198
+ return AIQArtifact(artifact=built_artifact, whl_path=wheel_data.whl_path)
File without changes