google-adk 0.0.1__py3-none-any.whl → 0.0.3__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 (345) hide show
  1. google/adk/._version.py +0 -0
  2. google/adk/__init__.py +20 -0
  3. google/adk/agents/__init__.py +32 -0
  4. google/adk/agents/active_streaming_tool.py +38 -0
  5. google/adk/agents/base_agent.py +345 -0
  6. google/adk/agents/callback_context.py +112 -0
  7. google/adk/agents/invocation_context.py +181 -0
  8. google/adk/agents/langgraph_agent.py +140 -0
  9. google/adk/agents/live_request_queue.py +64 -0
  10. google/adk/agents/llm_agent.py +376 -0
  11. google/adk/agents/loop_agent.py +62 -0
  12. google/adk/agents/parallel_agent.py +96 -0
  13. google/adk/agents/readonly_context.py +46 -0
  14. google/adk/agents/remote_agent.py +50 -0
  15. google/adk/agents/run_config.py +87 -0
  16. google/adk/agents/sequential_agent.py +45 -0
  17. google/adk/agents/transcription_entry.py +34 -0
  18. google/adk/artifacts/__init__.py +23 -0
  19. google/adk/artifacts/base_artifact_service.py +128 -0
  20. google/adk/artifacts/gcs_artifact_service.py +195 -0
  21. google/adk/artifacts/in_memory_artifact_service.py +133 -0
  22. google/adk/auth/__init__.py +22 -0
  23. google/adk/auth/auth_credential.py +220 -0
  24. google/adk/auth/auth_handler.py +268 -0
  25. google/adk/auth/auth_preprocessor.py +116 -0
  26. google/adk/auth/auth_schemes.py +67 -0
  27. google/adk/auth/auth_tool.py +55 -0
  28. google/adk/cli/__init__.py +15 -0
  29. google/adk/cli/__main__.py +18 -0
  30. google/adk/cli/agent_graph.py +148 -0
  31. google/adk/cli/browser/adk_favicon.svg +17 -0
  32. google/adk/cli/browser/assets/audio-processor.js +51 -0
  33. google/adk/cli/browser/assets/config/runtime-config.json +3 -0
  34. google/adk/cli/browser/index.html +33 -0
  35. google/adk/cli/browser/main-SY2WYYGV.js +75 -0
  36. google/adk/cli/browser/polyfills-FFHMD2TL.js +18 -0
  37. google/adk/cli/browser/styles-4VDSPQ37.css +17 -0
  38. google/adk/cli/cli.py +181 -0
  39. google/adk/cli/cli_deploy.py +181 -0
  40. google/adk/cli/cli_eval.py +282 -0
  41. google/adk/cli/cli_tools_click.py +524 -0
  42. google/adk/cli/fast_api.py +784 -0
  43. google/adk/cli/utils/__init__.py +49 -0
  44. google/adk/cli/utils/envs.py +57 -0
  45. google/adk/cli/utils/evals.py +93 -0
  46. google/adk/cli/utils/logs.py +72 -0
  47. google/adk/code_executors/__init__.py +49 -0
  48. google/adk/code_executors/base_code_executor.py +97 -0
  49. google/adk/code_executors/code_execution_utils.py +256 -0
  50. google/adk/code_executors/code_executor_context.py +202 -0
  51. google/adk/code_executors/container_code_executor.py +196 -0
  52. google/adk/code_executors/unsafe_local_code_executor.py +71 -0
  53. google/adk/code_executors/vertex_ai_code_executor.py +234 -0
  54. google/adk/docs/Makefile +20 -0
  55. google/adk/docs/build/doctrees/google-adk.doctree +0 -0
  56. google/adk/docs/build/html/_sources/google-adk.rst.txt +98 -0
  57. google/adk/docs/build/html/_sources/index.rst.txt +7 -0
  58. google/adk/docs/build/html/_static/autodoc_pydantic.css +27 -0
  59. google/adk/docs/build/html/_static/basic.css +925 -0
  60. google/adk/docs/build/html/_static/debug.css +85 -0
  61. google/adk/docs/build/html/_static/doctools.js +156 -0
  62. google/adk/docs/build/html/_static/documentation_options.js +29 -0
  63. google/adk/docs/build/html/_static/file.png +0 -0
  64. google/adk/docs/build/html/_static/language_data.js +199 -0
  65. google/adk/docs/build/html/_static/minus.png +0 -0
  66. google/adk/docs/build/html/_static/plus.png +0 -0
  67. google/adk/docs/build/html/_static/pygments.css +274 -0
  68. google/adk/docs/build/html/_static/scripts/furo-extensions.js +16 -0
  69. google/adk/docs/build/html/_static/scripts/furo.js +19 -0
  70. google/adk/docs/build/html/_static/scripts/furo.js.LICENSE.txt +7 -0
  71. google/adk/docs/build/html/_static/scripts/furo.js.map +1 -0
  72. google/adk/docs/build/html/_static/searchtools.js +620 -0
  73. google/adk/docs/build/html/_static/skeleton.css +312 -0
  74. google/adk/docs/build/html/_static/sphinx_highlight.js +170 -0
  75. google/adk/docs/build/html/_static/styles/furo-extensions.css +18 -0
  76. google/adk/docs/build/html/_static/styles/furo-extensions.css.map +1 -0
  77. google/adk/docs/build/html/_static/styles/furo.css +18 -0
  78. google/adk/docs/build/html/_static/styles/furo.css.map +1 -0
  79. google/adk/docs/build/html/genindex.html +861 -0
  80. google/adk/docs/build/html/google-adk.html +5461 -0
  81. google/adk/docs/build/html/index.html +567 -0
  82. google/adk/docs/build/html/objects.inv +0 -0
  83. google/adk/docs/build/html/py-modindex.html +373 -0
  84. google/adk/docs/build/html/search.html +333 -0
  85. google/adk/docs/build/html/searchindex.js +17 -0
  86. google/adk/docs/source/conf.py +133 -0
  87. google/adk/docs/source/google-adk.rst +98 -0
  88. google/adk/docs/source/index.rst +7 -0
  89. google/adk/evaluation/__init__.py +31 -0
  90. google/adk/evaluation/agent_evaluator.py +329 -0
  91. google/adk/evaluation/evaluation_constants.py +24 -0
  92. google/adk/evaluation/evaluation_generator.py +270 -0
  93. google/adk/evaluation/response_evaluator.py +135 -0
  94. google/adk/evaluation/trajectory_evaluator.py +184 -0
  95. google/adk/events/__init__.py +21 -0
  96. google/adk/events/event.py +130 -0
  97. google/adk/events/event_actions.py +55 -0
  98. google/adk/examples/__init__.py +28 -0
  99. google/adk/examples/base_example_provider.py +35 -0
  100. google/adk/examples/example.py +27 -0
  101. google/adk/examples/example_util.py +123 -0
  102. google/adk/examples/vertex_ai_example_store.py +104 -0
  103. google/adk/flows/__init__.py +14 -0
  104. google/adk/flows/llm_flows/__init__.py +20 -0
  105. google/adk/flows/llm_flows/_base_llm_processor.py +52 -0
  106. google/adk/flows/llm_flows/_code_execution.py +458 -0
  107. google/adk/flows/llm_flows/_nl_planning.py +129 -0
  108. google/adk/flows/llm_flows/agent_transfer.py +132 -0
  109. google/adk/flows/llm_flows/audio_transcriber.py +109 -0
  110. google/adk/flows/llm_flows/auto_flow.py +49 -0
  111. google/adk/flows/llm_flows/base_llm_flow.py +559 -0
  112. google/adk/flows/llm_flows/basic.py +72 -0
  113. google/adk/flows/llm_flows/contents.py +370 -0
  114. google/adk/flows/llm_flows/functions.py +486 -0
  115. google/adk/flows/llm_flows/identity.py +47 -0
  116. google/adk/flows/llm_flows/instructions.py +137 -0
  117. google/adk/flows/llm_flows/single_flow.py +57 -0
  118. google/adk/memory/__init__.py +35 -0
  119. google/adk/memory/base_memory_service.py +74 -0
  120. google/adk/memory/in_memory_memory_service.py +62 -0
  121. google/adk/memory/vertex_ai_rag_memory_service.py +177 -0
  122. google/adk/models/__init__.py +31 -0
  123. google/adk/models/anthropic_llm.py +243 -0
  124. google/adk/models/base_llm.py +87 -0
  125. google/adk/models/base_llm_connection.py +76 -0
  126. google/adk/models/gemini_llm_connection.py +200 -0
  127. google/adk/models/google_llm.py +331 -0
  128. google/adk/models/lite_llm.py +673 -0
  129. google/adk/models/llm_request.py +98 -0
  130. google/adk/models/llm_response.py +111 -0
  131. google/adk/models/registry.py +102 -0
  132. google/adk/planners/__init__.py +23 -0
  133. google/adk/planners/base_planner.py +66 -0
  134. google/adk/planners/built_in_planner.py +75 -0
  135. google/adk/planners/plan_re_act_planner.py +208 -0
  136. google/adk/runners.py +456 -0
  137. google/adk/sessions/__init__.py +41 -0
  138. google/adk/sessions/base_session_service.py +133 -0
  139. google/adk/sessions/database_session_service.py +522 -0
  140. google/adk/sessions/in_memory_session_service.py +206 -0
  141. google/adk/sessions/session.py +54 -0
  142. google/adk/sessions/state.py +71 -0
  143. google/adk/sessions/vertex_ai_session_service.py +356 -0
  144. google/adk/telemetry.py +189 -0
  145. google/adk/tests/__init__.py +14 -0
  146. google/adk/tests/integration/.env.example +10 -0
  147. google/adk/tests/integration/__init__.py +18 -0
  148. google/adk/tests/integration/conftest.py +119 -0
  149. google/adk/tests/integration/fixture/__init__.py +14 -0
  150. google/adk/tests/integration/fixture/agent_with_config/__init__.py +15 -0
  151. google/adk/tests/integration/fixture/agent_with_config/agent.py +88 -0
  152. google/adk/tests/integration/fixture/callback_agent/__init__.py +15 -0
  153. google/adk/tests/integration/fixture/callback_agent/agent.py +105 -0
  154. google/adk/tests/integration/fixture/context_update_test/OWNERS +1 -0
  155. google/adk/tests/integration/fixture/context_update_test/__init__.py +15 -0
  156. google/adk/tests/integration/fixture/context_update_test/agent.py +43 -0
  157. google/adk/tests/integration/fixture/context_update_test/successful_test.session.json +582 -0
  158. google/adk/tests/integration/fixture/context_variable_agent/__init__.py +15 -0
  159. google/adk/tests/integration/fixture/context_variable_agent/agent.py +115 -0
  160. google/adk/tests/integration/fixture/customer_support_ma/__init__.py +15 -0
  161. google/adk/tests/integration/fixture/customer_support_ma/agent.py +172 -0
  162. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/__init__.py +15 -0
  163. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/agent.py +338 -0
  164. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/order_query.test.json +69 -0
  165. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/test_config.json +6 -0
  166. google/adk/tests/integration/fixture/flow_complex_spark/__init__.py +15 -0
  167. google/adk/tests/integration/fixture/flow_complex_spark/agent.py +182 -0
  168. google/adk/tests/integration/fixture/flow_complex_spark/sample.session.json +190 -0
  169. google/adk/tests/integration/fixture/hello_world_agent/__init__.py +15 -0
  170. google/adk/tests/integration/fixture/hello_world_agent/agent.py +95 -0
  171. google/adk/tests/integration/fixture/hello_world_agent/roll_die.test.json +24 -0
  172. google/adk/tests/integration/fixture/hello_world_agent/test_config.json +6 -0
  173. google/adk/tests/integration/fixture/home_automation_agent/__init__.py +15 -0
  174. google/adk/tests/integration/fixture/home_automation_agent/agent.py +304 -0
  175. google/adk/tests/integration/fixture/home_automation_agent/simple_test.test.json +5 -0
  176. google/adk/tests/integration/fixture/home_automation_agent/simple_test2.test.json +5 -0
  177. google/adk/tests/integration/fixture/home_automation_agent/test_config.json +5 -0
  178. google/adk/tests/integration/fixture/home_automation_agent/test_files/dependent_tool_calls.test.json +18 -0
  179. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/eval_data.test.json +17 -0
  180. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/test_config.json +6 -0
  181. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_multi_turn_conversation.test.json +18 -0
  182. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test.test.json +17 -0
  183. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test2.test.json +5 -0
  184. google/adk/tests/integration/fixture/home_automation_agent/test_files/test_config.json +5 -0
  185. google/adk/tests/integration/fixture/tool_agent/__init__.py +15 -0
  186. google/adk/tests/integration/fixture/tool_agent/agent.py +218 -0
  187. google/adk/tests/integration/fixture/tool_agent/files/Agent_test_plan.pdf +0 -0
  188. google/adk/tests/integration/fixture/trip_planner_agent/__init__.py +15 -0
  189. google/adk/tests/integration/fixture/trip_planner_agent/agent.py +110 -0
  190. google/adk/tests/integration/fixture/trip_planner_agent/initial.session.json +13 -0
  191. google/adk/tests/integration/fixture/trip_planner_agent/test_config.json +5 -0
  192. google/adk/tests/integration/fixture/trip_planner_agent/test_files/initial.session.json +13 -0
  193. google/adk/tests/integration/fixture/trip_planner_agent/test_files/test_config.json +5 -0
  194. google/adk/tests/integration/fixture/trip_planner_agent/test_files/trip_inquiry_sub_agent.test.json +7 -0
  195. google/adk/tests/integration/fixture/trip_planner_agent/trip_inquiry.test.json +19 -0
  196. google/adk/tests/integration/models/__init__.py +14 -0
  197. google/adk/tests/integration/models/test_google_llm.py +65 -0
  198. google/adk/tests/integration/test_callback.py +70 -0
  199. google/adk/tests/integration/test_context_variable.py +67 -0
  200. google/adk/tests/integration/test_evalute_agent_in_fixture.py +76 -0
  201. google/adk/tests/integration/test_multi_agent.py +28 -0
  202. google/adk/tests/integration/test_multi_turn.py +42 -0
  203. google/adk/tests/integration/test_single_agent.py +23 -0
  204. google/adk/tests/integration/test_sub_agent.py +26 -0
  205. google/adk/tests/integration/test_system_instruction.py +177 -0
  206. google/adk/tests/integration/test_tools.py +287 -0
  207. google/adk/tests/integration/test_with_test_file.py +34 -0
  208. google/adk/tests/integration/tools/__init__.py +14 -0
  209. google/adk/tests/integration/utils/__init__.py +16 -0
  210. google/adk/tests/integration/utils/asserts.py +75 -0
  211. google/adk/tests/integration/utils/test_runner.py +97 -0
  212. google/adk/tests/unittests/__init__.py +14 -0
  213. google/adk/tests/unittests/agents/__init__.py +14 -0
  214. google/adk/tests/unittests/agents/test_base_agent.py +407 -0
  215. google/adk/tests/unittests/agents/test_langgraph_agent.py +191 -0
  216. google/adk/tests/unittests/agents/test_llm_agent_callbacks.py +138 -0
  217. google/adk/tests/unittests/agents/test_llm_agent_fields.py +231 -0
  218. google/adk/tests/unittests/agents/test_loop_agent.py +136 -0
  219. google/adk/tests/unittests/agents/test_parallel_agent.py +92 -0
  220. google/adk/tests/unittests/agents/test_sequential_agent.py +114 -0
  221. google/adk/tests/unittests/artifacts/__init__.py +14 -0
  222. google/adk/tests/unittests/artifacts/test_artifact_service.py +276 -0
  223. google/adk/tests/unittests/auth/test_auth_handler.py +575 -0
  224. google/adk/tests/unittests/conftest.py +73 -0
  225. google/adk/tests/unittests/fast_api/__init__.py +14 -0
  226. google/adk/tests/unittests/fast_api/test_fast_api.py +269 -0
  227. google/adk/tests/unittests/flows/__init__.py +14 -0
  228. google/adk/tests/unittests/flows/llm_flows/__init__.py +14 -0
  229. google/adk/tests/unittests/flows/llm_flows/_test_examples.py +142 -0
  230. google/adk/tests/unittests/flows/llm_flows/test_agent_transfer.py +311 -0
  231. google/adk/tests/unittests/flows/llm_flows/test_functions_long_running.py +244 -0
  232. google/adk/tests/unittests/flows/llm_flows/test_functions_request_euc.py +346 -0
  233. google/adk/tests/unittests/flows/llm_flows/test_functions_sequential.py +93 -0
  234. google/adk/tests/unittests/flows/llm_flows/test_functions_simple.py +258 -0
  235. google/adk/tests/unittests/flows/llm_flows/test_identity.py +66 -0
  236. google/adk/tests/unittests/flows/llm_flows/test_instructions.py +164 -0
  237. google/adk/tests/unittests/flows/llm_flows/test_model_callbacks.py +142 -0
  238. google/adk/tests/unittests/flows/llm_flows/test_other_configs.py +46 -0
  239. google/adk/tests/unittests/flows/llm_flows/test_tool_callbacks.py +269 -0
  240. google/adk/tests/unittests/models/__init__.py +14 -0
  241. google/adk/tests/unittests/models/test_google_llm.py +224 -0
  242. google/adk/tests/unittests/models/test_litellm.py +804 -0
  243. google/adk/tests/unittests/models/test_models.py +60 -0
  244. google/adk/tests/unittests/sessions/__init__.py +14 -0
  245. google/adk/tests/unittests/sessions/test_session_service.py +227 -0
  246. google/adk/tests/unittests/sessions/test_vertex_ai_session_service.py +246 -0
  247. google/adk/tests/unittests/streaming/__init__.py +14 -0
  248. google/adk/tests/unittests/streaming/test_streaming.py +50 -0
  249. google/adk/tests/unittests/tools/__init__.py +14 -0
  250. google/adk/tests/unittests/tools/apihub_tool/clients/test_apihub_client.py +499 -0
  251. google/adk/tests/unittests/tools/apihub_tool/test_apihub_toolset.py +204 -0
  252. google/adk/tests/unittests/tools/application_integration_tool/clients/test_connections_client.py +600 -0
  253. google/adk/tests/unittests/tools/application_integration_tool/clients/test_integration_client.py +630 -0
  254. google/adk/tests/unittests/tools/application_integration_tool/test_application_integration_toolset.py +345 -0
  255. google/adk/tests/unittests/tools/google_api_tool/__init__.py +13 -0
  256. google/adk/tests/unittests/tools/google_api_tool/test_googleapi_to_openapi_converter.py +657 -0
  257. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_auto_auth_credential_exchanger.py +145 -0
  258. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_base_auth_credential_exchanger.py +68 -0
  259. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_oauth2_exchanger.py +153 -0
  260. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_service_account_exchanger.py +196 -0
  261. google/adk/tests/unittests/tools/openapi_tool/auth/test_auth_helper.py +573 -0
  262. google/adk/tests/unittests/tools/openapi_tool/common/test_common.py +436 -0
  263. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test.yaml +1367 -0
  264. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_spec_parser.py +628 -0
  265. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_toolset.py +139 -0
  266. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_operation_parser.py +406 -0
  267. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_rest_api_tool.py +966 -0
  268. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_tool_auth_handler.py +201 -0
  269. google/adk/tests/unittests/tools/retrieval/__init__.py +14 -0
  270. google/adk/tests/unittests/tools/retrieval/test_vertex_ai_rag_retrieval.py +147 -0
  271. google/adk/tests/unittests/tools/test_agent_tool.py +167 -0
  272. google/adk/tests/unittests/tools/test_base_tool.py +141 -0
  273. google/adk/tests/unittests/tools/test_build_function_declaration.py +277 -0
  274. google/adk/tests/unittests/utils.py +304 -0
  275. google/adk/tools/__init__.py +51 -0
  276. google/adk/tools/_automatic_function_calling_util.py +346 -0
  277. google/adk/tools/agent_tool.py +176 -0
  278. google/adk/tools/apihub_tool/__init__.py +19 -0
  279. google/adk/tools/apihub_tool/apihub_toolset.py +209 -0
  280. google/adk/tools/apihub_tool/clients/__init__.py +13 -0
  281. google/adk/tools/apihub_tool/clients/apihub_client.py +332 -0
  282. google/adk/tools/apihub_tool/clients/secret_client.py +115 -0
  283. google/adk/tools/application_integration_tool/__init__.py +19 -0
  284. google/adk/tools/application_integration_tool/application_integration_toolset.py +230 -0
  285. google/adk/tools/application_integration_tool/clients/connections_client.py +903 -0
  286. google/adk/tools/application_integration_tool/clients/integration_client.py +253 -0
  287. google/adk/tools/base_tool.py +144 -0
  288. google/adk/tools/built_in_code_execution_tool.py +59 -0
  289. google/adk/tools/crewai_tool.py +72 -0
  290. google/adk/tools/example_tool.py +62 -0
  291. google/adk/tools/exit_loop_tool.py +23 -0
  292. google/adk/tools/function_parameter_parse_util.py +307 -0
  293. google/adk/tools/function_tool.py +87 -0
  294. google/adk/tools/get_user_choice_tool.py +28 -0
  295. google/adk/tools/google_api_tool/__init__.py +14 -0
  296. google/adk/tools/google_api_tool/google_api_tool.py +59 -0
  297. google/adk/tools/google_api_tool/google_api_tool_set.py +107 -0
  298. google/adk/tools/google_api_tool/google_api_tool_sets.py +55 -0
  299. google/adk/tools/google_api_tool/googleapi_to_openapi_converter.py +521 -0
  300. google/adk/tools/google_search_tool.py +68 -0
  301. google/adk/tools/langchain_tool.py +86 -0
  302. google/adk/tools/load_artifacts_tool.py +113 -0
  303. google/adk/tools/load_memory_tool.py +58 -0
  304. google/adk/tools/load_web_page.py +41 -0
  305. google/adk/tools/long_running_tool.py +39 -0
  306. google/adk/tools/mcp_tool/__init__.py +42 -0
  307. google/adk/tools/mcp_tool/conversion_utils.py +161 -0
  308. google/adk/tools/mcp_tool/mcp_tool.py +113 -0
  309. google/adk/tools/mcp_tool/mcp_toolset.py +272 -0
  310. google/adk/tools/openapi_tool/__init__.py +21 -0
  311. google/adk/tools/openapi_tool/auth/__init__.py +19 -0
  312. google/adk/tools/openapi_tool/auth/auth_helpers.py +498 -0
  313. google/adk/tools/openapi_tool/auth/credential_exchangers/__init__.py +25 -0
  314. google/adk/tools/openapi_tool/auth/credential_exchangers/auto_auth_credential_exchanger.py +105 -0
  315. google/adk/tools/openapi_tool/auth/credential_exchangers/base_credential_exchanger.py +55 -0
  316. google/adk/tools/openapi_tool/auth/credential_exchangers/oauth2_exchanger.py +117 -0
  317. google/adk/tools/openapi_tool/auth/credential_exchangers/service_account_exchanger.py +97 -0
  318. google/adk/tools/openapi_tool/common/__init__.py +19 -0
  319. google/adk/tools/openapi_tool/common/common.py +300 -0
  320. google/adk/tools/openapi_tool/openapi_spec_parser/__init__.py +32 -0
  321. google/adk/tools/openapi_tool/openapi_spec_parser/openapi_spec_parser.py +231 -0
  322. google/adk/tools/openapi_tool/openapi_spec_parser/openapi_toolset.py +144 -0
  323. google/adk/tools/openapi_tool/openapi_spec_parser/operation_parser.py +260 -0
  324. google/adk/tools/openapi_tool/openapi_spec_parser/rest_api_tool.py +496 -0
  325. google/adk/tools/openapi_tool/openapi_spec_parser/tool_auth_handler.py +268 -0
  326. google/adk/tools/preload_memory_tool.py +72 -0
  327. google/adk/tools/retrieval/__init__.py +36 -0
  328. google/adk/tools/retrieval/base_retrieval_tool.py +37 -0
  329. google/adk/tools/retrieval/files_retrieval.py +33 -0
  330. google/adk/tools/retrieval/llama_index_retrieval.py +41 -0
  331. google/adk/tools/retrieval/vertex_ai_rag_retrieval.py +107 -0
  332. google/adk/tools/tool_context.py +90 -0
  333. google/adk/tools/toolbox_tool.py +46 -0
  334. google/adk/tools/transfer_to_agent_tool.py +21 -0
  335. google/adk/tools/vertex_ai_search_tool.py +96 -0
  336. google/adk/version.py +16 -0
  337. google_adk-0.0.3.dist-info/METADATA +73 -0
  338. google_adk-0.0.3.dist-info/RECORD +340 -0
  339. {google_adk-0.0.1.dist-info → google_adk-0.0.3.dist-info}/WHEEL +1 -2
  340. google_adk-0.0.3.dist-info/entry_points.txt +3 -0
  341. agent_kit/__init__.py +0 -0
  342. google_adk-0.0.1.dist-info/LICENSE.txt +0 -170
  343. google_adk-0.0.1.dist-info/METADATA +0 -15
  344. google_adk-0.0.1.dist-info/RECORD +0 -6
  345. google_adk-0.0.1.dist-info/top_level.txt +0 -1
@@ -0,0 +1,177 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ import pytest
16
+
17
+ # Skip until fixed.
18
+ pytest.skip(allow_module_level=True)
19
+
20
+ from google.adk.agents import InvocationContext
21
+ from google.adk.sessions import Session
22
+ from google.genai import types
23
+
24
+ from .fixture import context_variable_agent
25
+ from .utils import TestRunner
26
+
27
+ nl_planner_si = """
28
+ You are an intelligent tool use agent built upon the Gemini large language model. When answering the question, try to leverage the available tools to gather the information instead of your memorized knowledge.
29
+
30
+ Follow this process when answering the question: (1) first come up with a plan in natural language text format; (2) Then use tools to execute the plan and provide reasoning between tool code snippets to make a summary of current state and next step. Tool code snippets and reasoning should be interleaved with each other. (3) In the end, return one final answer.
31
+
32
+ Follow this format when answering the question: (1) The planning part should be under /*PLANNING*/. (2) The tool code snippets should be under /*ACTION*/, and the reasoning parts should be under /*REASONING*/. (3) The final answer part should be under /*FINAL_ANSWER*/.
33
+
34
+
35
+ Below are the requirements for the planning:
36
+ The plan is made to answer the user query if following the plan. The plan is coherent and covers all aspects of information from user query, and only involves the tools that are accessible by the agent. The plan contains the decomposed steps as a numbered list where each step should use one or multiple available tools. By reading the plan, you can intuitively know which tools to trigger or what actions to take.
37
+ If the initial plan cannot be successfully executed, you should learn from previous execution results and revise your plan. The revised plan should be be under /*REPLANNING*/. Then use tools to follow the new plan.
38
+
39
+ Below are the requirements for the reasoning:
40
+ The reasoning makes a summary of the current trajectory based on the user query and tool outputs. Based on the tool outputs and plan, the reasoning also comes up with instructions to the next steps, making the trajectory closer to the final answer.
41
+
42
+
43
+
44
+ Below are the requirements for the final answer:
45
+ The final answer should be precise and follow query formatting requirements. Some queries may not be answerable with the available tools and information. In those cases, inform the user why you cannot process their query and ask for more information.
46
+
47
+
48
+
49
+ Below are the requirements for the tool code:
50
+
51
+ **Custom Tools:** The available tools are described in the context and can be directly used.
52
+ - Code must be valid self-contained Python snippets with no imports and no references to tools or Python libraries that are not in the context.
53
+ - You cannot use any parameters or fields that are not explicitly defined in the APIs in the context.
54
+ - Use "print" to output execution results for the next step or final answer that you need for responding to the user. Never generate ```tool_outputs yourself.
55
+ - The code snippets should be readable, efficient, and directly relevant to the user query and reasoning steps.
56
+ - When using the tools, you should use the library name together with the function name, e.g., vertex_search.search().
57
+ - If Python libraries are not provided in the context, NEVER write your own code other than the function calls using the provided tools.
58
+
59
+
60
+
61
+ VERY IMPORTANT instruction that you MUST follow in addition to the above instructions:
62
+
63
+ You should ask for clarification if you need more information to answer the question.
64
+ You should prefer using the information available in the context instead of repeated tool use.
65
+
66
+ You should ONLY generate code snippets prefixed with "```tool_code" if you need to use the tools to answer the question.
67
+
68
+ If you are asked to write code by user specifically,
69
+ - you should ALWAYS use "```python" to format the code.
70
+ - you should NEVER put "tool_code" to format the code.
71
+ - Good example:
72
+ ```python
73
+ print('hello')
74
+ ```
75
+ - Bad example:
76
+ ```tool_code
77
+ print('hello')
78
+ ```
79
+ """
80
+
81
+
82
+ @pytest.mark.parametrize(
83
+ "agent_runner",
84
+ [{"agent": context_variable_agent.agent.state_variable_echo_agent}],
85
+ indirect=True,
86
+ )
87
+ def test_context_variable(agent_runner: TestRunner):
88
+ session = Session(
89
+ context={
90
+ "customerId": "1234567890",
91
+ "customerInt": 30,
92
+ "customerFloat": 12.34,
93
+ "customerJson": {"name": "John Doe", "age": 30, "count": 11.1},
94
+ }
95
+ )
96
+ si = UnitFlow()._build_system_instruction(
97
+ InvocationContext(
98
+ invocation_id="1234567890", agent=agent_runner.agent, session=session
99
+ )
100
+ )
101
+
102
+ assert (
103
+ "Use the echo_info tool to echo 1234567890, 30, 12.34, and {'name': 'John"
104
+ " Doe', 'age': 30, 'count': 11.1}. Ask for it if you need to."
105
+ in si
106
+ )
107
+
108
+
109
+ @pytest.mark.parametrize(
110
+ "agent_runner",
111
+ [{
112
+ "agent": (
113
+ context_variable_agent.agent.state_variable_with_complicated_format_agent
114
+ )
115
+ }],
116
+ indirect=True,
117
+ )
118
+ def test_context_variable_with_complicated_format(agent_runner: TestRunner):
119
+ session = Session(
120
+ context={"customerId": "1234567890", "customer_int": 30},
121
+ artifacts={"fileName": [types.Part(text="test artifact")]},
122
+ )
123
+ si = _context_formatter.populate_context_and_artifact_variable_values(
124
+ agent_runner.agent.instruction,
125
+ session.get_state(),
126
+ session.get_artifact_dict(),
127
+ )
128
+
129
+ assert (
130
+ si
131
+ == "Use the echo_info tool to echo 1234567890, 30, { "
132
+ " non-identifier-float}}, test artifact, {'key1': 'value1'} and"
133
+ " {{'key2': 'value2'}}. Ask for it if you need to."
134
+ )
135
+
136
+
137
+ @pytest.mark.parametrize(
138
+ "agent_runner",
139
+ [{
140
+ "agent": (
141
+ context_variable_agent.agent.state_variable_with_nl_planner_agent
142
+ )
143
+ }],
144
+ indirect=True,
145
+ )
146
+ def test_nl_planner(agent_runner: TestRunner):
147
+ session = Session(context={"customerId": "1234567890"})
148
+ si = UnitFlow()._build_system_instruction(
149
+ InvocationContext(
150
+ invocation_id="1234567890",
151
+ agent=agent_runner.agent,
152
+ session=session,
153
+ )
154
+ )
155
+
156
+ for line in nl_planner_si.splitlines():
157
+ assert line in si
158
+
159
+
160
+ @pytest.mark.parametrize(
161
+ "agent_runner",
162
+ [{
163
+ "agent": (
164
+ context_variable_agent.agent.state_variable_with_function_instruction_agent
165
+ )
166
+ }],
167
+ indirect=True,
168
+ )
169
+ def test_function_instruction(agent_runner: TestRunner):
170
+ session = Session(context={"customerId": "1234567890"})
171
+ si = UnitFlow()._build_system_instruction(
172
+ InvocationContext(
173
+ invocation_id="1234567890", agent=agent_runner.agent, session=session
174
+ )
175
+ )
176
+
177
+ assert "This is the plain text sub agent instruction." in si
@@ -0,0 +1,287 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ import json
16
+
17
+ import pytest
18
+
19
+ # Skip until fixed.
20
+ pytest.skip(allow_module_level=True)
21
+
22
+ from .fixture import tool_agent
23
+ from .utils import TestRunner
24
+
25
+
26
+ @pytest.mark.parametrize(
27
+ "agent_runner",
28
+ [{"agent": tool_agent.agent.single_function_agent}],
29
+ indirect=True,
30
+ )
31
+ def test_single_function_calls_success(agent_runner: TestRunner):
32
+ _call_function_and_assert(
33
+ agent_runner,
34
+ "simple_function",
35
+ "test",
36
+ "success",
37
+ )
38
+
39
+
40
+ @pytest.mark.parametrize(
41
+ "agent_runner",
42
+ [{"agent": tool_agent.agent.root_agent}],
43
+ indirect=True,
44
+ )
45
+ def test_multiple_function_calls_success(agent_runner: TestRunner):
46
+ _call_function_and_assert(
47
+ agent_runner,
48
+ "simple_function",
49
+ "test",
50
+ "success",
51
+ )
52
+ _call_function_and_assert(
53
+ agent_runner,
54
+ "no_param_function",
55
+ None,
56
+ "Called no param function successfully",
57
+ )
58
+ _call_function_and_assert(
59
+ agent_runner,
60
+ "no_output_function",
61
+ "test",
62
+ "",
63
+ )
64
+ _call_function_and_assert(
65
+ agent_runner,
66
+ "multiple_param_types_function",
67
+ ["test", 1, 2.34, True],
68
+ "success",
69
+ )
70
+ _call_function_and_assert(
71
+ agent_runner,
72
+ "return_list_str_function",
73
+ "test",
74
+ "success",
75
+ )
76
+ _call_function_and_assert(
77
+ agent_runner,
78
+ "list_str_param_function",
79
+ ["test", "test2", "test3", "test4"],
80
+ "success",
81
+ )
82
+
83
+
84
+ @pytest.mark.skip(reason="Currently failing with 400 on MLDev.")
85
+ @pytest.mark.parametrize(
86
+ "agent_runner",
87
+ [{"agent": tool_agent.agent.root_agent}],
88
+ indirect=True,
89
+ )
90
+ def test_complex_function_calls_success(agent_runner: TestRunner):
91
+ param1 = {"name": "Test", "count": 3}
92
+ param2 = [
93
+ {"name": "Function", "count": 2},
94
+ {"name": "Retrieval", "count": 1},
95
+ ]
96
+ _call_function_and_assert(
97
+ agent_runner,
98
+ "complex_function_list_dict",
99
+ [param1, param2],
100
+ "test",
101
+ )
102
+
103
+
104
+ @pytest.mark.parametrize(
105
+ "agent_runner",
106
+ [{"agent": tool_agent.agent.root_agent}],
107
+ indirect=True,
108
+ )
109
+ def test_repetive_call_success(agent_runner: TestRunner):
110
+ _call_function_and_assert(
111
+ agent_runner,
112
+ "repetive_call_1",
113
+ "test",
114
+ "test_repetive",
115
+ )
116
+
117
+
118
+ @pytest.mark.parametrize(
119
+ "agent_runner",
120
+ [{"agent": tool_agent.agent.root_agent}],
121
+ indirect=True,
122
+ )
123
+ def test_function_calls_fail(agent_runner: TestRunner):
124
+ _call_function_and_assert(
125
+ agent_runner,
126
+ "throw_error_function",
127
+ "test",
128
+ None,
129
+ ValueError,
130
+ )
131
+
132
+
133
+ @pytest.mark.parametrize(
134
+ "agent_runner",
135
+ [{"agent": tool_agent.agent.root_agent}],
136
+ indirect=True,
137
+ )
138
+ def test_agent_tools_success(agent_runner: TestRunner):
139
+ _call_function_and_assert(
140
+ agent_runner,
141
+ "no_schema_agent",
142
+ "Hi",
143
+ "Hi",
144
+ )
145
+ _call_function_and_assert(
146
+ agent_runner,
147
+ "schema_agent",
148
+ "Agent_tools",
149
+ "Agent_tools_success",
150
+ )
151
+ _call_function_and_assert(
152
+ agent_runner, "no_input_schema_agent", "Tools", "Tools_success"
153
+ )
154
+ _call_function_and_assert(agent_runner, "no_output_schema_agent", "Hi", "Hi")
155
+
156
+
157
+ @pytest.mark.parametrize(
158
+ "agent_runner",
159
+ [{"agent": tool_agent.agent.root_agent}],
160
+ indirect=True,
161
+ )
162
+ def test_files_retrieval_success(agent_runner: TestRunner):
163
+ _call_function_and_assert(
164
+ agent_runner,
165
+ "test_case_retrieval",
166
+ "What is the testing strategy of agent 2.0?",
167
+ "test",
168
+ )
169
+ # For non relevant query, the agent should still be running fine, just return
170
+ # response might be different for different calls, so we don't compare the
171
+ # response here.
172
+ _call_function_and_assert(
173
+ agent_runner,
174
+ "test_case_retrieval",
175
+ "What is the whether in bay area?",
176
+ "",
177
+ )
178
+
179
+
180
+ @pytest.mark.parametrize(
181
+ "agent_runner",
182
+ [{"agent": tool_agent.agent.root_agent}],
183
+ indirect=True,
184
+ )
185
+ def test_rag_retrieval_success(agent_runner: TestRunner):
186
+ _call_function_and_assert(
187
+ agent_runner,
188
+ "valid_rag_retrieval",
189
+ "What is the testing strategy of agent 2.0?",
190
+ "test",
191
+ )
192
+ _call_function_and_assert(
193
+ agent_runner,
194
+ "valid_rag_retrieval",
195
+ "What is the whether in bay area?",
196
+ "No",
197
+ )
198
+
199
+
200
+ @pytest.mark.parametrize(
201
+ "agent_runner",
202
+ [{"agent": tool_agent.agent.root_agent}],
203
+ indirect=True,
204
+ )
205
+ def test_rag_retrieval_fail(agent_runner: TestRunner):
206
+ _call_function_and_assert(
207
+ agent_runner,
208
+ "invalid_rag_retrieval",
209
+ "What is the testing strategy of agent 2.0?",
210
+ None,
211
+ ValueError,
212
+ )
213
+ _call_function_and_assert(
214
+ agent_runner,
215
+ "non_exist_rag_retrieval",
216
+ "What is the whether in bay area?",
217
+ None,
218
+ ValueError,
219
+ )
220
+
221
+
222
+ @pytest.mark.parametrize(
223
+ "agent_runner",
224
+ [{"agent": tool_agent.agent.root_agent}],
225
+ indirect=True,
226
+ )
227
+ def test_langchain_tool_success(agent_runner: TestRunner):
228
+ _call_function_and_assert(
229
+ agent_runner,
230
+ "terminal",
231
+ "Run the following shell command 'echo test!'",
232
+ "test",
233
+ )
234
+
235
+
236
+ @pytest.mark.parametrize(
237
+ "agent_runner",
238
+ [{"agent": tool_agent.agent.root_agent}],
239
+ indirect=True,
240
+ )
241
+ def test_crewai_tool_success(agent_runner: TestRunner):
242
+ _call_function_and_assert(
243
+ agent_runner,
244
+ "direcotry_read_tool",
245
+ "Find all the file paths",
246
+ "file",
247
+ )
248
+
249
+
250
+ def _call_function_and_assert(
251
+ agent_runner: TestRunner,
252
+ function_name: str,
253
+ params,
254
+ expected=None,
255
+ exception: Exception = None,
256
+ ):
257
+ param_section = (
258
+ " with params"
259
+ f" {params if isinstance(params, str) else json.dumps(params)}"
260
+ if params is not None
261
+ else ""
262
+ )
263
+ query = f"Call {function_name}{param_section} and show me the result"
264
+ if exception:
265
+ _assert_raises(agent_runner, query, exception)
266
+ return
267
+
268
+ _assert_function_output(agent_runner, query, expected)
269
+
270
+
271
+ def _assert_raises(agent_runner: TestRunner, query: str, exception: Exception):
272
+ with pytest.raises(exception):
273
+ agent_runner.run(query)
274
+
275
+
276
+ def _assert_function_output(agent_runner: TestRunner, query: str, expected):
277
+ agent_runner.run(query)
278
+
279
+ # Retrieve the latest model response event
280
+ model_response_event = agent_runner.get_events()[-1]
281
+
282
+ # Assert the response content
283
+ assert model_response_event.content.role == "model"
284
+ assert (
285
+ expected.lower()
286
+ in model_response_event.content.parts[0].text.strip().lower()
287
+ )
@@ -0,0 +1,34 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ from google.adk.evaluation import AgentEvaluator
16
+
17
+
18
+ def test_with_single_test_file():
19
+ """Test the agent's basic ability via session file."""
20
+ AgentEvaluator.evaluate(
21
+ agent_module="tests.integration.fixture.home_automation_agent",
22
+ eval_dataset_file_path_or_dir="tests/integration/fixture/home_automation_agent/simple_test.test.json",
23
+ )
24
+
25
+
26
+ def test_with_folder_of_test_files_long_running():
27
+ """Test the agent's basic ability via a folder of session files."""
28
+ AgentEvaluator.evaluate(
29
+ agent_module="tests.integration.fixture.home_automation_agent",
30
+ eval_dataset_file_path_or_dir=(
31
+ "tests/integration/fixture/home_automation_agent/test_files"
32
+ ),
33
+ num_runs=4,
34
+ )
@@ -0,0 +1,14 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
@@ -0,0 +1,16 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ from .asserts import *
16
+ from .test_runner import TestRunner
@@ -0,0 +1,75 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ from typing import TypedDict
16
+
17
+ from .test_runner import TestRunner
18
+
19
+
20
+ class Message(TypedDict):
21
+ agent_name: str
22
+ expected_text: str
23
+
24
+
25
+ def assert_current_agent_is(agent_name: str, *, agent_runner: TestRunner):
26
+ assert agent_runner.get_current_agent_name() == agent_name
27
+
28
+
29
+ def assert_agent_says(
30
+ expected_text: str, *, agent_name: str, agent_runner: TestRunner
31
+ ):
32
+ for event in reversed(agent_runner.get_events()):
33
+ if event.author == agent_name and event.content.parts[0].text:
34
+ assert event.content.parts[0].text.strip() == expected_text
35
+ return
36
+
37
+
38
+ def assert_agent_says_in_order(
39
+ expected_conversaction: list[Message], agent_runner: TestRunner
40
+ ):
41
+ expected_conversaction_idx = len(expected_conversaction) - 1
42
+ for event in reversed(agent_runner.get_events()):
43
+ if event.content.parts and event.content.parts[0].text:
44
+ assert (
45
+ event.author
46
+ == expected_conversaction[expected_conversaction_idx]['agent_name']
47
+ )
48
+ assert (
49
+ event.content.parts[0].text.strip()
50
+ == expected_conversaction[expected_conversaction_idx]['expected_text']
51
+ )
52
+ expected_conversaction_idx -= 1
53
+ if expected_conversaction_idx < 0:
54
+ return
55
+
56
+
57
+ def assert_agent_transfer_path(
58
+ expected_path: list[str], *, agent_runner: TestRunner
59
+ ):
60
+ events = agent_runner.get_events()
61
+ idx_in_expected_path = len(expected_path) - 1
62
+ # iterate events in reverse order
63
+ for event in reversed(events):
64
+ function_calls = event.get_function_calls()
65
+ if (
66
+ len(function_calls) == 1
67
+ and function_calls[0].name == 'transfer_to_agent'
68
+ ):
69
+ assert (
70
+ function_calls[0].args['agent_name']
71
+ == expected_path[idx_in_expected_path]
72
+ )
73
+ idx_in_expected_path -= 1
74
+ if idx_in_expected_path < 0:
75
+ return
@@ -0,0 +1,97 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+
15
+ import importlib
16
+ from typing import Optional
17
+
18
+ from google.adk import Agent
19
+ from google.adk import Runner
20
+ from google.adk.artifacts import BaseArtifactService
21
+ from google.adk.artifacts import InMemoryArtifactService
22
+ from google.adk.events import Event
23
+ from google.adk.sessions import BaseSessionService
24
+ from google.adk.sessions import InMemorySessionService
25
+ from google.adk.sessions import Session
26
+ from google.genai import types
27
+
28
+
29
+ class TestRunner:
30
+ """Agents runner for testings."""
31
+
32
+ app_name = "test_app"
33
+ user_id = "test_user"
34
+
35
+ def __init__(
36
+ self,
37
+ agent: Agent,
38
+ artifact_service: BaseArtifactService = InMemoryArtifactService(),
39
+ session_service: BaseSessionService = InMemorySessionService(),
40
+ ) -> None:
41
+ self.agent = agent
42
+ self.agent_client = Runner(
43
+ app_name=self.app_name,
44
+ agent=agent,
45
+ artifact_service=artifact_service,
46
+ session_service=session_service,
47
+ )
48
+ self.session_service = session_service
49
+ self.current_session_id = session_service.create_session(
50
+ app_name=self.app_name, user_id=self.user_id
51
+ ).id
52
+
53
+ def new_session(self, session_id: Optional[str] = None) -> None:
54
+ self.current_session_id = self.session_service.create_session(
55
+ app_name=self.app_name, user_id=self.user_id, session_id=session_id
56
+ ).id
57
+
58
+ def run(self, prompt: str) -> list[Event]:
59
+ current_session = self.session_service.get_session(
60
+ app_name=self.app_name,
61
+ user_id=self.user_id,
62
+ session_id=self.current_session_id,
63
+ )
64
+ assert current_session is not None
65
+
66
+ return list(
67
+ self.agent_client.run(
68
+ user_id=current_session.user_id,
69
+ session_id=current_session.id,
70
+ new_message=types.Content(
71
+ role="user",
72
+ parts=[types.Part.from_text(text=prompt)],
73
+ ),
74
+ )
75
+ )
76
+
77
+ def get_current_session(self) -> Optional[Session]:
78
+ return self.session_service.get_session(
79
+ app_name=self.app_name,
80
+ user_id=self.user_id,
81
+ session_id=self.current_session_id,
82
+ )
83
+
84
+ def get_events(self) -> list[Event]:
85
+ return self.get_current_session().events
86
+
87
+ @classmethod
88
+ def from_agent_name(cls, agent_name: str):
89
+ agent_module_path = f"tests.integration.fixture.{agent_name}"
90
+ agent_module = importlib.import_module(agent_module_path)
91
+ agent: Agent = agent_module.agent.root_agent
92
+ return cls(agent)
93
+
94
+ def get_current_agent_name(self) -> str:
95
+ return self.agent_client._find_agent_to_run(
96
+ self.get_current_session(), self.agent
97
+ ).name