glaip-sdk 0.0.20__py3-none-any.whl → 0.7.7__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 (216) hide show
  1. glaip_sdk/__init__.py +44 -4
  2. glaip_sdk/_version.py +10 -3
  3. glaip_sdk/agents/__init__.py +27 -0
  4. glaip_sdk/agents/base.py +1250 -0
  5. glaip_sdk/branding.py +15 -6
  6. glaip_sdk/cli/account_store.py +540 -0
  7. glaip_sdk/cli/agent_config.py +2 -6
  8. glaip_sdk/cli/auth.py +271 -45
  9. glaip_sdk/cli/commands/__init__.py +2 -2
  10. glaip_sdk/cli/commands/accounts.py +746 -0
  11. glaip_sdk/cli/commands/agents/__init__.py +119 -0
  12. glaip_sdk/cli/commands/agents/_common.py +561 -0
  13. glaip_sdk/cli/commands/agents/create.py +151 -0
  14. glaip_sdk/cli/commands/agents/delete.py +64 -0
  15. glaip_sdk/cli/commands/agents/get.py +89 -0
  16. glaip_sdk/cli/commands/agents/list.py +129 -0
  17. glaip_sdk/cli/commands/agents/run.py +264 -0
  18. glaip_sdk/cli/commands/agents/sync_langflow.py +72 -0
  19. glaip_sdk/cli/commands/agents/update.py +112 -0
  20. glaip_sdk/cli/commands/common_config.py +104 -0
  21. glaip_sdk/cli/commands/configure.py +734 -143
  22. glaip_sdk/cli/commands/mcps/__init__.py +94 -0
  23. glaip_sdk/cli/commands/mcps/_common.py +459 -0
  24. glaip_sdk/cli/commands/mcps/connect.py +82 -0
  25. glaip_sdk/cli/commands/mcps/create.py +152 -0
  26. glaip_sdk/cli/commands/mcps/delete.py +73 -0
  27. glaip_sdk/cli/commands/mcps/get.py +212 -0
  28. glaip_sdk/cli/commands/mcps/list.py +69 -0
  29. glaip_sdk/cli/commands/mcps/tools.py +235 -0
  30. glaip_sdk/cli/commands/mcps/update.py +190 -0
  31. glaip_sdk/cli/commands/models.py +14 -12
  32. glaip_sdk/cli/commands/shared/__init__.py +21 -0
  33. glaip_sdk/cli/commands/shared/formatters.py +91 -0
  34. glaip_sdk/cli/commands/tools/__init__.py +69 -0
  35. glaip_sdk/cli/commands/tools/_common.py +80 -0
  36. glaip_sdk/cli/commands/tools/create.py +228 -0
  37. glaip_sdk/cli/commands/tools/delete.py +61 -0
  38. glaip_sdk/cli/commands/tools/get.py +103 -0
  39. glaip_sdk/cli/commands/tools/list.py +69 -0
  40. glaip_sdk/cli/commands/tools/script.py +49 -0
  41. glaip_sdk/cli/commands/tools/update.py +102 -0
  42. glaip_sdk/cli/commands/transcripts/__init__.py +90 -0
  43. glaip_sdk/cli/commands/transcripts/_common.py +9 -0
  44. glaip_sdk/cli/commands/transcripts/clear.py +5 -0
  45. glaip_sdk/cli/commands/transcripts/detail.py +5 -0
  46. glaip_sdk/cli/commands/transcripts_original.py +756 -0
  47. glaip_sdk/cli/commands/update.py +164 -23
  48. glaip_sdk/cli/config.py +49 -7
  49. glaip_sdk/cli/constants.py +38 -0
  50. glaip_sdk/cli/context.py +8 -0
  51. glaip_sdk/cli/core/__init__.py +79 -0
  52. glaip_sdk/cli/core/context.py +124 -0
  53. glaip_sdk/cli/core/output.py +851 -0
  54. glaip_sdk/cli/core/prompting.py +649 -0
  55. glaip_sdk/cli/core/rendering.py +187 -0
  56. glaip_sdk/cli/display.py +45 -32
  57. glaip_sdk/cli/entrypoint.py +20 -0
  58. glaip_sdk/cli/hints.py +57 -0
  59. glaip_sdk/cli/io.py +14 -17
  60. glaip_sdk/cli/main.py +344 -167
  61. glaip_sdk/cli/masking.py +21 -33
  62. glaip_sdk/cli/mcp_validators.py +5 -15
  63. glaip_sdk/cli/pager.py +15 -22
  64. glaip_sdk/cli/parsers/__init__.py +1 -3
  65. glaip_sdk/cli/parsers/json_input.py +11 -22
  66. glaip_sdk/cli/resolution.py +5 -10
  67. glaip_sdk/cli/rich_helpers.py +1 -3
  68. glaip_sdk/cli/slash/__init__.py +0 -9
  69. glaip_sdk/cli/slash/accounts_controller.py +580 -0
  70. glaip_sdk/cli/slash/accounts_shared.py +75 -0
  71. glaip_sdk/cli/slash/agent_session.py +65 -29
  72. glaip_sdk/cli/slash/prompt.py +24 -10
  73. glaip_sdk/cli/slash/remote_runs_controller.py +566 -0
  74. glaip_sdk/cli/slash/session.py +827 -232
  75. glaip_sdk/cli/slash/tui/__init__.py +34 -0
  76. glaip_sdk/cli/slash/tui/accounts.tcss +88 -0
  77. glaip_sdk/cli/slash/tui/accounts_app.py +933 -0
  78. glaip_sdk/cli/slash/tui/background_tasks.py +72 -0
  79. glaip_sdk/cli/slash/tui/clipboard.py +147 -0
  80. glaip_sdk/cli/slash/tui/context.py +59 -0
  81. glaip_sdk/cli/slash/tui/keybind_registry.py +235 -0
  82. glaip_sdk/cli/slash/tui/loading.py +58 -0
  83. glaip_sdk/cli/slash/tui/remote_runs_app.py +628 -0
  84. glaip_sdk/cli/slash/tui/terminal.py +402 -0
  85. glaip_sdk/cli/slash/tui/theme/__init__.py +15 -0
  86. glaip_sdk/cli/slash/tui/theme/catalog.py +79 -0
  87. glaip_sdk/cli/slash/tui/theme/manager.py +86 -0
  88. glaip_sdk/cli/slash/tui/theme/tokens.py +55 -0
  89. glaip_sdk/cli/slash/tui/toast.py +123 -0
  90. glaip_sdk/cli/transcript/__init__.py +12 -52
  91. glaip_sdk/cli/transcript/cache.py +258 -60
  92. glaip_sdk/cli/transcript/capture.py +72 -21
  93. glaip_sdk/cli/transcript/history.py +815 -0
  94. glaip_sdk/cli/transcript/launcher.py +1 -3
  95. glaip_sdk/cli/transcript/viewer.py +79 -329
  96. glaip_sdk/cli/update_notifier.py +385 -24
  97. glaip_sdk/cli/validators.py +16 -18
  98. glaip_sdk/client/__init__.py +3 -1
  99. glaip_sdk/client/_schedule_payloads.py +89 -0
  100. glaip_sdk/client/agent_runs.py +147 -0
  101. glaip_sdk/client/agents.py +370 -100
  102. glaip_sdk/client/base.py +78 -35
  103. glaip_sdk/client/hitl.py +136 -0
  104. glaip_sdk/client/main.py +25 -10
  105. glaip_sdk/client/mcps.py +166 -27
  106. glaip_sdk/client/payloads/agent/__init__.py +23 -0
  107. glaip_sdk/client/{_agent_payloads.py → payloads/agent/requests.py} +65 -74
  108. glaip_sdk/client/payloads/agent/responses.py +43 -0
  109. glaip_sdk/client/run_rendering.py +583 -79
  110. glaip_sdk/client/schedules.py +439 -0
  111. glaip_sdk/client/shared.py +21 -0
  112. glaip_sdk/client/tools.py +214 -56
  113. glaip_sdk/client/validators.py +20 -48
  114. glaip_sdk/config/constants.py +11 -0
  115. glaip_sdk/exceptions.py +1 -3
  116. glaip_sdk/hitl/__init__.py +48 -0
  117. glaip_sdk/hitl/base.py +64 -0
  118. glaip_sdk/hitl/callback.py +43 -0
  119. glaip_sdk/hitl/local.py +121 -0
  120. glaip_sdk/hitl/remote.py +523 -0
  121. glaip_sdk/icons.py +9 -3
  122. glaip_sdk/mcps/__init__.py +21 -0
  123. glaip_sdk/mcps/base.py +345 -0
  124. glaip_sdk/models/__init__.py +107 -0
  125. glaip_sdk/models/agent.py +47 -0
  126. glaip_sdk/models/agent_runs.py +117 -0
  127. glaip_sdk/models/common.py +42 -0
  128. glaip_sdk/models/mcp.py +33 -0
  129. glaip_sdk/models/schedule.py +224 -0
  130. glaip_sdk/models/tool.py +33 -0
  131. glaip_sdk/payload_schemas/__init__.py +1 -13
  132. glaip_sdk/payload_schemas/agent.py +1 -3
  133. glaip_sdk/registry/__init__.py +55 -0
  134. glaip_sdk/registry/agent.py +164 -0
  135. glaip_sdk/registry/base.py +139 -0
  136. glaip_sdk/registry/mcp.py +253 -0
  137. glaip_sdk/registry/tool.py +445 -0
  138. glaip_sdk/rich_components.py +58 -2
  139. glaip_sdk/runner/__init__.py +76 -0
  140. glaip_sdk/runner/base.py +84 -0
  141. glaip_sdk/runner/deps.py +112 -0
  142. glaip_sdk/runner/langgraph.py +872 -0
  143. glaip_sdk/runner/logging_config.py +77 -0
  144. glaip_sdk/runner/mcp_adapter/__init__.py +13 -0
  145. glaip_sdk/runner/mcp_adapter/base_mcp_adapter.py +43 -0
  146. glaip_sdk/runner/mcp_adapter/langchain_mcp_adapter.py +257 -0
  147. glaip_sdk/runner/mcp_adapter/mcp_config_builder.py +95 -0
  148. glaip_sdk/runner/tool_adapter/__init__.py +18 -0
  149. glaip_sdk/runner/tool_adapter/base_tool_adapter.py +44 -0
  150. glaip_sdk/runner/tool_adapter/langchain_tool_adapter.py +242 -0
  151. glaip_sdk/schedules/__init__.py +22 -0
  152. glaip_sdk/schedules/base.py +291 -0
  153. glaip_sdk/tools/__init__.py +22 -0
  154. glaip_sdk/tools/base.py +468 -0
  155. glaip_sdk/utils/__init__.py +59 -12
  156. glaip_sdk/utils/a2a/__init__.py +34 -0
  157. glaip_sdk/utils/a2a/event_processor.py +188 -0
  158. glaip_sdk/utils/agent_config.py +4 -14
  159. glaip_sdk/utils/bundler.py +403 -0
  160. glaip_sdk/utils/client.py +111 -0
  161. glaip_sdk/utils/client_utils.py +46 -28
  162. glaip_sdk/utils/datetime_helpers.py +58 -0
  163. glaip_sdk/utils/discovery.py +78 -0
  164. glaip_sdk/utils/display.py +25 -21
  165. glaip_sdk/utils/export.py +143 -0
  166. glaip_sdk/utils/general.py +1 -36
  167. glaip_sdk/utils/import_export.py +15 -16
  168. glaip_sdk/utils/import_resolver.py +524 -0
  169. glaip_sdk/utils/instructions.py +101 -0
  170. glaip_sdk/utils/rendering/__init__.py +115 -1
  171. glaip_sdk/utils/rendering/formatting.py +38 -23
  172. glaip_sdk/utils/rendering/layout/__init__.py +64 -0
  173. glaip_sdk/utils/rendering/{renderer → layout}/panels.py +10 -3
  174. glaip_sdk/utils/rendering/{renderer → layout}/progress.py +73 -12
  175. glaip_sdk/utils/rendering/layout/summary.py +74 -0
  176. glaip_sdk/utils/rendering/layout/transcript.py +606 -0
  177. glaip_sdk/utils/rendering/models.py +18 -8
  178. glaip_sdk/utils/rendering/renderer/__init__.py +9 -51
  179. glaip_sdk/utils/rendering/renderer/base.py +534 -882
  180. glaip_sdk/utils/rendering/renderer/config.py +4 -10
  181. glaip_sdk/utils/rendering/renderer/debug.py +30 -34
  182. glaip_sdk/utils/rendering/renderer/factory.py +138 -0
  183. glaip_sdk/utils/rendering/renderer/stream.py +13 -54
  184. glaip_sdk/utils/rendering/renderer/summary_window.py +79 -0
  185. glaip_sdk/utils/rendering/renderer/thinking.py +273 -0
  186. glaip_sdk/utils/rendering/renderer/toggle.py +182 -0
  187. glaip_sdk/utils/rendering/renderer/tool_panels.py +442 -0
  188. glaip_sdk/utils/rendering/renderer/transcript_mode.py +162 -0
  189. glaip_sdk/utils/rendering/state.py +204 -0
  190. glaip_sdk/utils/rendering/step_tree_state.py +100 -0
  191. glaip_sdk/utils/rendering/steps/__init__.py +34 -0
  192. glaip_sdk/utils/rendering/steps/event_processor.py +778 -0
  193. glaip_sdk/utils/rendering/steps/format.py +176 -0
  194. glaip_sdk/utils/rendering/{steps.py → steps/manager.py} +122 -26
  195. glaip_sdk/utils/rendering/timing.py +36 -0
  196. glaip_sdk/utils/rendering/viewer/__init__.py +21 -0
  197. glaip_sdk/utils/rendering/viewer/presenter.py +184 -0
  198. glaip_sdk/utils/resource_refs.py +29 -26
  199. glaip_sdk/utils/runtime_config.py +425 -0
  200. glaip_sdk/utils/serialization.py +32 -46
  201. glaip_sdk/utils/sync.py +162 -0
  202. glaip_sdk/utils/tool_detection.py +301 -0
  203. glaip_sdk/utils/tool_storage_provider.py +140 -0
  204. glaip_sdk/utils/validation.py +20 -28
  205. {glaip_sdk-0.0.20.dist-info → glaip_sdk-0.7.7.dist-info}/METADATA +78 -23
  206. glaip_sdk-0.7.7.dist-info/RECORD +213 -0
  207. {glaip_sdk-0.0.20.dist-info → glaip_sdk-0.7.7.dist-info}/WHEEL +2 -1
  208. glaip_sdk-0.7.7.dist-info/entry_points.txt +2 -0
  209. glaip_sdk-0.7.7.dist-info/top_level.txt +1 -0
  210. glaip_sdk/cli/commands/agents.py +0 -1412
  211. glaip_sdk/cli/commands/mcps.py +0 -1225
  212. glaip_sdk/cli/commands/tools.py +0 -597
  213. glaip_sdk/cli/utils.py +0 -1330
  214. glaip_sdk/models.py +0 -259
  215. glaip_sdk-0.0.20.dist-info/RECORD +0 -80
  216. glaip_sdk-0.0.20.dist-info/entry_points.txt +0 -3
@@ -0,0 +1,445 @@
1
+ """Tool registry for glaip_sdk.
2
+
3
+ This module provides a ToolRegistry that caches deployed tools
4
+ to avoid redundant API calls when deploying agents with tools.
5
+
6
+ Authors:
7
+ Christian Trisno Sen Long Chen (christian.t.s.l.chen@gdplabs.id)
8
+ """
9
+
10
+ from __future__ import annotations
11
+
12
+ import logging
13
+ from typing import TYPE_CHECKING, Any
14
+
15
+ from glaip_sdk.registry.base import BaseRegistry
16
+
17
+ if TYPE_CHECKING:
18
+ from glaip_sdk.tools import Tool
19
+
20
+ logger = logging.getLogger(__name__)
21
+
22
+
23
+ class ToolRegistry(BaseRegistry["Tool"]):
24
+ """Registry for tools.
25
+
26
+ Resolves tool references to glaip_sdk.models.Tool objects.
27
+ Caches results to avoid redundant API calls and duplicate uploads.
28
+
29
+ Handles:
30
+ - Tool classes (LangChain BaseTool subclasses) → upload, cache, return Tool
31
+ - glaip_sdk.models.Tool → return as-is (uses tool.id)
32
+ - String names → lookup on platform, cache, return Tool
33
+
34
+ Attributes:
35
+ _cache: Internal cache mapping names to Tool objects.
36
+
37
+ Example:
38
+ >>> registry = get_tool_registry()
39
+ >>> tool = registry.resolve(WebSearchTool)
40
+ >>> print(tool.id)
41
+ """
42
+
43
+ def _get_name_from_model_fields(self, ref: type) -> str | None:
44
+ """Extract name from Pydantic model_fields if available."""
45
+ model_fields = getattr(ref, "model_fields", {})
46
+ if "name" not in model_fields:
47
+ return None
48
+ field_info = model_fields["name"]
49
+ default = getattr(field_info, "default", None)
50
+ return default if isinstance(default, str) else None
51
+
52
+ def _get_string_attr(self, obj: Any, attr: str) -> str | None:
53
+ """Get attribute if it's a string, otherwise None."""
54
+ value = getattr(obj, attr, None)
55
+ return value if isinstance(value, str) else None
56
+
57
+ def _extract_name_from_instance(self, ref: Any) -> str | None:
58
+ """Extract name from a non-type instance.
59
+
60
+ Args:
61
+ ref: The instance to extract name from.
62
+
63
+ Returns:
64
+ The extracted name, or None if not found.
65
+ """
66
+ if isinstance(ref, type):
67
+ return None
68
+ return self._get_string_attr(ref, "name")
69
+
70
+ def _extract_name_from_class(self, ref: Any) -> str | None:
71
+ """Extract name from a class.
72
+
73
+ Args:
74
+ ref: The class to extract name from.
75
+
76
+ Returns:
77
+ The extracted name, or None if not found.
78
+ """
79
+ if not isinstance(ref, type):
80
+ return None
81
+ return self._get_string_attr(ref, "name") or self._get_name_from_model_fields(ref)
82
+
83
+ def _extract_name(self, ref: Any) -> str:
84
+ """Extract tool name from a reference.
85
+
86
+ Args:
87
+ ref: A tool class, instance, dict, or string name.
88
+
89
+ Returns:
90
+ The extracted tool name.
91
+
92
+ Raises:
93
+ ValueError: If name cannot be extracted from the reference.
94
+ """
95
+ # Lazy import to avoid circular dependency: Tool -> ToolRegistry -> Tool
96
+ from glaip_sdk.tools.base import Tool # noqa: PLC0415
97
+
98
+ if isinstance(ref, str):
99
+ return ref
100
+
101
+ # Tool instance (from Tool.from_langchain() or Tool.from_native())
102
+ if isinstance(ref, Tool):
103
+ return ref.get_name()
104
+
105
+ # Dict from API response - extract name or id
106
+ if isinstance(ref, dict):
107
+ return ref.get("name") or ref.get("id") or ""
108
+
109
+ # Tool instance (not a class) with name attribute
110
+ name = self._extract_name_from_instance(ref)
111
+ if name:
112
+ return name
113
+
114
+ # Tool class - try direct attribute first, then model_fields
115
+ name = self._extract_name_from_class(ref)
116
+ if name:
117
+ return name
118
+
119
+ raise ValueError(f"Cannot extract name from: {ref}")
120
+
121
+ def _cache_tool(self, tool: Tool, name: str) -> None:
122
+ """Cache a tool by name and ID if available.
123
+
124
+ Args:
125
+ tool: The tool to cache.
126
+ name: The tool name.
127
+ """
128
+ self._cache[name] = tool
129
+ if hasattr(tool, "id") and tool.id:
130
+ self._cache[tool.id] = tool
131
+
132
+ def _resolve_native_platform_tool(self, name: str, tool_class: type | None = None) -> Tool:
133
+ """Find a native tool on the platform and cache it.
134
+
135
+ Args:
136
+ name: The tool name to look up.
137
+ tool_class: Optional local implementation to preserve.
138
+
139
+ Returns:
140
+ The resolved Tool object.
141
+
142
+ Raises:
143
+ ValueError: If the tool is not found on the platform.
144
+ """
145
+ from glaip_sdk.utils.discovery import find_tool # noqa: PLC0415
146
+
147
+ logger.info("Looking up native tool: %s", name)
148
+ tool = find_tool(name)
149
+ if tool:
150
+ # Preserve local implementation if provided
151
+ if tool_class:
152
+ tool.tool_class = tool_class
153
+ self._cache_tool(tool, name)
154
+ return tool
155
+
156
+ raise ValueError(
157
+ f"Native tool '{name}' not found on platform. Ensure the tool is deployed or check for name mismatches."
158
+ )
159
+
160
+ def _resolve_tool_instance(self, ref: Any, name: str) -> Tool | None:
161
+ """Resolve a ToolClass instance.
162
+
163
+ Args:
164
+ ref: The ToolClass instance to resolve.
165
+ name: The extracted tool name.
166
+
167
+ Returns:
168
+ The resolved tool, or None if not a ToolClass instance.
169
+ """
170
+ # Lazy imports to avoid circular dependency: Tool -> ToolRegistry -> Tool
171
+ from glaip_sdk.tools.base import Tool as ToolClass # noqa: PLC0415
172
+ from glaip_sdk.tools.base import ToolType # noqa: PLC0415
173
+
174
+ # Use try/except to handle mocked Tool class in tests
175
+ try:
176
+ is_tool_instance = isinstance(ref, ToolClass)
177
+ except TypeError:
178
+ return None
179
+
180
+ if not is_tool_instance:
181
+ return None
182
+
183
+ # If Tool has an ID, it's already deployed - return as-is
184
+ if ref.id is not None:
185
+ logger.debug("Caching already deployed tool: %s", name)
186
+ self._cache_tool(ref, name)
187
+ return ref
188
+
189
+ # Tool.from_native() - look up on platform
190
+ if ref.tool_type == ToolType.NATIVE:
191
+ return self._resolve_native_platform_tool(name, tool_class=getattr(ref, "tool_class", None))
192
+
193
+ # Tool.from_langchain() - resolve the inner tool_class (promoted or uploaded)
194
+ if ref.tool_class is not None:
195
+ return self._resolve_custom_tool(ref.tool_class, name)
196
+
197
+ # Unresolvable Tool instance - neither native nor has tool_class
198
+ raise ValueError(
199
+ f"Cannot resolve Tool instance: {ref}. "
200
+ f"Tool has no id, is not NATIVE type, and has no tool_class. "
201
+ f"Ensure Tool is created via Tool.from_native() or Tool.from_langchain()."
202
+ )
203
+
204
+ def _resolve_deployed_tool(self, ref: Any, name: str) -> Tool | None:
205
+ """Resolve an already deployed tool (has id/name attributes).
206
+
207
+ Args:
208
+ ref: The tool reference to resolve.
209
+ name: The extracted tool name.
210
+
211
+ Returns:
212
+ The resolved tool, or None if not a deployed tool.
213
+ """
214
+ # Already deployed tool (not a ToolClass, but has id/name)
215
+ # This handles API response objects and backward compatibility
216
+ if not (hasattr(ref, "id") and hasattr(ref, "name") and not isinstance(ref, type)):
217
+ return None
218
+
219
+ if ref.id is not None:
220
+ logger.debug("Caching already deployed tool: %s", name)
221
+ # Use _cache_tool to cache by both name and ID for consistency
222
+ self._cache_tool(ref, name)
223
+ return ref
224
+
225
+ # Tool without ID (backward compatibility) - look up on platform
226
+ return self._resolve_native_platform_tool(name)
227
+
228
+ def _resolve_custom_tool(self, ref: Any, name: str) -> Tool | None:
229
+ """Resolve a custom tool class, promoting aip_agents.tools classes to NATIVE.
230
+
231
+ This method handles two main paths:
232
+ 1. **Promotion**: If the tool class is from `aip_agents.tools`, it is automatically
233
+ promoted to a `NATIVE` tool type. It then performs a platform lookup to link it
234
+ with the deployed native tool while preserving the local `tool_class` for local execution.
235
+ 2. **Upload**: If it is a standard LangChain tool, it is uploaded to the platform
236
+ as a custom tool.
237
+
238
+ Args:
239
+ ref: The tool reference (usually a class) to resolve.
240
+ name: The extracted tool name.
241
+
242
+ Returns:
243
+ The resolved tool, or None if not a custom tool.
244
+ """
245
+ # aip_agents tools are automatically promoted to NATIVE
246
+ if self._is_aip_agents_tool(ref):
247
+ from glaip_sdk.utils.tool_detection import get_tool_name # noqa: PLC0415
248
+
249
+ # Get name from class attribute or field
250
+ tool_name = get_tool_name(ref)
251
+ if tool_name is None:
252
+ raise ValueError(f"Tool class {ref.__name__} has no 'name' attribute")
253
+
254
+ return self._resolve_native_platform_tool(tool_name, tool_class=ref)
255
+
256
+ if not self._is_custom_tool(ref):
257
+ return None
258
+
259
+ # Regular custom tools - upload to platform
260
+ from glaip_sdk.utils.sync import update_or_create_tool # noqa: PLC0415
261
+
262
+ logger.info("Uploading custom tool: %s", name)
263
+ tool = update_or_create_tool(ref)
264
+
265
+ # Cache the resolved tool
266
+ self._cache_tool(tool, name)
267
+ if hasattr(tool, "id") and tool.id:
268
+ self._cache[tool.id] = tool
269
+
270
+ return tool
271
+
272
+ def _resolve_dict_tool(self, ref: Any, name: str) -> Tool | None:
273
+ """Resolve a tool from a dict (API response).
274
+
275
+ Args:
276
+ ref: The dict to resolve.
277
+ name: The extracted tool name.
278
+
279
+ Returns:
280
+ The resolved tool, or None if not a dict.
281
+ """
282
+ # Lazy imports to avoid circular dependency
283
+ from glaip_sdk.tools.base import Tool as ToolClass # noqa: PLC0415
284
+
285
+ if not isinstance(ref, dict):
286
+ return None
287
+
288
+ tool_id = ref.get("id")
289
+ if tool_id:
290
+ tool = ToolClass(id=tool_id, name=ref.get("name", ""))
291
+ # Use _cache_tool to cache by both name and ID for consistency
292
+ self._cache_tool(tool, name)
293
+ return tool
294
+ raise ValueError(f"Tool dict missing 'id': {ref}")
295
+
296
+ def _resolve_string_tool(self, ref: Any, name: str) -> Tool | None:
297
+ """Resolve a tool from a string name.
298
+
299
+ Args:
300
+ ref: The string to resolve.
301
+ name: The extracted tool name.
302
+
303
+ Returns:
304
+ The resolved tool, or None if not a string.
305
+ """
306
+ if not isinstance(ref, str):
307
+ return None
308
+
309
+ return self._resolve_native_platform_tool(name)
310
+
311
+ def _resolve_and_cache(self, ref: Any, name: str) -> Tool:
312
+ """Resolve tool reference - upload if class, find if string/native.
313
+
314
+ Args:
315
+ ref: The tool reference to resolve.
316
+ name: The extracted tool name.
317
+
318
+ Returns:
319
+ The resolved glaip_sdk.models.Tool object.
320
+
321
+ Raises:
322
+ ValueError: If tool cannot be resolved.
323
+ """
324
+ # Try each resolution strategy in order
325
+ resolvers = [
326
+ self._resolve_tool_instance,
327
+ self._resolve_deployed_tool,
328
+ self._resolve_custom_tool,
329
+ self._resolve_dict_tool,
330
+ self._resolve_string_tool,
331
+ ]
332
+
333
+ for resolver in resolvers:
334
+ result = resolver(ref, name)
335
+ if result is not None:
336
+ return result
337
+
338
+ raise ValueError(f"Could not resolve tool reference: {ref}")
339
+
340
+ def _is_aip_agents_tool(self, ref: Any) -> bool:
341
+ """Check if reference is an aip-agents tool.
342
+
343
+ Args:
344
+ ref: The reference to check.
345
+
346
+ Returns:
347
+ True if ref is from aip_agents.tools package.
348
+ """
349
+ try:
350
+ from glaip_sdk.utils.tool_detection import ( # noqa: PLC0415
351
+ is_aip_agents_tool,
352
+ )
353
+ except ImportError:
354
+ return False
355
+
356
+ return is_aip_agents_tool(ref)
357
+
358
+ def _is_custom_tool(self, ref: Any) -> bool:
359
+ """Check if reference is a custom tool class/instance.
360
+
361
+ Args:
362
+ ref: The reference to check.
363
+
364
+ Returns:
365
+ True if ref is a custom tool that needs uploading.
366
+ """
367
+ try:
368
+ from glaip_sdk.utils.tool_detection import ( # noqa: PLC0415
369
+ is_langchain_tool,
370
+ )
371
+
372
+ is_tool = is_langchain_tool(ref)
373
+ except ImportError:
374
+ is_tool = hasattr(ref, "args_schema") or hasattr(ref, "_run")
375
+ if is_tool:
376
+ logger.warning("tool_detection module missing; identifying tool via fallback attributes.")
377
+
378
+ # aip_agents tools are NOT custom - they're native
379
+ if is_tool and self._is_aip_agents_tool(ref):
380
+ return False
381
+
382
+ return is_tool
383
+
384
+ def resolve(self, ref: Any) -> Tool:
385
+ """Resolve a tool reference to a platform Tool object.
386
+
387
+ Overrides base resolve to handle SDK tools differently.
388
+
389
+ Args:
390
+ ref: The tool reference to resolve.
391
+
392
+ Returns:
393
+ The resolved glaip_sdk.models.Tool object.
394
+ """
395
+ # Check if it's a Tool instance (not a class)
396
+ if hasattr(ref, "id") and hasattr(ref, "name") and not isinstance(ref, type):
397
+ # If Tool has an ID, it's already deployed - return as-is
398
+ if ref.id is not None:
399
+ name = self._extract_name(ref)
400
+ if name not in self._cache:
401
+ # Use _cache_tool to cache by both name and ID for consistency
402
+ self._cache_tool(ref, name)
403
+ return ref
404
+ # Tool without ID (e.g., from Tool.from_native()) - needs platform lookup
405
+ # Fall through to normal resolution
406
+
407
+ return super().resolve(ref)
408
+
409
+
410
+ class _ToolRegistrySingleton:
411
+ """Singleton holder for ToolRegistry to avoid global statement."""
412
+
413
+ _instance: ToolRegistry | None = None
414
+
415
+ @classmethod
416
+ def get_instance(cls) -> ToolRegistry:
417
+ """Get or create the singleton instance.
418
+
419
+ Returns:
420
+ The global ToolRegistry instance.
421
+ """
422
+ if cls._instance is None:
423
+ cls._instance = ToolRegistry()
424
+ return cls._instance
425
+
426
+ @classmethod
427
+ def reset(cls) -> None:
428
+ """Reset the singleton instance (for testing)."""
429
+ cls._instance = None
430
+
431
+
432
+ def get_tool_registry() -> ToolRegistry:
433
+ """Get the singleton ToolRegistry instance.
434
+
435
+ Returns a global ToolRegistry that caches tools across the session.
436
+
437
+ Returns:
438
+ The global ToolRegistry instance.
439
+
440
+ Example:
441
+ >>> from glaip_sdk.registry import get_tool_registry
442
+ >>> registry = get_tool_registry()
443
+ >>> tool = registry.resolve("web_search")
444
+ """
445
+ return _ToolRegistrySingleton.get_instance()
@@ -1,10 +1,15 @@
1
- """Custom Rich components with copy-friendly defaults."""
1
+ """Custom Rich components with copy-friendly defaults.
2
+
3
+ Authors:
4
+ Raymond Christopher (raymond.christopher@gdplabs.id)
5
+ """
2
6
 
3
7
  from __future__ import annotations
4
8
 
5
9
  from rich import box
6
10
  from rich.panel import Panel
7
11
  from rich.table import Table
12
+ from rich.text import Text
8
13
 
9
14
 
10
15
  class AIPPanel(Panel):
@@ -66,4 +71,55 @@ class AIPGrid(Table):
66
71
  )
67
72
 
68
73
 
69
- __all__ = ["AIPPanel", "AIPTable", "AIPGrid"]
74
+ class RemoteRunsTable(AIPTable):
75
+ """Rich Table for displaying remote agent runs with pagination support."""
76
+
77
+ def __init__(self, *args, **kwargs):
78
+ """Initialize RemoteRunsTable with columns for run display.
79
+
80
+ Args:
81
+ *args: Positional arguments passed to AIPTable
82
+ **kwargs: Keyword arguments passed to AIPTable
83
+ """
84
+ kwargs.setdefault("row_styles", ("dim", "none"))
85
+ kwargs.setdefault("show_header", True)
86
+ super().__init__(*args, **kwargs)
87
+ # Add columns for run display
88
+ self.add_column("", width=2, no_wrap=True) # Selection gutter
89
+ self.add_column("Run UUID", style="cyan", width=36, no_wrap=True)
90
+ self.add_column("Type", style="yellow", width=8, no_wrap=True)
91
+ self.add_column("Status", style="magenta", width=12, no_wrap=True)
92
+ self.add_column("Started (UTC)", style="dim", width=20, no_wrap=True)
93
+ self.add_column("Completed (UTC)", style="dim", width=20, no_wrap=True)
94
+ self.add_column("Duration", style="green", width=10, no_wrap=True)
95
+ self.add_column("Input Preview", style="white", width=40, overflow="ellipsis")
96
+
97
+ def add_run_row(
98
+ self,
99
+ run_uuid: str,
100
+ run_type: str,
101
+ status: str,
102
+ started: str,
103
+ completed: str,
104
+ duration: str,
105
+ preview: str,
106
+ *,
107
+ selected: bool = False,
108
+ ) -> None:
109
+ """Append a run row with optional selection styling."""
110
+ gutter = Text("› ", style="bold bright_cyan") if selected else Text(" ")
111
+ row_style = "reverse" if selected else None
112
+ self.add_row(
113
+ gutter,
114
+ run_uuid,
115
+ run_type,
116
+ status,
117
+ started,
118
+ completed,
119
+ duration,
120
+ preview,
121
+ style=row_style,
122
+ )
123
+
124
+
125
+ __all__ = ["AIPPanel", "AIPTable", "AIPGrid", "RemoteRunsTable"]
@@ -0,0 +1,76 @@
1
+ """Local agent execution runners.
2
+
3
+ This module provides runners for executing glaip-sdk agents locally
4
+ without requiring the AIP backend server. The primary runner is
5
+ LangGraphRunner which uses the aip-agents library.
6
+
7
+ To use local execution, install with the [local] extra:
8
+ pip install "glaip-sdk[local]"
9
+
10
+ Authors:
11
+ Christian Trisno Sen Long Chen (christian.t.s.l.chen@gdplabs.id)
12
+
13
+ Example:
14
+ >>> from glaip_sdk.runner import get_default_runner
15
+ >>> from glaip_sdk.agents import Agent
16
+ >>>
17
+ >>> agent = Agent(name="my-agent", instruction="You are helpful.")
18
+ >>> runner = get_default_runner()
19
+ >>> result = runner.run(agent, "Hello!")
20
+ """
21
+
22
+ from typing import TYPE_CHECKING, Any
23
+
24
+ from glaip_sdk.runner.deps import (
25
+ LOCAL_RUNTIME_AVAILABLE,
26
+ check_local_runtime_available,
27
+ get_local_runtime_missing_message,
28
+ )
29
+
30
+ # Default runner instance
31
+ _default_runner: Any | None = None
32
+
33
+
34
+ def get_default_runner() -> Any:
35
+ """Get the default runner instance for local agent execution.
36
+
37
+ Returns:
38
+ The default LangGraphRunner instance.
39
+
40
+ Raises:
41
+ RuntimeError: If local runtime dependencies are not available.
42
+ """
43
+ global _default_runner
44
+
45
+ if not check_local_runtime_available():
46
+ raise RuntimeError(get_local_runtime_missing_message())
47
+
48
+ if _default_runner is None:
49
+ # Lazy import to avoid requiring aip-agents when runner is not used
50
+ from glaip_sdk.runner.langgraph import LangGraphRunner # noqa: PLC0415
51
+
52
+ _default_runner = LangGraphRunner()
53
+
54
+ return _default_runner
55
+
56
+
57
+ if TYPE_CHECKING:
58
+ from glaip_sdk.runner.langgraph import LangGraphRunner
59
+
60
+ __all__ = [
61
+ "LOCAL_RUNTIME_AVAILABLE",
62
+ "LangGraphRunner",
63
+ "check_local_runtime_available",
64
+ "get_default_runner",
65
+ "get_local_runtime_missing_message",
66
+ ]
67
+
68
+
69
+ def __getattr__(name: str) -> Any:
70
+ """Lazy import for LangGraphRunner to avoid requiring aip-agents when not used."""
71
+ if name == "LangGraphRunner":
72
+ from glaip_sdk.runner.langgraph import LangGraphRunner # noqa: PLC0415
73
+
74
+ globals()["LangGraphRunner"] = LangGraphRunner
75
+ return LangGraphRunner
76
+ raise AttributeError(f"module {__name__!r} has no attribute {name!r}")
@@ -0,0 +1,84 @@
1
+ """Abstract base class for agent execution runners.
2
+
3
+ Authors:
4
+ Christian Trisno Sen Long Chen (christian.t.s.l.chen@gdplabs.id)
5
+ """
6
+
7
+ from __future__ import annotations
8
+
9
+ from abc import ABC, abstractmethod
10
+ from typing import TYPE_CHECKING, Any
11
+
12
+ if TYPE_CHECKING:
13
+ from glaip_sdk.agents.base import Agent
14
+
15
+
16
+ class BaseRunner(ABC):
17
+ """Abstract base class for agent execution runners.
18
+
19
+ Runners are responsible for executing glaip-sdk Agent instances
20
+ and returning results. Different runner implementations may use
21
+ different execution backends (LangGraph, Google ADK, etc.).
22
+ """
23
+
24
+ @abstractmethod
25
+ def run(
26
+ self,
27
+ agent: Agent,
28
+ message: str,
29
+ verbose: bool = False,
30
+ runtime_config: dict[str, Any] | None = None,
31
+ chat_history: list[dict[str, str]] | None = None,
32
+ **kwargs: Any,
33
+ ) -> str:
34
+ """Execute agent synchronously and return final response text.
35
+
36
+ Args:
37
+ agent: The glaip_sdk Agent to execute.
38
+ message: The user message to send to the agent.
39
+ verbose: If True, emit debug trace output during execution.
40
+ Defaults to False.
41
+ runtime_config: Optional runtime configuration for tools, MCPs, etc.
42
+ Defaults to None.
43
+ chat_history: Optional list of prior conversation messages.
44
+ Defaults to None.
45
+ **kwargs: Additional keyword arguments passed to the backend.
46
+
47
+ Returns:
48
+ The final response text from the agent.
49
+
50
+ Raises:
51
+ RuntimeError: If execution fails.
52
+ """
53
+ ...
54
+
55
+ @abstractmethod
56
+ async def arun(
57
+ self,
58
+ agent: Agent,
59
+ message: str,
60
+ verbose: bool = False,
61
+ runtime_config: dict[str, Any] | None = None,
62
+ chat_history: list[dict[str, str]] | None = None,
63
+ **kwargs: Any,
64
+ ) -> str:
65
+ """Execute agent asynchronously and return final response text.
66
+
67
+ Args:
68
+ agent: The glaip_sdk Agent to execute.
69
+ message: The user message to send to the agent.
70
+ verbose: If True, emit debug trace output during execution.
71
+ Defaults to False.
72
+ runtime_config: Optional runtime configuration for tools, MCPs, etc.
73
+ Defaults to None.
74
+ chat_history: Optional list of prior conversation messages.
75
+ Defaults to None.
76
+ **kwargs: Additional keyword arguments passed to the backend.
77
+
78
+ Returns:
79
+ The final response text from the agent.
80
+
81
+ Raises:
82
+ RuntimeError: If execution fails.
83
+ """
84
+ ...