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,155 @@
1
+ """Demo: LangChain agent with allowed_tools whitelist filtering.
2
+
3
+ Shows how allowed_tools restricts tool access across 3 MCP servers:
4
+ - STDIO: only get_current_time allowed (not generate_uuid or get_weather_forecast)
5
+ - SSE: only get_random_quote allowed (not word_count or get_weather_forecast)
6
+ - HTTP: only get_random_fact allowed (not convert_to_base64 or get_weather_forecast)
7
+
8
+ Authors:
9
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
10
+ """
11
+
12
+ import asyncio
13
+ import logging
14
+ from typing import Any
15
+
16
+ from langchain_openai import ChatOpenAI
17
+
18
+ from aip_agents.agent import LangChainAgent
19
+ from aip_agents.examples.mcp_configs.configs import mcp_tool_whitelisting_demo
20
+ from aip_agents.schema.a2a import A2AStreamEventType
21
+ from aip_agents.utils.logger import logger_manager
22
+
23
+
24
+ def _normalize_event_type(raw_value: Any) -> A2AStreamEventType | None:
25
+ """Convert raw event_type payloads (enum or string) into A2AStreamEventType.
26
+
27
+ Args:
28
+ raw_value: Event type payload emitted by the agent stream (enum or string).
29
+
30
+ Returns:
31
+ Parsed A2AStreamEventType value, or None when the payload is unrecognized.
32
+ """
33
+ if isinstance(raw_value, A2AStreamEventType):
34
+ return raw_value
35
+ if isinstance(raw_value, str):
36
+ try:
37
+ return A2AStreamEventType(raw_value.lower())
38
+ except ValueError:
39
+ return None
40
+ return None
41
+
42
+
43
+ def _log_tool_call(event: dict[str, Any], is_allowed: bool) -> bool:
44
+ """Log tool call events and return True if at least one call existed.
45
+
46
+ Args:
47
+ event: A2A event payload containing tool_call metadata.
48
+ is_allowed: Whether the current query is expected to hit an allowed tool.
49
+
50
+ Returns:
51
+ True when at least one tool call entry is present; False otherwise.
52
+ """
53
+ tool_info = event.get("tool_info") or {}
54
+ tool_calls = tool_info.get("tool_calls") or []
55
+ if not tool_calls:
56
+ print("⚠️ Tool call event received without tool_calls payload")
57
+ return False
58
+
59
+ first_call = tool_calls[0]
60
+ tool_name = first_call.get("name", "unknown")
61
+ icon = "⚠️" if not is_allowed else "✅"
62
+ print(f"{icon} Tool called: {tool_name}")
63
+ return True
64
+
65
+
66
+ async def process_query(agent, query: str, is_allowed: bool = True) -> bool:
67
+ """Process a single query and report if tools were called.
68
+
69
+ Args:
70
+ agent: The LangChain agent to run.
71
+ query: The query to process.
72
+ is_allowed: Whether this query should use allowed tools.
73
+
74
+ Returns:
75
+ True if at least one tool was called while processing the query, False otherwise.
76
+ """
77
+ print(f"\nQuery: {query}")
78
+ tool_called = False
79
+ final_response = None
80
+ try:
81
+ async for event in agent.arun_a2a_stream(query=query):
82
+ event_type = _normalize_event_type(event.get("event_type"))
83
+ if event_type is None:
84
+ print(f"⚠️ Unknown event type payload: {event.get('event_type')!r}")
85
+ continue
86
+
87
+ if event_type is A2AStreamEventType.TOOL_CALL:
88
+ tool_called = _log_tool_call(event, is_allowed) or tool_called
89
+ continue
90
+
91
+ if event_type is A2AStreamEventType.FINAL_RESPONSE:
92
+ final_response = event.get("content")
93
+ break
94
+ except Exception as e:
95
+ print(f"❌ Error processing query: {e}")
96
+ return False
97
+ if final_response:
98
+ print(f"🧠 LLM response: {final_response}")
99
+ if not tool_called:
100
+ print("✅ No tool called (blocked or LLM answered directly)")
101
+ return tool_called
102
+
103
+
104
+ async def main():
105
+ """Demo allowed_tools filtering with 3 MCP servers (STDIO, SSE, HTTP)."""
106
+ logger_manager.set_level(logging.WARNING)
107
+
108
+ agent = LangChainAgent(
109
+ name="allowed_tools_demo",
110
+ instruction=(
111
+ "You are a helpful assistant with access to tools from 3 MCP servers.\n"
112
+ "Use the available tools to answer questions.\n"
113
+ "Note: Each server has specific tools whitelisted via allowed_tools."
114
+ ),
115
+ model=ChatOpenAI(model="gpt-4.1-mini"),
116
+ )
117
+ agent.add_mcp_server(mcp_tool_whitelisting_demo)
118
+
119
+ all_allowed_queries_called_tools = True
120
+ any_disallowed_query_called_tools = False
121
+
122
+ print("\n=== ALLOWED TOOLS (should work) ===")
123
+ allowed_queries = [
124
+ "Get the current time for me",
125
+ "Tell me an inspirational quote",
126
+ "Share a fun fact with me",
127
+ ]
128
+ for query in allowed_queries:
129
+ called = await process_query(agent, query, is_allowed=True)
130
+ all_allowed_queries_called_tools = all_allowed_queries_called_tools and called
131
+
132
+ print("\n\n=== DISALLOWED TOOLS (should fail or skip) ===")
133
+ disallowed_queries = [
134
+ "Generate a UUID for me", # Not allowed from STDIO
135
+ "Get the weather forecast for Jakarta", # Not allowed from any server
136
+ "Count words in 'hello world'", # Not allowed from SSE
137
+ ]
138
+ for query in disallowed_queries:
139
+ called = await process_query(agent, query, is_allowed=False)
140
+ any_disallowed_query_called_tools = any_disallowed_query_called_tools or called
141
+
142
+ print("\n=== SUMMARY: allowed_tools vs disabled_tools behavior ===")
143
+ if all_allowed_queries_called_tools:
144
+ print("✅ allowed_tools: All allowed queries called tools as expected.")
145
+ else:
146
+ print("❌ allowed_tools: Some allowed queries did not call any tools.")
147
+
148
+ if any_disallowed_query_called_tools:
149
+ print("❌ disabled_tools: Some disallowed queries still called tools.")
150
+ else:
151
+ print("✅ disabled_tools: No tools were called for any disallowed queries.")
152
+
153
+
154
+ if __name__ == "__main__":
155
+ asyncio.run(main())
@@ -0,0 +1,34 @@
1
+ """Example showing LangChain agent with MCP tools integration using SSE transport.
2
+
3
+ Authors:
4
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
5
+ """
6
+
7
+ import asyncio
8
+
9
+ from langchain_openai import ChatOpenAI
10
+
11
+ from aip_agents.agent import LangChainAgent
12
+ from aip_agents.examples.mcp_configs.configs import mcp_config_sse
13
+
14
+
15
+ async def main():
16
+ """Demonstrates the LangChainAgent with MCP tools via SSE transport."""
17
+ langchain_agent = LangChainAgent(
18
+ name="langchain_mcp_example",
19
+ instruction=(
20
+ "You are a helpful assistant that can browse the web. If you are blocked by captcha, try another website."
21
+ ),
22
+ model=ChatOpenAI(model="gpt-4.1", temperature=0),
23
+ )
24
+ langchain_agent.add_mcp_server(mcp_config_sse)
25
+
26
+ response = await langchain_agent.arun(query="What's the rating of the movie 'The Matrix'? browse it")
27
+ print(response["output"])
28
+ print("-" * 50)
29
+ response = await langchain_agent.arun(query="When is the next solar eclipse?, browse it")
30
+ print(response["output"])
31
+
32
+
33
+ if __name__ == "__main__":
34
+ asyncio.run(main())
@@ -0,0 +1,40 @@
1
+ """Example showing LangChain agent with MCP tools integration and streaming capabilities using SSE transport.
2
+
3
+ Authors:
4
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
5
+ """
6
+
7
+ import asyncio
8
+
9
+ from langchain_openai import ChatOpenAI
10
+
11
+ from aip_agents.agent import LangChainAgent
12
+ from aip_agents.examples.mcp_configs.configs import mcp_config_sse
13
+
14
+
15
+ async def main():
16
+ """Demonstrates the LangChainAgent with MCP tools via SSE transport and streaming capabilities."""
17
+ langchain_agent = LangChainAgent(
18
+ name="langchain_mcp_stream_example",
19
+ instruction="""You are a helpful assistant that can browse the web using playwright tools.""",
20
+ model=ChatOpenAI(model="gpt-4.1", temperature=0),
21
+ )
22
+ langchain_agent.add_mcp_server(mcp_config_sse)
23
+
24
+ full_response = ""
25
+ async for chunk in langchain_agent.arun_stream(
26
+ query="What are the result of Copa Del Rey Final in 2024/2025 season, browse it.",
27
+ ):
28
+ if isinstance(chunk, str):
29
+ print(chunk, end="", flush=True)
30
+ full_response += chunk
31
+ elif isinstance(chunk, dict) and "messages" in chunk:
32
+ print("\n(Stream finished with final state object)")
33
+ elif isinstance(chunk, dict):
34
+ pass
35
+
36
+ print(full_response)
37
+
38
+
39
+ if __name__ == "__main__":
40
+ asyncio.run(main())
@@ -0,0 +1,30 @@
1
+ """Example showing LangChain agent with MCP tools integration using stdio transport.
2
+
3
+ Authors:
4
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
5
+ """
6
+
7
+ import asyncio
8
+
9
+ from langchain_openai import ChatOpenAI
10
+
11
+ from aip_agents.agent import LangChainAgent
12
+ from aip_agents.examples.mcp_configs.configs import mcp_config_stdio
13
+
14
+
15
+ async def main():
16
+ """Demonstrates the LangChainAgent with MCP tools via stdio transport."""
17
+ langchain_agent = LangChainAgent(
18
+ name="langchain_mcp_example",
19
+ instruction="""You are a helpful assistant that can provide weather forecasts.
20
+ For weather, specify the day in lowercase (e.g., 'monday').""",
21
+ model=ChatOpenAI(model="gpt-4.1", temperature=0),
22
+ )
23
+ langchain_agent.add_mcp_server(mcp_config_stdio)
24
+
25
+ response = await langchain_agent.arun(query="What's the weather forecast for monday?")
26
+ print(response["output"])
27
+
28
+
29
+ if __name__ == "__main__":
30
+ asyncio.run(main())
@@ -0,0 +1,41 @@
1
+ """Example showing LangChain agent with MCP tools integration and streaming capabilities using stdio transport.
2
+
3
+ Authors:
4
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
5
+ """
6
+
7
+ import asyncio
8
+
9
+ from langchain_openai import ChatOpenAI
10
+
11
+ from aip_agents.agent import LangChainAgent
12
+ from aip_agents.examples.mcp_configs.configs import mcp_config_stdio
13
+
14
+
15
+ async def main():
16
+ """Demonstrates the LangChainAgent with MCP tools via stdio transport and streaming capabilities."""
17
+ langchain_agent = LangChainAgent(
18
+ name="langchain_mcp_stream_example",
19
+ instruction="""You are a helpful assistant that can provide weather forecasts.
20
+ For weather, specify the day in lowercase (e.g., 'monday').""",
21
+ model=ChatOpenAI(model="gpt-4.1", temperature=0),
22
+ )
23
+ langchain_agent.add_mcp_server(mcp_config_stdio)
24
+
25
+ full_response = ""
26
+ async for chunk in langchain_agent.arun_stream(
27
+ query="What's the weather forecast for monday?",
28
+ ):
29
+ if isinstance(chunk, str):
30
+ print(chunk, end="", flush=True)
31
+ full_response += chunk
32
+ elif isinstance(chunk, dict) and "messages" in chunk:
33
+ print("\n(Stream finished with final state object)")
34
+ elif isinstance(chunk, dict):
35
+ pass
36
+
37
+ print(full_response)
38
+
39
+
40
+ if __name__ == "__main__":
41
+ asyncio.run(main())
@@ -0,0 +1,36 @@
1
+ """Minimal LangChain agent example demonstrating streaming capabilities.
2
+
3
+ Authors:
4
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
5
+
6
+ """
7
+
8
+ import asyncio
9
+
10
+ from langchain_openai import ChatOpenAI
11
+
12
+ from aip_agents.agent import LangChainAgent
13
+ from aip_agents.examples.tools.langchain_arithmetic_tools import add_numbers
14
+
15
+
16
+ async def langchain_stream_example():
17
+ """Demonstrates the LangChainAgent's arun_stream method with async execution."""
18
+ agent = LangChainAgent(
19
+ name="LangChainStreamingCalculator",
20
+ instruction="You are a helpful calculator assistant that can add numbers. "
21
+ "When asked to add numbers, use the add_numbers tool. "
22
+ "Explain your steps clearly for streaming demonstration.",
23
+ model=ChatOpenAI(model="gpt-4.1", temperature=0),
24
+ tools=[add_numbers],
25
+ )
26
+
27
+ # Stream the response chunks
28
+ async for chunk in agent.arun_stream(
29
+ query="What is the sum of 23 and 47? And then add 10 to that, then add 5 more."
30
+ ):
31
+ if isinstance(chunk, str):
32
+ print(chunk, end="", flush=True)
33
+
34
+
35
+ if __name__ == "__main__":
36
+ asyncio.run(langchain_stream_example())
@@ -0,0 +1,39 @@
1
+ """Minimal LangChain agent example demonstrating streaming capabilities.
2
+
3
+ Authors:
4
+ Christian Trisno Sen Long Chen (christian.t.s.l.chen@gdplabs.id)
5
+ """
6
+
7
+ import asyncio
8
+
9
+ import dotenv
10
+
11
+ from aip_agents.agent import LangChainAgent
12
+ from aip_agents.examples.tools.langchain_arithmetic_tools import add_numbers
13
+
14
+ dotenv.load_dotenv(override=True)
15
+
16
+
17
+ async def langchain_stream_example():
18
+ """Demonstrates the LangChainAgent's arun_stream method with async execution."""
19
+ agent = LangChainAgent(
20
+ name="LangChainStreamingCalculator",
21
+ instruction="You are a helpful calculator assistant that can add numbers. "
22
+ "When asked to add numbers, use the add_numbers tool. "
23
+ "Explain your steps clearly for streaming demonstration.",
24
+ model="openai/gpt-4o",
25
+ tools=[add_numbers],
26
+ )
27
+
28
+ # Stream the response chunks
29
+ async for chunk in agent.arun_stream(
30
+ query="What is the sum of 23 and 47? And then add 10 to that, then add 5 more."
31
+ ):
32
+ if isinstance(chunk, str):
33
+ print(chunk, end="", flush=True) # Print text chunks as they come in
34
+ elif isinstance(chunk, dict):
35
+ print(f"Dict chunk: {chunk}")
36
+
37
+
38
+ if __name__ == "__main__":
39
+ asyncio.run(langchain_stream_example())
@@ -0,0 +1,163 @@
1
+ """Example demonstrating basic Langflow agent usage.
2
+
3
+ This example shows how to create and use a LangflowAgent for both
4
+ synchronous and asynchronous execution modes.
5
+
6
+ Authors:
7
+ Christian Trisno Sen Long Chen (christian.t.s.l.chen@gdplabs.id)
8
+ """
9
+
10
+ import asyncio
11
+ import os
12
+ import uuid
13
+
14
+ from dotenv import load_dotenv
15
+
16
+ from aip_agents.agent.langflow_agent import LangflowAgent
17
+ from aip_agents.clients.langflow import LangflowApiClient
18
+
19
+ load_dotenv()
20
+
21
+
22
+ async def fetch_flow_id() -> tuple[str, str]:
23
+ """Fetch available flows and return the first flow ID and name."""
24
+ print("Fetching available flows...")
25
+ temp_client = LangflowApiClient(base_url="https://langflow.obrol.id", api_key=os.getenv("LANGFLOW_API_KEY"))
26
+
27
+ try:
28
+ flows = await temp_client.get_all_flows(remove_example_flows=True)
29
+ print(flows)
30
+
31
+ if not flows:
32
+ print("❌ No flows found!")
33
+ return "", ""
34
+
35
+ print(f"Found {len(flows)} flows:")
36
+ for i, flow in enumerate(flows[:5]): # Show first 5 flows
37
+ flow_name = flow.get("name", "Unnamed")
38
+ flow_id = flow.get("id", "No ID")
39
+ print(f" {i + 1}. {flow_name} (ID: {flow_id})")
40
+
41
+ selected_flow = flows[2]
42
+ flow_id = selected_flow.get("id")
43
+ flow_name = selected_flow.get("name", "Unknown Flow")
44
+
45
+ if not flow_id:
46
+ print("❌ Selected flow has no ID!")
47
+ return "", ""
48
+
49
+ print(f"✅ Selected flow: {flow_name} (ID: {flow_id})")
50
+ print("-" * 50)
51
+ return flow_id, flow_name
52
+
53
+ except Exception as e:
54
+ print(f"❌ Error fetching flows: {e}")
55
+ print("Falling back to hardcoded flow ID...")
56
+ return "6dd45ac0-5c05-44c1-9825-66c6c6e516f7", "Fallback Flow"
57
+
58
+
59
+ async def create_agent(flow_id: str, flow_name: str) -> LangflowAgent:
60
+ """Create and configure the Langflow agent.
61
+
62
+ Args:
63
+ flow_id (str): The flow ID to use for the agent.
64
+ flow_name (str): The name of the flow.
65
+
66
+ Returns:
67
+ LangflowAgent: The configured Langflow agent.
68
+ """
69
+ agent = LangflowAgent(
70
+ name="HelloWorldLangflow",
71
+ flow_id=flow_id,
72
+ description=f"A Langflow agent using flow: {flow_name}",
73
+ base_url="https://langflow.obrol.id",
74
+ )
75
+
76
+ print(f"Created agent: {agent.name}")
77
+ print(f"Flow ID: {flow_id}")
78
+ print(f"API Base URL: {agent.api_client.base_url}")
79
+
80
+ # Test health check
81
+ print("Testing health check...")
82
+ is_healthy = await agent.health_check()
83
+ print(f"API Health: {'✓ Healthy' if is_healthy else '✗ Unhealthy'}")
84
+ print("-" * 50)
85
+
86
+ return agent
87
+
88
+
89
+ async def demonstrate_regular_execution(agent: LangflowAgent) -> None:
90
+ """Demonstrate regular execution.
91
+
92
+ Args:
93
+ agent (LangflowAgent): The Langflow agent to demonstrate with.
94
+ """
95
+ print("Example 1: Regular execution")
96
+ result1 = await agent.arun("Hello, how are you?")
97
+ print(f"Message 1: {result1}")
98
+ print("-" * 50)
99
+
100
+
101
+ async def demonstrate_streaming(agent: LangflowAgent) -> None:
102
+ """Demonstrate streaming execution.
103
+
104
+ Args:
105
+ agent (LangflowAgent): The Langflow agent to demonstrate with.
106
+ """
107
+ print("Example 2: Regular streaming")
108
+ try:
109
+ print("Streaming response:")
110
+ async for chunk in agent.arun_stream("Hello, how are you?"):
111
+ print(chunk)
112
+ print() # New line after streaming
113
+ except Exception as e:
114
+ print(f"Streaming Error: {e}")
115
+ print("-" * 50)
116
+
117
+
118
+ async def demonstrate_session_management(agent: LangflowAgent) -> None:
119
+ """Demonstrate session management.
120
+
121
+ Args:
122
+ agent (LangflowAgent): The Langflow agent to demonstrate with.
123
+ """
124
+ print("Example 3: Session management")
125
+ session_id = f"conversation-{uuid.uuid4()}"
126
+ try:
127
+ # First message with thread_id
128
+ result1 = await agent.arun("My name is Alice", configurable={"thread_id": session_id})
129
+ print(f"Message 1: {result1}")
130
+
131
+ # Second message in same thread - should remember context
132
+ result2 = await agent.arun("What's my name?", configurable={"thread_id": session_id})
133
+ print(f"Message 2: {result2}")
134
+
135
+ # Third message in different thread - should not remember
136
+ result3 = await agent.arun("What's my name?", configurable={"thread_id": f"new-conversation-{uuid.uuid4()}"})
137
+ print(f"Message 3 (new thread): {result3}")
138
+
139
+ except Exception as e:
140
+ print(f"Session Error: {e}")
141
+ print("-" * 50)
142
+
143
+
144
+ async def main():
145
+ """Demonstrate basic Langflow agent usage."""
146
+ # Fetch flow ID and name
147
+ flow_id, flow_name = await fetch_flow_id()
148
+ if not flow_id:
149
+ return
150
+
151
+ # Create agent
152
+ agent = await create_agent(flow_id, flow_name)
153
+
154
+ # Run demonstrations
155
+ await demonstrate_regular_execution(agent)
156
+ await demonstrate_streaming(agent)
157
+ await demonstrate_session_management(agent)
158
+
159
+ print("Demo completed!")
160
+
161
+
162
+ if __name__ == "__main__":
163
+ asyncio.run(main())
@@ -0,0 +1,39 @@
1
+ """Minimal LangGraph agent example demonstrating asynchronous run."""
2
+
3
+ import asyncio
4
+
5
+ from langchain_openai import ChatOpenAI
6
+
7
+ from aip_agents.agent import LangGraphAgent
8
+ from aip_agents.examples.tools.langchain_arithmetic_tools import add_numbers
9
+
10
+
11
+ async def langgraph_example():
12
+ """Demonstrates the LangGraphAgent's arun method."""
13
+ model = ChatOpenAI(model="gpt-4.1", temperature=0)
14
+ tools = [add_numbers]
15
+ agent_name = "LangGraphArithmeticAgent"
16
+
17
+ langgraph_agent = LangGraphAgent(
18
+ name=agent_name,
19
+ instruction="You are a helpful assistant that can add two numbers using the add_numbers tool.",
20
+ model=model,
21
+ tools=tools,
22
+ )
23
+
24
+ query = "What is the sum of 23 and 47? And then add 10 to that, then add 5 more."
25
+ print(f"--- Agent: {agent_name} ---")
26
+ print(f"Query: {query}")
27
+
28
+ print("\nRunning arun...")
29
+ response = await langgraph_agent.arun(
30
+ query=query,
31
+ configurable={"configurable": {"thread_id": "lgraph_arith_example_arun"}},
32
+ )
33
+ print(f"[arun] Final Response: {response['output']}")
34
+ print("--- End of LangGraph Example ---")
35
+
36
+
37
+ if __name__ == "__main__":
38
+ # OPENAI_API_KEY should be set in the environment.
39
+ asyncio.run(langgraph_example())
@@ -0,0 +1,41 @@
1
+ """Minimal LangGraph agent with bosa support example demonstrating asynchronous run.
2
+
3
+ Authors:
4
+ Saul Sayers (saul.sayers@gdplabs.id)
5
+ """
6
+
7
+ import asyncio
8
+
9
+ from langchain_openai import ChatOpenAI
10
+
11
+ from aip_agents.agent import LangGraphAgent
12
+ from aip_agents.tools import BOSA_AUTOMATED_TOOLS
13
+
14
+
15
+ async def langgraph_bosa_example():
16
+ """Demonstrates the LangGraphAgent's arun method."""
17
+ model = ChatOpenAI(model="gpt-4.1", temperature=0)
18
+ agent_name = "BOSAConnectorTwitterAgent"
19
+
20
+ langgraph_agent = LangGraphAgent(
21
+ name=agent_name,
22
+ instruction="You are a helpful assistant that use BOSA connector to connect with Twitter API.",
23
+ model=model,
24
+ tools=BOSA_AUTOMATED_TOOLS["twitter"],
25
+ )
26
+
27
+ query = "Get 3 tweets about the latest Air India Incident"
28
+ print(f"--- Agent: {agent_name} ---")
29
+ print(f"Query: {query}")
30
+
31
+ print("\nRunning arun...")
32
+ response = await langgraph_agent.arun(
33
+ query=query,
34
+ configurable={"configurable": {"thread_id": "lgraph_arith_example_arun"}},
35
+ )
36
+ print(f"[arun] Final Response: {response['output']}")
37
+ print("--- End of LangGraph Example ---")
38
+
39
+
40
+ if __name__ == "__main__":
41
+ asyncio.run(langgraph_bosa_example())
@@ -0,0 +1,31 @@
1
+ """Example showing LangGraph agent with MCP tools integration.
2
+
3
+ Authors:
4
+ Fachriza Dian Adhiatma (fachriza.d.adhiatma@gdplabs.id)
5
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
6
+ """
7
+
8
+ import asyncio
9
+
10
+ from langchain_openai import ChatOpenAI
11
+
12
+ from aip_agents.agent import LangGraphAgent
13
+ from aip_agents.examples.mcp_configs.configs import mcp_config_http
14
+
15
+
16
+ async def main():
17
+ """Demonstrates the LangGraphAgent with MCP tools via Streamable HTTP transport."""
18
+ langgraph_agent = LangGraphAgent(
19
+ name="langgraph_mcp_example",
20
+ instruction="""You are a helpful assistant that can provide weather forecasts.
21
+ For weather, specify the day in lowercase (e.g., 'monday').""",
22
+ model=ChatOpenAI(model="gpt-4.1", temperature=0),
23
+ )
24
+ langgraph_agent.add_mcp_server(mcp_config_http)
25
+
26
+ response = await langgraph_agent.arun(query="What's the weather forecast for monday?")
27
+ print(f"Response: {response['output']}")
28
+
29
+
30
+ if __name__ == "__main__":
31
+ asyncio.run(main())