aip-agents-binary 0.5.20__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 (280) hide show
  1. aip_agents/__init__.py +65 -0
  2. aip_agents/a2a/__init__.py +19 -0
  3. aip_agents/a2a/server/__init__.py +10 -0
  4. aip_agents/a2a/server/base_executor.py +1086 -0
  5. aip_agents/a2a/server/google_adk_executor.py +198 -0
  6. aip_agents/a2a/server/langflow_executor.py +180 -0
  7. aip_agents/a2a/server/langgraph_executor.py +270 -0
  8. aip_agents/a2a/types.py +232 -0
  9. aip_agents/agent/__init__.py +27 -0
  10. aip_agents/agent/base_agent.py +970 -0
  11. aip_agents/agent/base_langgraph_agent.py +2942 -0
  12. aip_agents/agent/google_adk_agent.py +926 -0
  13. aip_agents/agent/google_adk_constants.py +6 -0
  14. aip_agents/agent/hitl/__init__.py +24 -0
  15. aip_agents/agent/hitl/config.py +28 -0
  16. aip_agents/agent/hitl/langgraph_hitl_mixin.py +515 -0
  17. aip_agents/agent/hitl/manager.py +532 -0
  18. aip_agents/agent/hitl/models.py +18 -0
  19. aip_agents/agent/hitl/prompt/__init__.py +9 -0
  20. aip_agents/agent/hitl/prompt/base.py +42 -0
  21. aip_agents/agent/hitl/prompt/deferred.py +73 -0
  22. aip_agents/agent/hitl/registry.py +149 -0
  23. aip_agents/agent/interface.py +138 -0
  24. aip_agents/agent/interfaces.py +65 -0
  25. aip_agents/agent/langflow_agent.py +464 -0
  26. aip_agents/agent/langgraph_memory_enhancer_agent.py +433 -0
  27. aip_agents/agent/langgraph_react_agent.py +2514 -0
  28. aip_agents/agent/system_instruction_context.py +34 -0
  29. aip_agents/clients/__init__.py +10 -0
  30. aip_agents/clients/langflow/__init__.py +10 -0
  31. aip_agents/clients/langflow/client.py +477 -0
  32. aip_agents/clients/langflow/types.py +18 -0
  33. aip_agents/constants.py +23 -0
  34. aip_agents/credentials/manager.py +132 -0
  35. aip_agents/examples/__init__.py +5 -0
  36. aip_agents/examples/compare_streaming_client.py +783 -0
  37. aip_agents/examples/compare_streaming_server.py +142 -0
  38. aip_agents/examples/demo_memory_recall.py +401 -0
  39. aip_agents/examples/hello_world_a2a_google_adk_client.py +49 -0
  40. aip_agents/examples/hello_world_a2a_google_adk_client_agent.py +48 -0
  41. aip_agents/examples/hello_world_a2a_google_adk_client_streaming.py +60 -0
  42. aip_agents/examples/hello_world_a2a_google_adk_server.py +79 -0
  43. aip_agents/examples/hello_world_a2a_langchain_client.py +39 -0
  44. aip_agents/examples/hello_world_a2a_langchain_client_agent.py +39 -0
  45. aip_agents/examples/hello_world_a2a_langchain_client_lm_invoker.py +37 -0
  46. aip_agents/examples/hello_world_a2a_langchain_client_streaming.py +41 -0
  47. aip_agents/examples/hello_world_a2a_langchain_reference_client_streaming.py +60 -0
  48. aip_agents/examples/hello_world_a2a_langchain_reference_server.py +105 -0
  49. aip_agents/examples/hello_world_a2a_langchain_server.py +79 -0
  50. aip_agents/examples/hello_world_a2a_langchain_server_lm_invoker.py +78 -0
  51. aip_agents/examples/hello_world_a2a_langflow_client.py +83 -0
  52. aip_agents/examples/hello_world_a2a_langflow_server.py +82 -0
  53. aip_agents/examples/hello_world_a2a_langgraph_artifact_client.py +73 -0
  54. aip_agents/examples/hello_world_a2a_langgraph_artifact_client_streaming.py +76 -0
  55. aip_agents/examples/hello_world_a2a_langgraph_artifact_server.py +92 -0
  56. aip_agents/examples/hello_world_a2a_langgraph_client.py +54 -0
  57. aip_agents/examples/hello_world_a2a_langgraph_client_agent.py +54 -0
  58. aip_agents/examples/hello_world_a2a_langgraph_client_agent_lm_invoker.py +32 -0
  59. aip_agents/examples/hello_world_a2a_langgraph_client_streaming.py +50 -0
  60. aip_agents/examples/hello_world_a2a_langgraph_client_streaming_lm_invoker.py +44 -0
  61. aip_agents/examples/hello_world_a2a_langgraph_client_streaming_tool_streaming.py +92 -0
  62. aip_agents/examples/hello_world_a2a_langgraph_server.py +84 -0
  63. aip_agents/examples/hello_world_a2a_langgraph_server_lm_invoker.py +79 -0
  64. aip_agents/examples/hello_world_a2a_langgraph_server_tool_streaming.py +132 -0
  65. aip_agents/examples/hello_world_a2a_mcp_langgraph.py +196 -0
  66. aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_client.py +244 -0
  67. aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_server.py +251 -0
  68. aip_agents/examples/hello_world_a2a_with_metadata_langchain_client.py +57 -0
  69. aip_agents/examples/hello_world_a2a_with_metadata_langchain_server_lm_invoker.py +80 -0
  70. aip_agents/examples/hello_world_google_adk.py +41 -0
  71. aip_agents/examples/hello_world_google_adk_mcp_http.py +34 -0
  72. aip_agents/examples/hello_world_google_adk_mcp_http_stream.py +40 -0
  73. aip_agents/examples/hello_world_google_adk_mcp_sse.py +44 -0
  74. aip_agents/examples/hello_world_google_adk_mcp_sse_stream.py +48 -0
  75. aip_agents/examples/hello_world_google_adk_mcp_stdio.py +44 -0
  76. aip_agents/examples/hello_world_google_adk_mcp_stdio_stream.py +48 -0
  77. aip_agents/examples/hello_world_google_adk_stream.py +44 -0
  78. aip_agents/examples/hello_world_langchain.py +28 -0
  79. aip_agents/examples/hello_world_langchain_lm_invoker.py +15 -0
  80. aip_agents/examples/hello_world_langchain_mcp_http.py +34 -0
  81. aip_agents/examples/hello_world_langchain_mcp_http_interactive.py +130 -0
  82. aip_agents/examples/hello_world_langchain_mcp_http_stream.py +42 -0
  83. aip_agents/examples/hello_world_langchain_mcp_multi_server.py +155 -0
  84. aip_agents/examples/hello_world_langchain_mcp_sse.py +34 -0
  85. aip_agents/examples/hello_world_langchain_mcp_sse_stream.py +40 -0
  86. aip_agents/examples/hello_world_langchain_mcp_stdio.py +30 -0
  87. aip_agents/examples/hello_world_langchain_mcp_stdio_stream.py +41 -0
  88. aip_agents/examples/hello_world_langchain_stream.py +36 -0
  89. aip_agents/examples/hello_world_langchain_stream_lm_invoker.py +39 -0
  90. aip_agents/examples/hello_world_langflow_agent.py +163 -0
  91. aip_agents/examples/hello_world_langgraph.py +39 -0
  92. aip_agents/examples/hello_world_langgraph_bosa_twitter.py +41 -0
  93. aip_agents/examples/hello_world_langgraph_mcp_http.py +31 -0
  94. aip_agents/examples/hello_world_langgraph_mcp_http_stream.py +34 -0
  95. aip_agents/examples/hello_world_langgraph_mcp_sse.py +35 -0
  96. aip_agents/examples/hello_world_langgraph_mcp_sse_stream.py +50 -0
  97. aip_agents/examples/hello_world_langgraph_mcp_stdio.py +35 -0
  98. aip_agents/examples/hello_world_langgraph_mcp_stdio_stream.py +50 -0
  99. aip_agents/examples/hello_world_langgraph_stream.py +43 -0
  100. aip_agents/examples/hello_world_langgraph_stream_lm_invoker.py +37 -0
  101. aip_agents/examples/hello_world_model_switch_cli.py +210 -0
  102. aip_agents/examples/hello_world_multi_agent_adk.py +75 -0
  103. aip_agents/examples/hello_world_multi_agent_langchain.py +54 -0
  104. aip_agents/examples/hello_world_multi_agent_langgraph.py +66 -0
  105. aip_agents/examples/hello_world_multi_agent_langgraph_lm_invoker.py +69 -0
  106. aip_agents/examples/hello_world_pii_logger.py +21 -0
  107. aip_agents/examples/hello_world_sentry.py +133 -0
  108. aip_agents/examples/hello_world_step_limits.py +273 -0
  109. aip_agents/examples/hello_world_stock_a2a_server.py +103 -0
  110. aip_agents/examples/hello_world_tool_output_client.py +46 -0
  111. aip_agents/examples/hello_world_tool_output_server.py +114 -0
  112. aip_agents/examples/hitl_demo.py +724 -0
  113. aip_agents/examples/mcp_configs/configs.py +63 -0
  114. aip_agents/examples/mcp_servers/common.py +76 -0
  115. aip_agents/examples/mcp_servers/mcp_name.py +29 -0
  116. aip_agents/examples/mcp_servers/mcp_server_http.py +19 -0
  117. aip_agents/examples/mcp_servers/mcp_server_sse.py +19 -0
  118. aip_agents/examples/mcp_servers/mcp_server_stdio.py +19 -0
  119. aip_agents/examples/mcp_servers/mcp_time.py +10 -0
  120. aip_agents/examples/pii_demo_langgraph_client.py +69 -0
  121. aip_agents/examples/pii_demo_langgraph_server.py +126 -0
  122. aip_agents/examples/pii_demo_multi_agent_client.py +80 -0
  123. aip_agents/examples/pii_demo_multi_agent_server.py +247 -0
  124. aip_agents/examples/todolist_planning_a2a_langchain_client.py +70 -0
  125. aip_agents/examples/todolist_planning_a2a_langgraph_server.py +88 -0
  126. aip_agents/examples/tools/__init__.py +27 -0
  127. aip_agents/examples/tools/adk_arithmetic_tools.py +36 -0
  128. aip_agents/examples/tools/adk_weather_tool.py +60 -0
  129. aip_agents/examples/tools/data_generator_tool.py +103 -0
  130. aip_agents/examples/tools/data_visualization_tool.py +312 -0
  131. aip_agents/examples/tools/image_artifact_tool.py +136 -0
  132. aip_agents/examples/tools/langchain_arithmetic_tools.py +26 -0
  133. aip_agents/examples/tools/langchain_currency_exchange_tool.py +88 -0
  134. aip_agents/examples/tools/langchain_graph_artifact_tool.py +172 -0
  135. aip_agents/examples/tools/langchain_weather_tool.py +48 -0
  136. aip_agents/examples/tools/langgraph_streaming_tool.py +130 -0
  137. aip_agents/examples/tools/mock_retrieval_tool.py +56 -0
  138. aip_agents/examples/tools/pii_demo_tools.py +189 -0
  139. aip_agents/examples/tools/random_chart_tool.py +142 -0
  140. aip_agents/examples/tools/serper_tool.py +202 -0
  141. aip_agents/examples/tools/stock_tools.py +82 -0
  142. aip_agents/examples/tools/table_generator_tool.py +167 -0
  143. aip_agents/examples/tools/time_tool.py +82 -0
  144. aip_agents/examples/tools/weather_forecast_tool.py +38 -0
  145. aip_agents/executor/agent_executor.py +473 -0
  146. aip_agents/executor/base.py +48 -0
  147. aip_agents/mcp/__init__.py +1 -0
  148. aip_agents/mcp/client/__init__.py +14 -0
  149. aip_agents/mcp/client/base_mcp_client.py +369 -0
  150. aip_agents/mcp/client/connection_manager.py +193 -0
  151. aip_agents/mcp/client/google_adk/__init__.py +11 -0
  152. aip_agents/mcp/client/google_adk/client.py +381 -0
  153. aip_agents/mcp/client/langchain/__init__.py +11 -0
  154. aip_agents/mcp/client/langchain/client.py +265 -0
  155. aip_agents/mcp/client/persistent_session.py +359 -0
  156. aip_agents/mcp/client/session_pool.py +351 -0
  157. aip_agents/mcp/client/transports.py +215 -0
  158. aip_agents/mcp/utils/__init__.py +7 -0
  159. aip_agents/mcp/utils/config_validator.py +139 -0
  160. aip_agents/memory/__init__.py +14 -0
  161. aip_agents/memory/adapters/__init__.py +10 -0
  162. aip_agents/memory/adapters/base_adapter.py +717 -0
  163. aip_agents/memory/adapters/mem0.py +84 -0
  164. aip_agents/memory/base.py +84 -0
  165. aip_agents/memory/constants.py +49 -0
  166. aip_agents/memory/factory.py +86 -0
  167. aip_agents/memory/guidance.py +20 -0
  168. aip_agents/memory/simple_memory.py +47 -0
  169. aip_agents/middleware/__init__.py +17 -0
  170. aip_agents/middleware/base.py +88 -0
  171. aip_agents/middleware/manager.py +128 -0
  172. aip_agents/middleware/todolist.py +274 -0
  173. aip_agents/schema/__init__.py +69 -0
  174. aip_agents/schema/a2a.py +56 -0
  175. aip_agents/schema/agent.py +111 -0
  176. aip_agents/schema/hitl.py +157 -0
  177. aip_agents/schema/langgraph.py +37 -0
  178. aip_agents/schema/model_id.py +97 -0
  179. aip_agents/schema/step_limit.py +108 -0
  180. aip_agents/schema/storage.py +40 -0
  181. aip_agents/sentry/__init__.py +11 -0
  182. aip_agents/sentry/sentry.py +151 -0
  183. aip_agents/storage/__init__.py +41 -0
  184. aip_agents/storage/base.py +85 -0
  185. aip_agents/storage/clients/__init__.py +12 -0
  186. aip_agents/storage/clients/minio_client.py +318 -0
  187. aip_agents/storage/config.py +62 -0
  188. aip_agents/storage/providers/__init__.py +15 -0
  189. aip_agents/storage/providers/base.py +106 -0
  190. aip_agents/storage/providers/memory.py +114 -0
  191. aip_agents/storage/providers/object_storage.py +214 -0
  192. aip_agents/tools/__init__.py +33 -0
  193. aip_agents/tools/bosa_tools.py +105 -0
  194. aip_agents/tools/browser_use/__init__.py +82 -0
  195. aip_agents/tools/browser_use/action_parser.py +103 -0
  196. aip_agents/tools/browser_use/browser_use_tool.py +1112 -0
  197. aip_agents/tools/browser_use/llm_config.py +120 -0
  198. aip_agents/tools/browser_use/minio_storage.py +198 -0
  199. aip_agents/tools/browser_use/schemas.py +119 -0
  200. aip_agents/tools/browser_use/session.py +76 -0
  201. aip_agents/tools/browser_use/session_errors.py +132 -0
  202. aip_agents/tools/browser_use/steel_session_recording.py +317 -0
  203. aip_agents/tools/browser_use/streaming.py +813 -0
  204. aip_agents/tools/browser_use/structured_data_parser.py +257 -0
  205. aip_agents/tools/browser_use/structured_data_recovery.py +204 -0
  206. aip_agents/tools/browser_use/types.py +78 -0
  207. aip_agents/tools/code_sandbox/__init__.py +26 -0
  208. aip_agents/tools/code_sandbox/constant.py +13 -0
  209. aip_agents/tools/code_sandbox/e2b_cloud_sandbox_extended.py +257 -0
  210. aip_agents/tools/code_sandbox/e2b_sandbox_tool.py +411 -0
  211. aip_agents/tools/constants.py +165 -0
  212. aip_agents/tools/document_loader/__init__.py +44 -0
  213. aip_agents/tools/document_loader/base_reader.py +302 -0
  214. aip_agents/tools/document_loader/docx_reader_tool.py +68 -0
  215. aip_agents/tools/document_loader/excel_reader_tool.py +171 -0
  216. aip_agents/tools/document_loader/pdf_reader_tool.py +79 -0
  217. aip_agents/tools/document_loader/pdf_splitter.py +169 -0
  218. aip_agents/tools/gl_connector/__init__.py +5 -0
  219. aip_agents/tools/gl_connector/tool.py +351 -0
  220. aip_agents/tools/memory_search/__init__.py +22 -0
  221. aip_agents/tools/memory_search/base.py +200 -0
  222. aip_agents/tools/memory_search/mem0.py +258 -0
  223. aip_agents/tools/memory_search/schema.py +48 -0
  224. aip_agents/tools/memory_search_tool.py +26 -0
  225. aip_agents/tools/time_tool.py +117 -0
  226. aip_agents/tools/tool_config_injector.py +300 -0
  227. aip_agents/tools/web_search/__init__.py +15 -0
  228. aip_agents/tools/web_search/serper_tool.py +187 -0
  229. aip_agents/types/__init__.py +70 -0
  230. aip_agents/types/a2a_events.py +13 -0
  231. aip_agents/utils/__init__.py +79 -0
  232. aip_agents/utils/a2a_connector.py +1757 -0
  233. aip_agents/utils/artifact_helpers.py +502 -0
  234. aip_agents/utils/constants.py +22 -0
  235. aip_agents/utils/datetime/__init__.py +34 -0
  236. aip_agents/utils/datetime/normalization.py +231 -0
  237. aip_agents/utils/datetime/timezone.py +206 -0
  238. aip_agents/utils/env_loader.py +27 -0
  239. aip_agents/utils/event_handler_registry.py +58 -0
  240. aip_agents/utils/file_prompt_utils.py +176 -0
  241. aip_agents/utils/final_response_builder.py +211 -0
  242. aip_agents/utils/formatter_llm_client.py +231 -0
  243. aip_agents/utils/langgraph/__init__.py +19 -0
  244. aip_agents/utils/langgraph/converter.py +128 -0
  245. aip_agents/utils/langgraph/tool_managers/__init__.py +15 -0
  246. aip_agents/utils/langgraph/tool_managers/a2a_tool_manager.py +99 -0
  247. aip_agents/utils/langgraph/tool_managers/base_tool_manager.py +66 -0
  248. aip_agents/utils/langgraph/tool_managers/delegation_tool_manager.py +1071 -0
  249. aip_agents/utils/langgraph/tool_output_management.py +967 -0
  250. aip_agents/utils/logger.py +195 -0
  251. aip_agents/utils/metadata/__init__.py +27 -0
  252. aip_agents/utils/metadata/activity_metadata_helper.py +407 -0
  253. aip_agents/utils/metadata/activity_narrative/__init__.py +35 -0
  254. aip_agents/utils/metadata/activity_narrative/builder.py +817 -0
  255. aip_agents/utils/metadata/activity_narrative/constants.py +51 -0
  256. aip_agents/utils/metadata/activity_narrative/context.py +49 -0
  257. aip_agents/utils/metadata/activity_narrative/formatters.py +230 -0
  258. aip_agents/utils/metadata/activity_narrative/utils.py +35 -0
  259. aip_agents/utils/metadata/schemas/__init__.py +16 -0
  260. aip_agents/utils/metadata/schemas/activity_schema.py +29 -0
  261. aip_agents/utils/metadata/schemas/thinking_schema.py +31 -0
  262. aip_agents/utils/metadata/thinking_metadata_helper.py +38 -0
  263. aip_agents/utils/metadata_helper.py +358 -0
  264. aip_agents/utils/name_preprocessor/__init__.py +17 -0
  265. aip_agents/utils/name_preprocessor/base_name_preprocessor.py +73 -0
  266. aip_agents/utils/name_preprocessor/google_name_preprocessor.py +100 -0
  267. aip_agents/utils/name_preprocessor/name_preprocessor.py +87 -0
  268. aip_agents/utils/name_preprocessor/openai_name_preprocessor.py +48 -0
  269. aip_agents/utils/pii/__init__.py +25 -0
  270. aip_agents/utils/pii/pii_handler.py +397 -0
  271. aip_agents/utils/pii/pii_helper.py +207 -0
  272. aip_agents/utils/pii/uuid_deanonymizer_mapping.py +195 -0
  273. aip_agents/utils/reference_helper.py +273 -0
  274. aip_agents/utils/sse_chunk_transformer.py +831 -0
  275. aip_agents/utils/step_limit_manager.py +265 -0
  276. aip_agents/utils/token_usage_helper.py +156 -0
  277. aip_agents_binary-0.5.20.dist-info/METADATA +681 -0
  278. aip_agents_binary-0.5.20.dist-info/RECORD +280 -0
  279. aip_agents_binary-0.5.20.dist-info/WHEEL +5 -0
  280. aip_agents_binary-0.5.20.dist-info/top_level.txt +1 -0
@@ -0,0 +1,120 @@
1
+ """Helpers for configuring browser-use LLM instances and environment flags.
2
+
3
+ Authors:
4
+ Raymond Christopher (raymond.christopher@gdplabs.id)
5
+ """
6
+
7
+ from __future__ import annotations
8
+
9
+ import os
10
+ from typing import Any
11
+
12
+ from browser_use.llm import ChatOpenAI
13
+ from browser_use.logging_config import setup_logging
14
+
15
+
16
+ def model_disallows_tunable_params(model: Any) -> bool:
17
+ """Return True if the provider forbids temperature/frequency overrides for the model.
18
+
19
+ Args:
20
+ model: The model name or identifier to check.
21
+
22
+ Returns:
23
+ bool: True if the model disallows tunable parameters, False otherwise.
24
+ """
25
+ model_name = str(model).lower()
26
+
27
+ if model_name.startswith("gpt-5"):
28
+ return True
29
+
30
+ short_name = model_name.split("-", 1)[0]
31
+ return short_name.startswith("o") and short_name[1:].isdigit()
32
+
33
+
34
+ def supports_temperature_override(model: Any) -> bool:
35
+ """Return True when the given model supports setting a custom temperature.
36
+
37
+ Args:
38
+ model: The model name or identifier to check.
39
+
40
+ Returns:
41
+ bool: True if the model supports temperature override, False otherwise.
42
+ """
43
+ return not model_disallows_tunable_params(model)
44
+
45
+
46
+ def supports_frequency_penalty(model: Any) -> bool:
47
+ """Return True when the given model supports custom frequency penalties.
48
+
49
+ Args:
50
+ model: The model name or identifier to check.
51
+
52
+ Returns:
53
+ bool: True if the model supports frequency penalty override, False otherwise.
54
+ """
55
+ return not model_disallows_tunable_params(model)
56
+
57
+
58
+ def build_browser_use_llm(
59
+ *,
60
+ model: Any,
61
+ reasoning_effort: Any,
62
+ temperature: float | None,
63
+ api_key: str,
64
+ base_url: str | None = None,
65
+ ) -> ChatOpenAI:
66
+ """Construct a ChatOpenAI instance with browser-use specific safeguards.
67
+
68
+ Args:
69
+ model: The model name or identifier to use.
70
+ reasoning_effort: The reasoning effort level for the model.
71
+ temperature: Optional temperature setting for the model. Can be None.
72
+ api_key: The API key for authentication.
73
+ base_url: The base URL for the model.
74
+
75
+ Returns:
76
+ ChatOpenAI: The configured ChatOpenAI instance.
77
+ """
78
+ llm_kwargs: dict[str, Any] = {
79
+ "model": model,
80
+ "reasoning_effort": reasoning_effort,
81
+ "api_key": api_key,
82
+ "base_url": base_url,
83
+ }
84
+
85
+ if temperature is not None and supports_temperature_override(model):
86
+ llm_kwargs["temperature"] = temperature
87
+ else:
88
+ llm_kwargs["temperature"] = None
89
+
90
+ if not supports_frequency_penalty(model):
91
+ llm_kwargs["frequency_penalty"] = None
92
+
93
+ return ChatOpenAI(**llm_kwargs)
94
+
95
+
96
+ def configure_browser_use_environment(enable_cloud_sync: bool, logging_level: str) -> None:
97
+ """Ensure Browser Use environment flags are aligned with tool configuration.
98
+
99
+ Args:
100
+ enable_cloud_sync: Whether to enable cloud synchronization for browser sessions.
101
+ logging_level: The desired logging level for browser use operations.
102
+ """
103
+ desired_sync = "true" if enable_cloud_sync else "false"
104
+ if os.environ.get("BROWSER_USE_CLOUD_SYNC") != desired_sync:
105
+ os.environ["BROWSER_USE_CLOUD_SYNC"] = desired_sync
106
+
107
+ desired_level = logging_level.lower()
108
+ current_level = os.environ.get("BROWSER_USE_LOGGING_LEVEL")
109
+ if not current_level or current_level.lower() != desired_level:
110
+ os.environ["BROWSER_USE_LOGGING_LEVEL"] = desired_level
111
+ setup_logging(log_level=desired_level, force_setup=True)
112
+
113
+
114
+ __all__ = [
115
+ "build_browser_use_llm",
116
+ "configure_browser_use_environment",
117
+ "model_disallows_tunable_params",
118
+ "supports_frequency_penalty",
119
+ "supports_temperature_override",
120
+ ]
@@ -0,0 +1,198 @@
1
+ """MinIO Storage Handler for Steel Session Recording.
2
+
3
+ This module provides MinIO cloud storage functionality for uploading and managing
4
+ video files from Steel session recordings. It handles file uploads, bucket management,
5
+ and presigned URL generation for secure file access.
6
+
7
+ The module supports:
8
+ - Automatic bucket creation and validation
9
+ - Secure file uploads with proper content types
10
+ - Presigned URL generation for temporary file access
11
+ - Directory prefix support for organized storage
12
+ - Comprehensive error handling and logging
13
+
14
+ Authors:
15
+ Reinhart Linanda (reinhart.linanda@gdplabs.id)
16
+ """
17
+
18
+ import os
19
+
20
+ from dotenv import load_dotenv
21
+ from minio import Minio
22
+ from minio.error import S3Error
23
+
24
+ load_dotenv()
25
+
26
+ OBJECT_NAME_PREFIX = "steel-recordings/"
27
+
28
+
29
+ class MinIOStorage:
30
+ """Handles MinIO cloud storage operations for video files.
31
+
32
+ This class provides a complete interface for MinIO operations including:
33
+ - Connection management with authentication
34
+ - Bucket existence validation and creation
35
+ - File upload with proper metadata
36
+ - Presigned URL generation for secure access
37
+
38
+ Attributes:
39
+ endpoint: MinIO server endpoint URL.
40
+ access_key: MinIO access key for authentication.
41
+ secret_key: MinIO secret key for authentication.
42
+ secure: Whether to use HTTPS/TLS for connections.
43
+ bucket_name: Target bucket for file storage.
44
+ directory_prefix: Optional directory prefix for organized storage.
45
+ client: MinIO client instance for API operations.
46
+ """
47
+
48
+ def __init__(self):
49
+ """Initialize MinIO storage with configuration from environment variables.
50
+
51
+ Reads configuration from the following environment variables:
52
+ - OBJECT_STORAGE_URL: MinIO server endpoint
53
+ - OBJECT_STORAGE_USER: MinIO access key
54
+ - OBJECT_STORAGE_PASSWORD: MinIO secret key
55
+ - OBJECT_STORAGE_SECURE: Whether to use HTTPS (default: False)
56
+ - OBJECT_STORAGE_BUCKET: Target bucket name
57
+ - OBJECT_STORAGE_DIRECTORY_PREFIX: Optional directory prefix
58
+
59
+ Raises:
60
+ ValueError: If required environment variables are missing.
61
+ Exception: If bucket creation or validation fails.
62
+
63
+ Note:
64
+ The method automatically ensures the target bucket exists,
65
+ creating it if necessary.
66
+ """
67
+ self.endpoint = os.getenv("OBJECT_STORAGE_URL")
68
+ self.access_key = os.getenv("OBJECT_STORAGE_USER")
69
+ self.secret_key = os.getenv("OBJECT_STORAGE_PASSWORD")
70
+ self.secure = os.getenv("OBJECT_STORAGE_SECURE", "False").lower() == "true"
71
+ self.bucket_name = os.getenv("OBJECT_STORAGE_BUCKET")
72
+ self.directory_prefix = os.getenv("OBJECT_STORAGE_DIRECTORY_PREFIX")
73
+
74
+ if not all([self.endpoint, self.access_key, self.secret_key, self.bucket_name]):
75
+ raise ValueError(
76
+ "MinIO configuration incomplete. Set OBJECT_STORAGE_URL, OBJECT_STORAGE_USER, OBJECT_STORAGE_PASSWORD,"
77
+ "and OBJECT_STORAGE_BUCKET."
78
+ )
79
+
80
+ self.client = Minio(
81
+ self.endpoint,
82
+ access_key=self.access_key,
83
+ secret_key=self.secret_key,
84
+ secure=self.secure,
85
+ )
86
+
87
+ # Ensure bucket exists
88
+ self._ensure_bucket_exists()
89
+
90
+ def _ensure_bucket_exists(self):
91
+ """Ensure the configured bucket exists, create if it doesn't.
92
+
93
+ This method checks if the target bucket exists and creates it if necessary.
94
+ It's called during initialization to ensure the storage is ready for use.
95
+
96
+ Raises:
97
+ Exception: If bucket creation or validation fails.
98
+
99
+ Note:
100
+ Bucket creation failures are logged as warnings but don't prevent
101
+ the class from being usable (existing buckets can still be accessed).
102
+ """
103
+ try:
104
+ if not self.client.bucket_exists(self.bucket_name):
105
+ self.client.make_bucket(self.bucket_name)
106
+ except S3Error as e:
107
+ raise Exception(f"Warning: Could not ensure bucket exists: {e}") from e
108
+
109
+ def get_object_name(self, object_name: str) -> str:
110
+ """Get the object name with the directory prefix.
111
+
112
+ This method constructs the full object path by combining the directory
113
+ prefix with the base object name. It ensures consistent path structure
114
+ for all stored files.
115
+
116
+ Args:
117
+ object_name: Name of the object in MinIO.
118
+
119
+ Returns:
120
+ str: Object name with the directory prefix and steel-recordings subdirectory.
121
+ Format: {prefix}/steel-recordings/{object_name} or steel-recordings/{object_name}
122
+
123
+ Note:
124
+ The method automatically adds a "steel-recordings" subdirectory to
125
+ organize video files separately from other content.
126
+ """
127
+ object_name = f"{OBJECT_NAME_PREFIX}{object_name}"
128
+ if self.directory_prefix:
129
+ return f"{self.directory_prefix}/{object_name}"
130
+ return object_name
131
+
132
+ def upload_file(self, file_path: str, object_name: str) -> None:
133
+ """Upload a file to MinIO bucket.
134
+
135
+ This method handles the complete file upload process including:
136
+ - File existence validation
137
+ - Proper content type setting for video files
138
+ - Error handling and logging
139
+ - Structured object naming with prefixes
140
+
141
+ Args:
142
+ file_path: Local path to the file to upload.
143
+ object_name: Name to use for the object in MinIO.
144
+
145
+ Raises:
146
+ Exception: If the file doesn't exist or upload fails.
147
+ Specific S3Error details are included in the exception message.
148
+
149
+ Note:
150
+ The method automatically sets the content type to "video/webm"
151
+ for proper video file handling in browsers and applications.
152
+ """
153
+ try:
154
+ if not os.path.exists(file_path):
155
+ raise Exception(f"File not found: {file_path}")
156
+
157
+ # Upload the file
158
+ self.client.fput_object(
159
+ self.bucket_name,
160
+ self.get_object_name(object_name),
161
+ file_path,
162
+ content_type="video/webm",
163
+ )
164
+
165
+ except S3Error as e:
166
+ raise Exception(f"Error uploading to MinIO: {e}") from e
167
+ except Exception as e:
168
+ raise Exception(f"Unexpected error during upload: {e}") from e
169
+
170
+ def get_file_url(self, object_name: str) -> str:
171
+ """Generate a presigned URL for accessing the uploaded file.
172
+
173
+ This method creates a temporary, secure URL that allows access to the
174
+ uploaded file without requiring MinIO credentials. The URL is valid
175
+ for a limited time and provides secure, direct access to the file.
176
+
177
+ Args:
178
+ object_name: Name of the object in MinIO.
179
+
180
+ Returns:
181
+ str: Presigned URL for secure file access.
182
+ The URL includes authentication tokens and is valid for a limited time.
183
+
184
+ Raises:
185
+ Exception: If presigned URL generation fails.
186
+ S3Error details are included in the exception message.
187
+
188
+ Note:
189
+ Presigned URLs are useful for sharing files temporarily without
190
+ exposing MinIO credentials or requiring users to have storage access.
191
+ """
192
+ try:
193
+ return self.client.presigned_get_object(
194
+ self.bucket_name,
195
+ self.get_object_name(object_name),
196
+ )
197
+ except S3Error as e:
198
+ raise Exception(f"Error generating presigned URL: {e}") from e
@@ -0,0 +1,119 @@
1
+ """Configuration schemas for Browser Use tool.
2
+
3
+ Authors:
4
+ Reinhart Linanda (reinhart.linanda@gdplabs.id)
5
+ """
6
+
7
+ import os
8
+ from typing import Literal
9
+
10
+ from openai.types.shared.chat_model import ChatModel
11
+ from pydantic import BaseModel, Field
12
+
13
+
14
+ class BrowserUseToolInput(BaseModel):
15
+ """Input schema for Browser Use tool."""
16
+
17
+ task: str = Field(..., description="Task prompt for the AI agent to execute in the browser")
18
+
19
+
20
+ class BrowserUseToolConfig(BaseModel):
21
+ """Tool configuration schema for Browser Use tool."""
22
+
23
+ steel_api_key: str = Field(
24
+ default_factory=lambda: os.getenv("STEEL_API_KEY"),
25
+ description="Steel API key for Steel access",
26
+ )
27
+ steel_base_url: str = Field(
28
+ default="https://api.steel.dev",
29
+ description="Steel API base URL",
30
+ )
31
+ steel_ws_url: str = Field(
32
+ default="wss://connect.steel.dev",
33
+ description="Steel WebSocket URL for browser connection",
34
+ )
35
+ steel_timeout_in_ms: int = Field(
36
+ default=600_000,
37
+ description="Timeout for Steel operations in milliseconds",
38
+ )
39
+
40
+ browser_use_llm_openai_api_key: str = Field(
41
+ default_factory=lambda: os.getenv("OPENAI_API_KEY"),
42
+ description="OpenAI API key for browser-use access",
43
+ )
44
+ browser_use_llm_openai_model: ChatModel | str = Field(
45
+ default="o3",
46
+ description="OpenAI model to use for browser-use agent",
47
+ )
48
+ browser_use_llm_openai_temperature: float | None = Field(
49
+ default=None,
50
+ description="Temperature setting for OpenAI model (None uses provider default)",
51
+ )
52
+ browser_use_llm_openai_reasoning_effort: Literal["minimal", "low", "medium", "high"] = Field(
53
+ default="low",
54
+ description="Reasoning effort for OpenAI model",
55
+ )
56
+ browser_use_llm_openai_base_url: str | None = Field(
57
+ default=None,
58
+ description="Base URL for OpenAI model (None uses provider default)",
59
+ )
60
+
61
+ browser_use_page_extraction_llm_openai_api_key: str = Field(
62
+ default_factory=lambda: os.getenv("OPENAI_API_KEY"),
63
+ description="OpenAI API key for page extraction access",
64
+ )
65
+ browser_use_page_extraction_llm_openai_model: ChatModel | str = Field(
66
+ default="gpt-5-mini",
67
+ description="OpenAI model to use for page extraction",
68
+ )
69
+ browser_use_page_extraction_llm_openai_temperature: float | None = Field(
70
+ default=None,
71
+ description="Temperature setting for OpenAI model (None uses provider default)",
72
+ )
73
+ browser_use_page_extraction_llm_openai_reasoning_effort: Literal["minimal", "low", "medium", "high"] = Field(
74
+ default="minimal",
75
+ description="Reasoning effort for OpenAI model",
76
+ )
77
+ browser_use_page_extraction_llm_openai_base_url: str | None = Field(
78
+ default=None,
79
+ description="Base URL for OpenAI model (None uses provider default)",
80
+ )
81
+
82
+ browser_use_extend_system_message: str | None = Field(
83
+ default=None,
84
+ description="Extend system message for browser-use agent",
85
+ )
86
+
87
+ browser_use_vision_detail_level: Literal["auto", "low", "high"] = Field(
88
+ default="auto",
89
+ description="Detail level for vision",
90
+ )
91
+
92
+ browser_use_enable_cloud_sync: bool = Field(
93
+ default=False,
94
+ description="Enable Browser Use cloud sync and telemetry events",
95
+ )
96
+
97
+ browser_use_logging_level: Literal["debug", "info", "warning", "error", "result"] = Field(
98
+ default="info",
99
+ description="Logging verbosity for browser-use internals",
100
+ )
101
+
102
+ browser_use_llm_timeout_in_s: int = Field(
103
+ default=60,
104
+ description="Timeout for LLM in seconds",
105
+ )
106
+ browser_use_step_timeout_in_s: int = Field(
107
+ default=180,
108
+ description="Timeout for step in seconds",
109
+ )
110
+ browser_use_max_session_retries: int = Field(
111
+ default=2,
112
+ ge=0,
113
+ description="How many times to recreate the Steel session after recoverable disconnects.",
114
+ )
115
+ browser_use_session_retry_delay_in_s: float = Field(
116
+ default=3.0,
117
+ ge=0.0,
118
+ description="Delay (in seconds) before attempting to recover from a lost Steel session.",
119
+ )
@@ -0,0 +1,76 @@
1
+ """Session management for browser-use framework.
2
+
3
+ Authors:
4
+ Reinhart Linanda (reinhart.linanda@gdplabs.id)
5
+
6
+ References:
7
+ https://github.com/browser-use/browser-use/blob/0.5.9/browser_use/browser/session.py
8
+ """
9
+
10
+ import asyncio
11
+
12
+ from browser_use.agent.views import DOMElementNode
13
+ from browser_use.browser.session import BrowserSession as BrowserUseSession
14
+ from browser_use.browser.session import require_healthy_browser
15
+ from browser_use.browser.views import BrowserError
16
+ from browser_use.observability import observe_debug
17
+ from browser_use.utils import _log_pretty_url, time_execution_async
18
+
19
+
20
+ class BrowserSession(BrowserUseSession):
21
+ """Represents an active browser session with a running browser process somewhere."""
22
+
23
+ @require_healthy_browser(usable_page=True, reopen_page=True)
24
+ @time_execution_async("--input_text_element_node")
25
+ @observe_debug(ignore_input=True, name="input_text_element_node")
26
+ async def _input_text_element_node(self, element_node: DOMElementNode, text: str):
27
+ """Input text into an element with proper error handling and state management.
28
+
29
+ Args:
30
+ element_node (DOMElementNode): The element node to input text into.
31
+ text (str): The text to input into the element.
32
+ """
33
+ try:
34
+ element_handle = await self.get_locate_element(element_node)
35
+
36
+ if element_handle is None:
37
+ raise BrowserError(f"Element: {repr(element_node)} not found")
38
+
39
+ # Ensure element is ready for input
40
+ try:
41
+ await element_handle.wait_for_element_state("stable", timeout=1_000)
42
+ is_visible = await self._is_visible(element_handle)
43
+ if is_visible:
44
+ await element_handle.scroll_into_view_if_needed(timeout=1_000)
45
+ except Exception as state_error:
46
+ self.logger.debug(
47
+ "Skipping pre-input visibility preparation for %s due to %s",
48
+ repr(element_node),
49
+ state_error,
50
+ )
51
+
52
+ # let's first try to click and type
53
+ try:
54
+ await element_handle.evaluate('el => {el.textContent = ""; el.value = "";}')
55
+ await element_handle.click()
56
+ await asyncio.sleep(0.1) # Increased sleep time
57
+ page = await self.get_current_page()
58
+ await page.keyboard.insert_text(text)
59
+ return
60
+ except Exception as e:
61
+ self.logger.debug(f"Input text with click and type failed, trying element handle method: {e}")
62
+ # fall through to BrowserUseSession fallback below
63
+
64
+ except Exception as e:
65
+ # Get current page URL safely for error message
66
+ try:
67
+ page = await self.get_current_page()
68
+ page_url = _log_pretty_url(page.url)
69
+ except Exception:
70
+ page_url = "unknown page"
71
+
72
+ self.logger.debug(
73
+ f"❌ Failed to input text into element: {repr(element_node)} "
74
+ f"on page {page_url}: {type(e).__name__}: {e}"
75
+ )
76
+ raise BrowserError(f"Failed to input text into index {element_node.highlight_index}") from e
@@ -0,0 +1,132 @@
1
+ """Centralized Steel/browser session error policies used by BrowserUseTool.
2
+
3
+ Authors:
4
+ Raymond Christopher (raymond.christopher@gdplabs.id)
5
+ """
6
+
7
+ from collections.abc import Iterable
8
+ from dataclasses import dataclass
9
+
10
+
11
+ @dataclass(frozen=True)
12
+ class SessionErrorCategory:
13
+ """Represents a fatal Steel/browser disconnect classification."""
14
+
15
+ name: str
16
+ markers: tuple[str, ...]
17
+ fatal: bool
18
+ retryable: bool
19
+
20
+
21
+ _FATAL_SESSION_ERROR_CATEGORIES: tuple[SessionErrorCategory, ...] = (
22
+ SessionErrorCategory(
23
+ name="browser_closed",
24
+ markers=(
25
+ "browser has been closed",
26
+ "target page, context or browser has been closed",
27
+ ),
28
+ fatal=True,
29
+ retryable=True,
30
+ ),
31
+ SessionErrorCategory(
32
+ name="websocket_disconnect",
33
+ markers=(
34
+ "code=1006",
35
+ "websocket was closed before the connection was established",
36
+ "websocket error",
37
+ ),
38
+ fatal=True,
39
+ retryable=True,
40
+ ),
41
+ )
42
+
43
+ _WARNING_MARKERS: dict[str, str] = {
44
+ "502 bad gateway": "page_load_warning",
45
+ "page link: about:blank": "blank_page_warning",
46
+ "no webpage content": "blank_page_warning",
47
+ "scroll_into_view_if_needed": "element_interaction_warning",
48
+ }
49
+
50
+ _FATAL_LOOKUP: dict[str, SessionErrorCategory] = {
51
+ marker.lower(): category for category in _FATAL_SESSION_ERROR_CATEGORIES for marker in category.markers
52
+ }
53
+
54
+
55
+ def categorize_fatal_message(message: str) -> SessionErrorCategory | None:
56
+ """Return the fatal session category associated with the given message, if any.
57
+
58
+ Args:
59
+ message: The error message to categorize.
60
+
61
+ Returns:
62
+ The SessionErrorCategory if the message matches a fatal error pattern,
63
+ None otherwise.
64
+ """
65
+ if not message:
66
+ return None
67
+
68
+ lowered = message.lower()
69
+ for marker, category in _FATAL_LOOKUP.items():
70
+ if marker in lowered:
71
+ return category
72
+ return None
73
+
74
+
75
+ def categorize_warning_message(message: str) -> str | None:
76
+ """Return the name of a known non-fatal warning when present in the message.
77
+
78
+ Args:
79
+ message: The error message to check for warning patterns.
80
+
81
+ Returns:
82
+ The warning name if the message matches a known warning pattern,
83
+ None otherwise.
84
+ """
85
+ if not message:
86
+ return None
87
+
88
+ lowered = message.lower()
89
+ for marker, warning_name in _WARNING_MARKERS.items():
90
+ if marker in lowered:
91
+ return warning_name
92
+ return None
93
+
94
+
95
+ def find_fatal_message(messages: Iterable[str]) -> tuple[str, SessionErrorCategory] | None:
96
+ """Return the first fatal message detected in the iterable.
97
+
98
+ Args:
99
+ messages: An iterable of error messages to search through.
100
+
101
+ Returns:
102
+ A tuple of (message, category) for the first fatal message found,
103
+ None if no fatal messages are detected.
104
+ """
105
+ for message in messages:
106
+ category = categorize_fatal_message(message)
107
+ if category:
108
+ return message, category
109
+ return None
110
+
111
+
112
+ def is_recoverable_message(message: str) -> bool:
113
+ """Return True when the message maps to a retryable session disconnect.
114
+
115
+ Args:
116
+ message: The error message to check for recoverability.
117
+
118
+ Returns:
119
+ True if the message corresponds to a retryable fatal error,
120
+ False otherwise.
121
+ """
122
+ category = categorize_fatal_message(message)
123
+ return bool(category and category.retryable)
124
+
125
+
126
+ __all__ = [
127
+ "SessionErrorCategory",
128
+ "categorize_fatal_message",
129
+ "categorize_warning_message",
130
+ "find_fatal_message",
131
+ "is_recoverable_message",
132
+ ]