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.
- aip_agents/__init__.py +65 -0
- aip_agents/a2a/__init__.py +19 -0
- aip_agents/a2a/server/__init__.py +10 -0
- aip_agents/a2a/server/base_executor.py +1086 -0
- aip_agents/a2a/server/google_adk_executor.py +198 -0
- aip_agents/a2a/server/langflow_executor.py +180 -0
- aip_agents/a2a/server/langgraph_executor.py +270 -0
- aip_agents/a2a/types.py +232 -0
- aip_agents/agent/__init__.py +27 -0
- aip_agents/agent/base_agent.py +970 -0
- aip_agents/agent/base_langgraph_agent.py +2942 -0
- aip_agents/agent/google_adk_agent.py +926 -0
- aip_agents/agent/google_adk_constants.py +6 -0
- aip_agents/agent/hitl/__init__.py +24 -0
- aip_agents/agent/hitl/config.py +28 -0
- aip_agents/agent/hitl/langgraph_hitl_mixin.py +515 -0
- aip_agents/agent/hitl/manager.py +532 -0
- aip_agents/agent/hitl/models.py +18 -0
- aip_agents/agent/hitl/prompt/__init__.py +9 -0
- aip_agents/agent/hitl/prompt/base.py +42 -0
- aip_agents/agent/hitl/prompt/deferred.py +73 -0
- aip_agents/agent/hitl/registry.py +149 -0
- aip_agents/agent/interface.py +138 -0
- aip_agents/agent/interfaces.py +65 -0
- aip_agents/agent/langflow_agent.py +464 -0
- aip_agents/agent/langgraph_memory_enhancer_agent.py +433 -0
- aip_agents/agent/langgraph_react_agent.py +2514 -0
- aip_agents/agent/system_instruction_context.py +34 -0
- aip_agents/clients/__init__.py +10 -0
- aip_agents/clients/langflow/__init__.py +10 -0
- aip_agents/clients/langflow/client.py +477 -0
- aip_agents/clients/langflow/types.py +18 -0
- aip_agents/constants.py +23 -0
- aip_agents/credentials/manager.py +132 -0
- aip_agents/examples/__init__.py +5 -0
- aip_agents/examples/compare_streaming_client.py +783 -0
- aip_agents/examples/compare_streaming_server.py +142 -0
- aip_agents/examples/demo_memory_recall.py +401 -0
- aip_agents/examples/hello_world_a2a_google_adk_client.py +49 -0
- aip_agents/examples/hello_world_a2a_google_adk_client_agent.py +48 -0
- aip_agents/examples/hello_world_a2a_google_adk_client_streaming.py +60 -0
- aip_agents/examples/hello_world_a2a_google_adk_server.py +79 -0
- aip_agents/examples/hello_world_a2a_langchain_client.py +39 -0
- aip_agents/examples/hello_world_a2a_langchain_client_agent.py +39 -0
- aip_agents/examples/hello_world_a2a_langchain_client_lm_invoker.py +37 -0
- aip_agents/examples/hello_world_a2a_langchain_client_streaming.py +41 -0
- aip_agents/examples/hello_world_a2a_langchain_reference_client_streaming.py +60 -0
- aip_agents/examples/hello_world_a2a_langchain_reference_server.py +105 -0
- aip_agents/examples/hello_world_a2a_langchain_server.py +79 -0
- aip_agents/examples/hello_world_a2a_langchain_server_lm_invoker.py +78 -0
- aip_agents/examples/hello_world_a2a_langflow_client.py +83 -0
- aip_agents/examples/hello_world_a2a_langflow_server.py +82 -0
- aip_agents/examples/hello_world_a2a_langgraph_artifact_client.py +73 -0
- aip_agents/examples/hello_world_a2a_langgraph_artifact_client_streaming.py +76 -0
- aip_agents/examples/hello_world_a2a_langgraph_artifact_server.py +92 -0
- aip_agents/examples/hello_world_a2a_langgraph_client.py +54 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_agent.py +54 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_agent_lm_invoker.py +32 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_streaming.py +50 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_streaming_lm_invoker.py +44 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_streaming_tool_streaming.py +92 -0
- aip_agents/examples/hello_world_a2a_langgraph_server.py +84 -0
- aip_agents/examples/hello_world_a2a_langgraph_server_lm_invoker.py +79 -0
- aip_agents/examples/hello_world_a2a_langgraph_server_tool_streaming.py +132 -0
- aip_agents/examples/hello_world_a2a_mcp_langgraph.py +196 -0
- aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_client.py +244 -0
- aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_server.py +251 -0
- aip_agents/examples/hello_world_a2a_with_metadata_langchain_client.py +57 -0
- aip_agents/examples/hello_world_a2a_with_metadata_langchain_server_lm_invoker.py +80 -0
- aip_agents/examples/hello_world_google_adk.py +41 -0
- aip_agents/examples/hello_world_google_adk_mcp_http.py +34 -0
- aip_agents/examples/hello_world_google_adk_mcp_http_stream.py +40 -0
- aip_agents/examples/hello_world_google_adk_mcp_sse.py +44 -0
- aip_agents/examples/hello_world_google_adk_mcp_sse_stream.py +48 -0
- aip_agents/examples/hello_world_google_adk_mcp_stdio.py +44 -0
- aip_agents/examples/hello_world_google_adk_mcp_stdio_stream.py +48 -0
- aip_agents/examples/hello_world_google_adk_stream.py +44 -0
- aip_agents/examples/hello_world_langchain.py +28 -0
- aip_agents/examples/hello_world_langchain_lm_invoker.py +15 -0
- aip_agents/examples/hello_world_langchain_mcp_http.py +34 -0
- aip_agents/examples/hello_world_langchain_mcp_http_interactive.py +130 -0
- aip_agents/examples/hello_world_langchain_mcp_http_stream.py +42 -0
- aip_agents/examples/hello_world_langchain_mcp_multi_server.py +155 -0
- aip_agents/examples/hello_world_langchain_mcp_sse.py +34 -0
- aip_agents/examples/hello_world_langchain_mcp_sse_stream.py +40 -0
- aip_agents/examples/hello_world_langchain_mcp_stdio.py +30 -0
- aip_agents/examples/hello_world_langchain_mcp_stdio_stream.py +41 -0
- aip_agents/examples/hello_world_langchain_stream.py +36 -0
- aip_agents/examples/hello_world_langchain_stream_lm_invoker.py +39 -0
- aip_agents/examples/hello_world_langflow_agent.py +163 -0
- aip_agents/examples/hello_world_langgraph.py +39 -0
- aip_agents/examples/hello_world_langgraph_bosa_twitter.py +41 -0
- aip_agents/examples/hello_world_langgraph_mcp_http.py +31 -0
- aip_agents/examples/hello_world_langgraph_mcp_http_stream.py +34 -0
- aip_agents/examples/hello_world_langgraph_mcp_sse.py +35 -0
- aip_agents/examples/hello_world_langgraph_mcp_sse_stream.py +50 -0
- aip_agents/examples/hello_world_langgraph_mcp_stdio.py +35 -0
- aip_agents/examples/hello_world_langgraph_mcp_stdio_stream.py +50 -0
- aip_agents/examples/hello_world_langgraph_stream.py +43 -0
- aip_agents/examples/hello_world_langgraph_stream_lm_invoker.py +37 -0
- aip_agents/examples/hello_world_model_switch_cli.py +210 -0
- aip_agents/examples/hello_world_multi_agent_adk.py +75 -0
- aip_agents/examples/hello_world_multi_agent_langchain.py +54 -0
- aip_agents/examples/hello_world_multi_agent_langgraph.py +66 -0
- aip_agents/examples/hello_world_multi_agent_langgraph_lm_invoker.py +69 -0
- aip_agents/examples/hello_world_pii_logger.py +21 -0
- aip_agents/examples/hello_world_sentry.py +133 -0
- aip_agents/examples/hello_world_step_limits.py +273 -0
- aip_agents/examples/hello_world_stock_a2a_server.py +103 -0
- aip_agents/examples/hello_world_tool_output_client.py +46 -0
- aip_agents/examples/hello_world_tool_output_server.py +114 -0
- aip_agents/examples/hitl_demo.py +724 -0
- aip_agents/examples/mcp_configs/configs.py +63 -0
- aip_agents/examples/mcp_servers/common.py +76 -0
- aip_agents/examples/mcp_servers/mcp_name.py +29 -0
- aip_agents/examples/mcp_servers/mcp_server_http.py +19 -0
- aip_agents/examples/mcp_servers/mcp_server_sse.py +19 -0
- aip_agents/examples/mcp_servers/mcp_server_stdio.py +19 -0
- aip_agents/examples/mcp_servers/mcp_time.py +10 -0
- aip_agents/examples/pii_demo_langgraph_client.py +69 -0
- aip_agents/examples/pii_demo_langgraph_server.py +126 -0
- aip_agents/examples/pii_demo_multi_agent_client.py +80 -0
- aip_agents/examples/pii_demo_multi_agent_server.py +247 -0
- aip_agents/examples/todolist_planning_a2a_langchain_client.py +70 -0
- aip_agents/examples/todolist_planning_a2a_langgraph_server.py +88 -0
- aip_agents/examples/tools/__init__.py +27 -0
- aip_agents/examples/tools/adk_arithmetic_tools.py +36 -0
- aip_agents/examples/tools/adk_weather_tool.py +60 -0
- aip_agents/examples/tools/data_generator_tool.py +103 -0
- aip_agents/examples/tools/data_visualization_tool.py +312 -0
- aip_agents/examples/tools/image_artifact_tool.py +136 -0
- aip_agents/examples/tools/langchain_arithmetic_tools.py +26 -0
- aip_agents/examples/tools/langchain_currency_exchange_tool.py +88 -0
- aip_agents/examples/tools/langchain_graph_artifact_tool.py +172 -0
- aip_agents/examples/tools/langchain_weather_tool.py +48 -0
- aip_agents/examples/tools/langgraph_streaming_tool.py +130 -0
- aip_agents/examples/tools/mock_retrieval_tool.py +56 -0
- aip_agents/examples/tools/pii_demo_tools.py +189 -0
- aip_agents/examples/tools/random_chart_tool.py +142 -0
- aip_agents/examples/tools/serper_tool.py +202 -0
- aip_agents/examples/tools/stock_tools.py +82 -0
- aip_agents/examples/tools/table_generator_tool.py +167 -0
- aip_agents/examples/tools/time_tool.py +82 -0
- aip_agents/examples/tools/weather_forecast_tool.py +38 -0
- aip_agents/executor/agent_executor.py +473 -0
- aip_agents/executor/base.py +48 -0
- aip_agents/mcp/__init__.py +1 -0
- aip_agents/mcp/client/__init__.py +14 -0
- aip_agents/mcp/client/base_mcp_client.py +369 -0
- aip_agents/mcp/client/connection_manager.py +193 -0
- aip_agents/mcp/client/google_adk/__init__.py +11 -0
- aip_agents/mcp/client/google_adk/client.py +381 -0
- aip_agents/mcp/client/langchain/__init__.py +11 -0
- aip_agents/mcp/client/langchain/client.py +265 -0
- aip_agents/mcp/client/persistent_session.py +359 -0
- aip_agents/mcp/client/session_pool.py +351 -0
- aip_agents/mcp/client/transports.py +215 -0
- aip_agents/mcp/utils/__init__.py +7 -0
- aip_agents/mcp/utils/config_validator.py +139 -0
- aip_agents/memory/__init__.py +14 -0
- aip_agents/memory/adapters/__init__.py +10 -0
- aip_agents/memory/adapters/base_adapter.py +717 -0
- aip_agents/memory/adapters/mem0.py +84 -0
- aip_agents/memory/base.py +84 -0
- aip_agents/memory/constants.py +49 -0
- aip_agents/memory/factory.py +86 -0
- aip_agents/memory/guidance.py +20 -0
- aip_agents/memory/simple_memory.py +47 -0
- aip_agents/middleware/__init__.py +17 -0
- aip_agents/middleware/base.py +88 -0
- aip_agents/middleware/manager.py +128 -0
- aip_agents/middleware/todolist.py +274 -0
- aip_agents/schema/__init__.py +69 -0
- aip_agents/schema/a2a.py +56 -0
- aip_agents/schema/agent.py +111 -0
- aip_agents/schema/hitl.py +157 -0
- aip_agents/schema/langgraph.py +37 -0
- aip_agents/schema/model_id.py +97 -0
- aip_agents/schema/step_limit.py +108 -0
- aip_agents/schema/storage.py +40 -0
- aip_agents/sentry/__init__.py +11 -0
- aip_agents/sentry/sentry.py +151 -0
- aip_agents/storage/__init__.py +41 -0
- aip_agents/storage/base.py +85 -0
- aip_agents/storage/clients/__init__.py +12 -0
- aip_agents/storage/clients/minio_client.py +318 -0
- aip_agents/storage/config.py +62 -0
- aip_agents/storage/providers/__init__.py +15 -0
- aip_agents/storage/providers/base.py +106 -0
- aip_agents/storage/providers/memory.py +114 -0
- aip_agents/storage/providers/object_storage.py +214 -0
- aip_agents/tools/__init__.py +33 -0
- aip_agents/tools/bosa_tools.py +105 -0
- aip_agents/tools/browser_use/__init__.py +82 -0
- aip_agents/tools/browser_use/action_parser.py +103 -0
- aip_agents/tools/browser_use/browser_use_tool.py +1112 -0
- aip_agents/tools/browser_use/llm_config.py +120 -0
- aip_agents/tools/browser_use/minio_storage.py +198 -0
- aip_agents/tools/browser_use/schemas.py +119 -0
- aip_agents/tools/browser_use/session.py +76 -0
- aip_agents/tools/browser_use/session_errors.py +132 -0
- aip_agents/tools/browser_use/steel_session_recording.py +317 -0
- aip_agents/tools/browser_use/streaming.py +813 -0
- aip_agents/tools/browser_use/structured_data_parser.py +257 -0
- aip_agents/tools/browser_use/structured_data_recovery.py +204 -0
- aip_agents/tools/browser_use/types.py +78 -0
- aip_agents/tools/code_sandbox/__init__.py +26 -0
- aip_agents/tools/code_sandbox/constant.py +13 -0
- aip_agents/tools/code_sandbox/e2b_cloud_sandbox_extended.py +257 -0
- aip_agents/tools/code_sandbox/e2b_sandbox_tool.py +411 -0
- aip_agents/tools/constants.py +165 -0
- aip_agents/tools/document_loader/__init__.py +44 -0
- aip_agents/tools/document_loader/base_reader.py +302 -0
- aip_agents/tools/document_loader/docx_reader_tool.py +68 -0
- aip_agents/tools/document_loader/excel_reader_tool.py +171 -0
- aip_agents/tools/document_loader/pdf_reader_tool.py +79 -0
- aip_agents/tools/document_loader/pdf_splitter.py +169 -0
- aip_agents/tools/gl_connector/__init__.py +5 -0
- aip_agents/tools/gl_connector/tool.py +351 -0
- aip_agents/tools/memory_search/__init__.py +22 -0
- aip_agents/tools/memory_search/base.py +200 -0
- aip_agents/tools/memory_search/mem0.py +258 -0
- aip_agents/tools/memory_search/schema.py +48 -0
- aip_agents/tools/memory_search_tool.py +26 -0
- aip_agents/tools/time_tool.py +117 -0
- aip_agents/tools/tool_config_injector.py +300 -0
- aip_agents/tools/web_search/__init__.py +15 -0
- aip_agents/tools/web_search/serper_tool.py +187 -0
- aip_agents/types/__init__.py +70 -0
- aip_agents/types/a2a_events.py +13 -0
- aip_agents/utils/__init__.py +79 -0
- aip_agents/utils/a2a_connector.py +1757 -0
- aip_agents/utils/artifact_helpers.py +502 -0
- aip_agents/utils/constants.py +22 -0
- aip_agents/utils/datetime/__init__.py +34 -0
- aip_agents/utils/datetime/normalization.py +231 -0
- aip_agents/utils/datetime/timezone.py +206 -0
- aip_agents/utils/env_loader.py +27 -0
- aip_agents/utils/event_handler_registry.py +58 -0
- aip_agents/utils/file_prompt_utils.py +176 -0
- aip_agents/utils/final_response_builder.py +211 -0
- aip_agents/utils/formatter_llm_client.py +231 -0
- aip_agents/utils/langgraph/__init__.py +19 -0
- aip_agents/utils/langgraph/converter.py +128 -0
- aip_agents/utils/langgraph/tool_managers/__init__.py +15 -0
- aip_agents/utils/langgraph/tool_managers/a2a_tool_manager.py +99 -0
- aip_agents/utils/langgraph/tool_managers/base_tool_manager.py +66 -0
- aip_agents/utils/langgraph/tool_managers/delegation_tool_manager.py +1071 -0
- aip_agents/utils/langgraph/tool_output_management.py +967 -0
- aip_agents/utils/logger.py +195 -0
- aip_agents/utils/metadata/__init__.py +27 -0
- aip_agents/utils/metadata/activity_metadata_helper.py +407 -0
- aip_agents/utils/metadata/activity_narrative/__init__.py +35 -0
- aip_agents/utils/metadata/activity_narrative/builder.py +817 -0
- aip_agents/utils/metadata/activity_narrative/constants.py +51 -0
- aip_agents/utils/metadata/activity_narrative/context.py +49 -0
- aip_agents/utils/metadata/activity_narrative/formatters.py +230 -0
- aip_agents/utils/metadata/activity_narrative/utils.py +35 -0
- aip_agents/utils/metadata/schemas/__init__.py +16 -0
- aip_agents/utils/metadata/schemas/activity_schema.py +29 -0
- aip_agents/utils/metadata/schemas/thinking_schema.py +31 -0
- aip_agents/utils/metadata/thinking_metadata_helper.py +38 -0
- aip_agents/utils/metadata_helper.py +358 -0
- aip_agents/utils/name_preprocessor/__init__.py +17 -0
- aip_agents/utils/name_preprocessor/base_name_preprocessor.py +73 -0
- aip_agents/utils/name_preprocessor/google_name_preprocessor.py +100 -0
- aip_agents/utils/name_preprocessor/name_preprocessor.py +87 -0
- aip_agents/utils/name_preprocessor/openai_name_preprocessor.py +48 -0
- aip_agents/utils/pii/__init__.py +25 -0
- aip_agents/utils/pii/pii_handler.py +397 -0
- aip_agents/utils/pii/pii_helper.py +207 -0
- aip_agents/utils/pii/uuid_deanonymizer_mapping.py +195 -0
- aip_agents/utils/reference_helper.py +273 -0
- aip_agents/utils/sse_chunk_transformer.py +831 -0
- aip_agents/utils/step_limit_manager.py +265 -0
- aip_agents/utils/token_usage_helper.py +156 -0
- aip_agents_binary-0.5.20.dist-info/METADATA +681 -0
- aip_agents_binary-0.5.20.dist-info/RECORD +280 -0
- aip_agents_binary-0.5.20.dist-info/WHEEL +5 -0
- 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
|
+
]
|