google-adk 0.0.3__py3-none-any.whl → 0.0.5__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 (185) hide show
  1. google/adk/agents/run_config.py +4 -0
  2. google/adk/auth/auth_preprocessor.py +19 -16
  3. google/adk/cli/browser/index.html +1 -1
  4. google/adk/cli/browser/{main-SY2WYYGV.js → main-SLIAU2JL.js} +46 -30
  5. google/adk/cli/cli.py +8 -8
  6. google/adk/cli/cli_deploy.py +2 -4
  7. google/adk/cli/cli_tools_click.py +6 -6
  8. google/adk/flows/llm_flows/base_llm_flow.py +1 -1
  9. google/adk/flows/llm_flows/contents.py +21 -1
  10. google/adk/flows/llm_flows/functions.py +3 -1
  11. google/adk/models/google_llm.py +0 -1
  12. google/adk/runners.py +13 -2
  13. google/adk/version.py +1 -1
  14. {google_adk-0.0.3.dist-info → google_adk-0.0.5.dist-info}/METADATA +4 -2
  15. google_adk-0.0.5.dist-info/RECORD +175 -0
  16. {google_adk-0.0.3.dist-info → google_adk-0.0.5.dist-info}/WHEEL +1 -1
  17. google_adk-0.0.5.dist-info/licenses/LICENSE +202 -0
  18. google/adk/._version.py +0 -0
  19. google/adk/docs/Makefile +0 -20
  20. google/adk/docs/build/doctrees/google-adk.doctree +0 -0
  21. google/adk/docs/build/html/_sources/google-adk.rst.txt +0 -98
  22. google/adk/docs/build/html/_sources/index.rst.txt +0 -7
  23. google/adk/docs/build/html/_static/autodoc_pydantic.css +0 -27
  24. google/adk/docs/build/html/_static/basic.css +0 -925
  25. google/adk/docs/build/html/_static/debug.css +0 -85
  26. google/adk/docs/build/html/_static/doctools.js +0 -156
  27. google/adk/docs/build/html/_static/documentation_options.js +0 -29
  28. google/adk/docs/build/html/_static/file.png +0 -0
  29. google/adk/docs/build/html/_static/language_data.js +0 -199
  30. google/adk/docs/build/html/_static/minus.png +0 -0
  31. google/adk/docs/build/html/_static/plus.png +0 -0
  32. google/adk/docs/build/html/_static/pygments.css +0 -274
  33. google/adk/docs/build/html/_static/scripts/furo-extensions.js +0 -16
  34. google/adk/docs/build/html/_static/scripts/furo.js +0 -19
  35. google/adk/docs/build/html/_static/scripts/furo.js.LICENSE.txt +0 -7
  36. google/adk/docs/build/html/_static/scripts/furo.js.map +0 -1
  37. google/adk/docs/build/html/_static/searchtools.js +0 -620
  38. google/adk/docs/build/html/_static/skeleton.css +0 -312
  39. google/adk/docs/build/html/_static/sphinx_highlight.js +0 -170
  40. google/adk/docs/build/html/_static/styles/furo-extensions.css +0 -18
  41. google/adk/docs/build/html/_static/styles/furo-extensions.css.map +0 -1
  42. google/adk/docs/build/html/_static/styles/furo.css +0 -18
  43. google/adk/docs/build/html/_static/styles/furo.css.map +0 -1
  44. google/adk/docs/build/html/genindex.html +0 -861
  45. google/adk/docs/build/html/google-adk.html +0 -5461
  46. google/adk/docs/build/html/index.html +0 -567
  47. google/adk/docs/build/html/objects.inv +0 -0
  48. google/adk/docs/build/html/py-modindex.html +0 -373
  49. google/adk/docs/build/html/search.html +0 -333
  50. google/adk/docs/build/html/searchindex.js +0 -17
  51. google/adk/docs/source/conf.py +0 -133
  52. google/adk/docs/source/google-adk.rst +0 -98
  53. google/adk/docs/source/index.rst +0 -7
  54. google/adk/tests/__init__.py +0 -14
  55. google/adk/tests/integration/.env.example +0 -10
  56. google/adk/tests/integration/__init__.py +0 -18
  57. google/adk/tests/integration/conftest.py +0 -119
  58. google/adk/tests/integration/fixture/__init__.py +0 -14
  59. google/adk/tests/integration/fixture/agent_with_config/__init__.py +0 -15
  60. google/adk/tests/integration/fixture/agent_with_config/agent.py +0 -88
  61. google/adk/tests/integration/fixture/callback_agent/__init__.py +0 -15
  62. google/adk/tests/integration/fixture/callback_agent/agent.py +0 -105
  63. google/adk/tests/integration/fixture/context_update_test/OWNERS +0 -1
  64. google/adk/tests/integration/fixture/context_update_test/__init__.py +0 -15
  65. google/adk/tests/integration/fixture/context_update_test/agent.py +0 -43
  66. google/adk/tests/integration/fixture/context_update_test/successful_test.session.json +0 -582
  67. google/adk/tests/integration/fixture/context_variable_agent/__init__.py +0 -15
  68. google/adk/tests/integration/fixture/context_variable_agent/agent.py +0 -115
  69. google/adk/tests/integration/fixture/customer_support_ma/__init__.py +0 -15
  70. google/adk/tests/integration/fixture/customer_support_ma/agent.py +0 -172
  71. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/__init__.py +0 -15
  72. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/agent.py +0 -338
  73. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/order_query.test.json +0 -69
  74. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/test_config.json +0 -6
  75. google/adk/tests/integration/fixture/flow_complex_spark/__init__.py +0 -15
  76. google/adk/tests/integration/fixture/flow_complex_spark/agent.py +0 -182
  77. google/adk/tests/integration/fixture/flow_complex_spark/sample.session.json +0 -190
  78. google/adk/tests/integration/fixture/hello_world_agent/__init__.py +0 -15
  79. google/adk/tests/integration/fixture/hello_world_agent/agent.py +0 -95
  80. google/adk/tests/integration/fixture/hello_world_agent/roll_die.test.json +0 -24
  81. google/adk/tests/integration/fixture/hello_world_agent/test_config.json +0 -6
  82. google/adk/tests/integration/fixture/home_automation_agent/__init__.py +0 -15
  83. google/adk/tests/integration/fixture/home_automation_agent/agent.py +0 -304
  84. google/adk/tests/integration/fixture/home_automation_agent/simple_test.test.json +0 -5
  85. google/adk/tests/integration/fixture/home_automation_agent/simple_test2.test.json +0 -5
  86. google/adk/tests/integration/fixture/home_automation_agent/test_config.json +0 -5
  87. google/adk/tests/integration/fixture/home_automation_agent/test_files/dependent_tool_calls.test.json +0 -18
  88. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/eval_data.test.json +0 -17
  89. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/test_config.json +0 -6
  90. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_multi_turn_conversation.test.json +0 -18
  91. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test.test.json +0 -17
  92. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test2.test.json +0 -5
  93. google/adk/tests/integration/fixture/home_automation_agent/test_files/test_config.json +0 -5
  94. google/adk/tests/integration/fixture/tool_agent/__init__.py +0 -15
  95. google/adk/tests/integration/fixture/tool_agent/agent.py +0 -218
  96. google/adk/tests/integration/fixture/tool_agent/files/Agent_test_plan.pdf +0 -0
  97. google/adk/tests/integration/fixture/trip_planner_agent/__init__.py +0 -15
  98. google/adk/tests/integration/fixture/trip_planner_agent/agent.py +0 -110
  99. google/adk/tests/integration/fixture/trip_planner_agent/initial.session.json +0 -13
  100. google/adk/tests/integration/fixture/trip_planner_agent/test_config.json +0 -5
  101. google/adk/tests/integration/fixture/trip_planner_agent/test_files/initial.session.json +0 -13
  102. google/adk/tests/integration/fixture/trip_planner_agent/test_files/test_config.json +0 -5
  103. google/adk/tests/integration/fixture/trip_planner_agent/test_files/trip_inquiry_sub_agent.test.json +0 -7
  104. google/adk/tests/integration/fixture/trip_planner_agent/trip_inquiry.test.json +0 -19
  105. google/adk/tests/integration/models/__init__.py +0 -14
  106. google/adk/tests/integration/models/test_google_llm.py +0 -65
  107. google/adk/tests/integration/test_callback.py +0 -70
  108. google/adk/tests/integration/test_context_variable.py +0 -67
  109. google/adk/tests/integration/test_evalute_agent_in_fixture.py +0 -76
  110. google/adk/tests/integration/test_multi_agent.py +0 -28
  111. google/adk/tests/integration/test_multi_turn.py +0 -42
  112. google/adk/tests/integration/test_single_agent.py +0 -23
  113. google/adk/tests/integration/test_sub_agent.py +0 -26
  114. google/adk/tests/integration/test_system_instruction.py +0 -177
  115. google/adk/tests/integration/test_tools.py +0 -287
  116. google/adk/tests/integration/test_with_test_file.py +0 -34
  117. google/adk/tests/integration/tools/__init__.py +0 -14
  118. google/adk/tests/integration/utils/__init__.py +0 -16
  119. google/adk/tests/integration/utils/asserts.py +0 -75
  120. google/adk/tests/integration/utils/test_runner.py +0 -97
  121. google/adk/tests/unittests/__init__.py +0 -14
  122. google/adk/tests/unittests/agents/__init__.py +0 -14
  123. google/adk/tests/unittests/agents/test_base_agent.py +0 -407
  124. google/adk/tests/unittests/agents/test_langgraph_agent.py +0 -191
  125. google/adk/tests/unittests/agents/test_llm_agent_callbacks.py +0 -138
  126. google/adk/tests/unittests/agents/test_llm_agent_fields.py +0 -231
  127. google/adk/tests/unittests/agents/test_loop_agent.py +0 -136
  128. google/adk/tests/unittests/agents/test_parallel_agent.py +0 -92
  129. google/adk/tests/unittests/agents/test_sequential_agent.py +0 -114
  130. google/adk/tests/unittests/artifacts/__init__.py +0 -14
  131. google/adk/tests/unittests/artifacts/test_artifact_service.py +0 -276
  132. google/adk/tests/unittests/auth/test_auth_handler.py +0 -575
  133. google/adk/tests/unittests/conftest.py +0 -73
  134. google/adk/tests/unittests/fast_api/__init__.py +0 -14
  135. google/adk/tests/unittests/fast_api/test_fast_api.py +0 -269
  136. google/adk/tests/unittests/flows/__init__.py +0 -14
  137. google/adk/tests/unittests/flows/llm_flows/__init__.py +0 -14
  138. google/adk/tests/unittests/flows/llm_flows/_test_examples.py +0 -142
  139. google/adk/tests/unittests/flows/llm_flows/test_agent_transfer.py +0 -311
  140. google/adk/tests/unittests/flows/llm_flows/test_functions_long_running.py +0 -244
  141. google/adk/tests/unittests/flows/llm_flows/test_functions_request_euc.py +0 -346
  142. google/adk/tests/unittests/flows/llm_flows/test_functions_sequential.py +0 -93
  143. google/adk/tests/unittests/flows/llm_flows/test_functions_simple.py +0 -258
  144. google/adk/tests/unittests/flows/llm_flows/test_identity.py +0 -66
  145. google/adk/tests/unittests/flows/llm_flows/test_instructions.py +0 -164
  146. google/adk/tests/unittests/flows/llm_flows/test_model_callbacks.py +0 -142
  147. google/adk/tests/unittests/flows/llm_flows/test_other_configs.py +0 -46
  148. google/adk/tests/unittests/flows/llm_flows/test_tool_callbacks.py +0 -269
  149. google/adk/tests/unittests/models/__init__.py +0 -14
  150. google/adk/tests/unittests/models/test_google_llm.py +0 -224
  151. google/adk/tests/unittests/models/test_litellm.py +0 -804
  152. google/adk/tests/unittests/models/test_models.py +0 -60
  153. google/adk/tests/unittests/sessions/__init__.py +0 -14
  154. google/adk/tests/unittests/sessions/test_session_service.py +0 -227
  155. google/adk/tests/unittests/sessions/test_vertex_ai_session_service.py +0 -246
  156. google/adk/tests/unittests/streaming/__init__.py +0 -14
  157. google/adk/tests/unittests/streaming/test_streaming.py +0 -50
  158. google/adk/tests/unittests/tools/__init__.py +0 -14
  159. google/adk/tests/unittests/tools/apihub_tool/clients/test_apihub_client.py +0 -499
  160. google/adk/tests/unittests/tools/apihub_tool/test_apihub_toolset.py +0 -204
  161. google/adk/tests/unittests/tools/application_integration_tool/clients/test_connections_client.py +0 -600
  162. google/adk/tests/unittests/tools/application_integration_tool/clients/test_integration_client.py +0 -630
  163. google/adk/tests/unittests/tools/application_integration_tool/test_application_integration_toolset.py +0 -345
  164. google/adk/tests/unittests/tools/google_api_tool/__init__.py +0 -13
  165. google/adk/tests/unittests/tools/google_api_tool/test_googleapi_to_openapi_converter.py +0 -657
  166. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_auto_auth_credential_exchanger.py +0 -145
  167. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_base_auth_credential_exchanger.py +0 -68
  168. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_oauth2_exchanger.py +0 -153
  169. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_service_account_exchanger.py +0 -196
  170. google/adk/tests/unittests/tools/openapi_tool/auth/test_auth_helper.py +0 -573
  171. google/adk/tests/unittests/tools/openapi_tool/common/test_common.py +0 -436
  172. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test.yaml +0 -1367
  173. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_spec_parser.py +0 -628
  174. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_toolset.py +0 -139
  175. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_operation_parser.py +0 -406
  176. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_rest_api_tool.py +0 -966
  177. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_tool_auth_handler.py +0 -201
  178. google/adk/tests/unittests/tools/retrieval/__init__.py +0 -14
  179. google/adk/tests/unittests/tools/retrieval/test_vertex_ai_rag_retrieval.py +0 -147
  180. google/adk/tests/unittests/tools/test_agent_tool.py +0 -167
  181. google/adk/tests/unittests/tools/test_base_tool.py +0 -141
  182. google/adk/tests/unittests/tools/test_build_function_declaration.py +0 -277
  183. google/adk/tests/unittests/utils.py +0 -304
  184. google_adk-0.0.3.dist-info/RECORD +0 -340
  185. {google_adk-0.0.3.dist-info → google_adk-0.0.5.dist-info}/entry_points.txt +0 -0
@@ -1,311 +0,0 @@
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.agents.llm_agent import Agent
16
- from google.adk.agents.loop_agent import LoopAgent
17
- from google.adk.agents.sequential_agent import SequentialAgent
18
- from google.adk.tools import exit_loop
19
- from google.genai.types import Part
20
-
21
- from ... import utils
22
-
23
-
24
- def transfer_call_part(agent_name: str) -> Part:
25
- return Part.from_function_call(
26
- name='transfer_to_agent', args={'agent_name': agent_name}
27
- )
28
-
29
-
30
- TRANSFER_RESPONSE_PART = Part.from_function_response(
31
- name='transfer_to_agent', response={}
32
- )
33
-
34
-
35
- def test_auto_to_auto():
36
- response = [
37
- transfer_call_part('sub_agent_1'),
38
- 'response1',
39
- 'response2',
40
- ]
41
- mockModel = utils.MockModel.create(responses=response)
42
- # root (auto) - sub_agent_1 (auto)
43
- sub_agent_1 = Agent(name='sub_agent_1', model=mockModel)
44
- root_agent = Agent(
45
- name='root_agent',
46
- model=mockModel,
47
- sub_agents=[sub_agent_1],
48
- )
49
-
50
- runner = utils.InMemoryRunner(root_agent)
51
-
52
- # Asserts the transfer.
53
- assert utils.simplify_events(runner.run('test1')) == [
54
- ('root_agent', transfer_call_part('sub_agent_1')),
55
- ('root_agent', TRANSFER_RESPONSE_PART),
56
- ('sub_agent_1', 'response1'),
57
- ]
58
-
59
- # sub_agent_1 should still be the current agent.
60
- assert utils.simplify_events(runner.run('test2')) == [
61
- ('sub_agent_1', 'response2'),
62
- ]
63
-
64
-
65
- def test_auto_to_single():
66
- response = [
67
- transfer_call_part('sub_agent_1'),
68
- 'response1',
69
- 'response2',
70
- ]
71
- mockModel = utils.MockModel.create(responses=response)
72
- # root (auto) - sub_agent_1 (single)
73
- sub_agent_1 = Agent(
74
- name='sub_agent_1',
75
- model=mockModel,
76
- disallow_transfer_to_parent=True,
77
- disallow_transfer_to_peers=True,
78
- )
79
- root_agent = Agent(
80
- name='root_agent', model=mockModel, sub_agents=[sub_agent_1]
81
- )
82
-
83
- runner = utils.InMemoryRunner(root_agent)
84
-
85
- # Asserts the responses.
86
- assert utils.simplify_events(runner.run('test1')) == [
87
- ('root_agent', transfer_call_part('sub_agent_1')),
88
- ('root_agent', TRANSFER_RESPONSE_PART),
89
- ('sub_agent_1', 'response1'),
90
- ]
91
-
92
- # root_agent should still be the current agent, becaues sub_agent_1 is single.
93
- assert utils.simplify_events(runner.run('test2')) == [
94
- ('root_agent', 'response2'),
95
- ]
96
-
97
-
98
- def test_auto_to_auto_to_single():
99
- response = [
100
- transfer_call_part('sub_agent_1'),
101
- # sub_agent_1 transfers to sub_agent_1_1.
102
- transfer_call_part('sub_agent_1_1'),
103
- 'response1',
104
- 'response2',
105
- ]
106
- mockModel = utils.MockModel.create(responses=response)
107
- # root (auto) - sub_agent_1 (auto) - sub_agent_1_1 (single)
108
- sub_agent_1_1 = Agent(
109
- name='sub_agent_1_1',
110
- model=mockModel,
111
- disallow_transfer_to_parent=True,
112
- disallow_transfer_to_peers=True,
113
- )
114
- sub_agent_1 = Agent(
115
- name='sub_agent_1', model=mockModel, sub_agents=[sub_agent_1_1]
116
- )
117
- root_agent = Agent(
118
- name='root_agent', model=mockModel, sub_agents=[sub_agent_1]
119
- )
120
-
121
- runner = utils.InMemoryRunner(root_agent)
122
-
123
- # Asserts the responses.
124
- assert utils.simplify_events(runner.run('test1')) == [
125
- ('root_agent', transfer_call_part('sub_agent_1')),
126
- ('root_agent', TRANSFER_RESPONSE_PART),
127
- ('sub_agent_1', transfer_call_part('sub_agent_1_1')),
128
- ('sub_agent_1', TRANSFER_RESPONSE_PART),
129
- ('sub_agent_1_1', 'response1'),
130
- ]
131
-
132
- # sub_agent_1 should still be the current agent. sub_agent_1_1 is single so it should
133
- # not be the current agent, otherwise the conversation will be tied to
134
- # sub_agent_1_1 forever.
135
- assert utils.simplify_events(runner.run('test2')) == [
136
- ('sub_agent_1', 'response2'),
137
- ]
138
-
139
-
140
- def test_auto_to_sequential():
141
- response = [
142
- transfer_call_part('sub_agent_1'),
143
- # sub_agent_1 responds directly instead of transfering.
144
- 'response1',
145
- 'response2',
146
- 'response3',
147
- ]
148
- mockModel = utils.MockModel.create(responses=response)
149
- # root (auto) - sub_agent_1 (sequential) - sub_agent_1_1 (single)
150
- # \ sub_agent_1_2 (single)
151
- sub_agent_1_1 = Agent(
152
- name='sub_agent_1_1',
153
- model=mockModel,
154
- disallow_transfer_to_parent=True,
155
- disallow_transfer_to_peers=True,
156
- )
157
- sub_agent_1_2 = Agent(
158
- name='sub_agent_1_2',
159
- model=mockModel,
160
- disallow_transfer_to_parent=True,
161
- disallow_transfer_to_peers=True,
162
- )
163
- sub_agent_1 = SequentialAgent(
164
- name='sub_agent_1',
165
- sub_agents=[sub_agent_1_1, sub_agent_1_2],
166
- )
167
- root_agent = Agent(
168
- name='root_agent',
169
- model=mockModel,
170
- sub_agents=[sub_agent_1],
171
- )
172
-
173
- runner = utils.InMemoryRunner(root_agent)
174
-
175
- # Asserts the transfer.
176
- assert utils.simplify_events(runner.run('test1')) == [
177
- ('root_agent', transfer_call_part('sub_agent_1')),
178
- ('root_agent', TRANSFER_RESPONSE_PART),
179
- ('sub_agent_1_1', 'response1'),
180
- ('sub_agent_1_2', 'response2'),
181
- ]
182
-
183
- # root_agent should still be the current agent because sub_agent_1 is sequential.
184
- assert utils.simplify_events(runner.run('test2')) == [
185
- ('root_agent', 'response3'),
186
- ]
187
-
188
-
189
- def test_auto_to_sequential_to_auto():
190
- response = [
191
- transfer_call_part('sub_agent_1'),
192
- # sub_agent_1 responds directly instead of transfering.
193
- 'response1',
194
- transfer_call_part('sub_agent_1_2_1'),
195
- 'response2',
196
- 'response3',
197
- 'response4',
198
- ]
199
- mockModel = utils.MockModel.create(responses=response)
200
- # root (auto) - sub_agent_1 (seq) - sub_agent_1_1 (single)
201
- # \ sub_agent_1_2 (auto) - sub_agent_1_2_1 (auto)
202
- # \ sub_agent_1_3 (single)
203
- sub_agent_1_1 = Agent(
204
- name='sub_agent_1_1',
205
- model=mockModel,
206
- disallow_transfer_to_parent=True,
207
- disallow_transfer_to_peers=True,
208
- )
209
- sub_agent_1_2_1 = Agent(name='sub_agent_1_2_1', model=mockModel)
210
- sub_agent_1_2 = Agent(
211
- name='sub_agent_1_2',
212
- model=mockModel,
213
- sub_agents=[sub_agent_1_2_1],
214
- )
215
- sub_agent_1_3 = Agent(
216
- name='sub_agent_1_3',
217
- model=mockModel,
218
- disallow_transfer_to_parent=True,
219
- disallow_transfer_to_peers=True,
220
- )
221
- sub_agent_1 = SequentialAgent(
222
- name='sub_agent_1',
223
- sub_agents=[sub_agent_1_1, sub_agent_1_2, sub_agent_1_3],
224
- )
225
- root_agent = Agent(
226
- name='root_agent',
227
- model=mockModel,
228
- sub_agents=[sub_agent_1],
229
- )
230
-
231
- runner = utils.InMemoryRunner(root_agent)
232
-
233
- # Asserts the transfer.
234
- assert utils.simplify_events(runner.run('test1')) == [
235
- ('root_agent', transfer_call_part('sub_agent_1')),
236
- ('root_agent', TRANSFER_RESPONSE_PART),
237
- ('sub_agent_1_1', 'response1'),
238
- ('sub_agent_1_2', transfer_call_part('sub_agent_1_2_1')),
239
- ('sub_agent_1_2', TRANSFER_RESPONSE_PART),
240
- ('sub_agent_1_2_1', 'response2'),
241
- ('sub_agent_1_3', 'response3'),
242
- ]
243
-
244
- # root_agent should still be the current agent because sub_agent_1 is sequential.
245
- assert utils.simplify_events(runner.run('test2')) == [
246
- ('root_agent', 'response4'),
247
- ]
248
-
249
-
250
- def test_auto_to_loop():
251
- response = [
252
- transfer_call_part('sub_agent_1'),
253
- # sub_agent_1 responds directly instead of transfering.
254
- 'response1',
255
- 'response2',
256
- 'response3',
257
- Part.from_function_call(name='exit_loop', args={}),
258
- 'response4',
259
- 'response5',
260
- ]
261
- mockModel = utils.MockModel.create(responses=response)
262
- # root (auto) - sub_agent_1 (loop) - sub_agent_1_1 (single)
263
- # \ sub_agent_1_2 (single)
264
- sub_agent_1_1 = Agent(
265
- name='sub_agent_1_1',
266
- model=mockModel,
267
- disallow_transfer_to_parent=True,
268
- disallow_transfer_to_peers=True,
269
- )
270
- sub_agent_1_2 = Agent(
271
- name='sub_agent_1_2',
272
- model=mockModel,
273
- disallow_transfer_to_parent=True,
274
- disallow_transfer_to_peers=True,
275
- tools=[exit_loop],
276
- )
277
- sub_agent_1 = LoopAgent(
278
- name='sub_agent_1',
279
- sub_agents=[sub_agent_1_1, sub_agent_1_2],
280
- )
281
- root_agent = Agent(
282
- name='root_agent',
283
- model=mockModel,
284
- sub_agents=[sub_agent_1],
285
- )
286
-
287
- runner = utils.InMemoryRunner(root_agent)
288
-
289
- # Asserts the transfer.
290
- assert utils.simplify_events(runner.run('test1')) == [
291
- # Transfers to sub_agent_1.
292
- ('root_agent', transfer_call_part('sub_agent_1')),
293
- ('root_agent', TRANSFER_RESPONSE_PART),
294
- # Loops.
295
- ('sub_agent_1_1', 'response1'),
296
- ('sub_agent_1_2', 'response2'),
297
- ('sub_agent_1_1', 'response3'),
298
- # Exits.
299
- ('sub_agent_1_2', Part.from_function_call(name='exit_loop', args={})),
300
- (
301
- 'sub_agent_1_2',
302
- Part.from_function_response(name='exit_loop', response={}),
303
- ),
304
- # root_agent summarizes.
305
- ('root_agent', 'response4'),
306
- ]
307
-
308
- # root_agent should still be the current agent because sub_agent_1 is loop.
309
- assert utils.simplify_events(runner.run('test2')) == [
310
- ('root_agent', 'response5'),
311
- ]
@@ -1,244 +0,0 @@
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.agents import Agent
16
- from google.adk.tools import ToolContext
17
- from google.adk.tools.long_running_tool import LongRunningFunctionTool
18
- from google.genai.types import Part
19
-
20
- from ... import utils
21
-
22
-
23
- def test_async_function():
24
- responses = [
25
- Part.from_function_call(name='increase_by_one', args={'x': 1}),
26
- 'response1',
27
- 'response2',
28
- 'response3',
29
- 'response4',
30
- ]
31
- mockModel = utils.MockModel.create(responses=responses)
32
- function_called = 0
33
-
34
- def increase_by_one(x: int, tool_context: ToolContext) -> int:
35
- nonlocal function_called
36
-
37
- function_called += 1
38
- return {'status': 'pending'}
39
-
40
- # Calls the first time.
41
- agent = Agent(
42
- name='root_agent',
43
- model=mockModel,
44
- tools=[LongRunningFunctionTool(func=increase_by_one)],
45
- )
46
- runner = utils.InMemoryRunner(agent)
47
- events = runner.run('test1')
48
-
49
- # Asserts the requests.
50
- assert len(mockModel.requests) == 2
51
- # 1 item: user content
52
- assert mockModel.requests[0].contents == [
53
- utils.UserContent('test1'),
54
- ]
55
- increase_by_one_call = Part.from_function_call(
56
- name='increase_by_one', args={'x': 1}
57
- )
58
- pending_response = Part.from_function_response(
59
- name='increase_by_one', response={'status': 'pending'}
60
- )
61
-
62
- assert utils.simplify_contents(mockModel.requests[1].contents) == [
63
- ('user', 'test1'),
64
- ('model', increase_by_one_call),
65
- ('user', pending_response),
66
- ]
67
-
68
- # Asserts the function calls.
69
- assert function_called == 1
70
-
71
- # Asserts the responses.
72
- assert utils.simplify_events(events) == [
73
- (
74
- 'root_agent',
75
- Part.from_function_call(name='increase_by_one', args={'x': 1}),
76
- ),
77
- (
78
- 'root_agent',
79
- Part.from_function_response(
80
- name='increase_by_one', response={'status': 'pending'}
81
- ),
82
- ),
83
- ('root_agent', 'response1'),
84
- ]
85
- assert events[0].long_running_tool_ids
86
-
87
- # Updates with another pending progress.
88
- still_waiting_response = Part.from_function_response(
89
- name='increase_by_one', response={'status': 'still waiting'}
90
- )
91
- events = runner.run(utils.UserContent(still_waiting_response))
92
- # We have one new request.
93
- assert len(mockModel.requests) == 3
94
- assert utils.simplify_contents(mockModel.requests[2].contents) == [
95
- ('user', 'test1'),
96
- ('model', increase_by_one_call),
97
- ('user', still_waiting_response),
98
- ]
99
-
100
- assert utils.simplify_events(events) == [('root_agent', 'response2')]
101
-
102
- # Calls when the result is ready.
103
- result_response = Part.from_function_response(
104
- name='increase_by_one', response={'result': 2}
105
- )
106
- events = runner.run(utils.UserContent(result_response))
107
- # We have one new request.
108
- assert len(mockModel.requests) == 4
109
- assert utils.simplify_contents(mockModel.requests[3].contents) == [
110
- ('user', 'test1'),
111
- ('model', increase_by_one_call),
112
- ('user', result_response),
113
- ]
114
- assert utils.simplify_events(events) == [('root_agent', 'response3')]
115
-
116
- # Calls when the result is ready. Here we still accept the result and do
117
- # another summarization. Whether this is the right behavior is TBD.
118
- another_result_response = Part.from_function_response(
119
- name='increase_by_one', response={'result': 3}
120
- )
121
- events = runner.run(utils.UserContent(another_result_response))
122
- # We have one new request.
123
- assert len(mockModel.requests) == 5
124
- assert utils.simplify_contents(mockModel.requests[4].contents) == [
125
- ('user', 'test1'),
126
- ('model', increase_by_one_call),
127
- ('user', another_result_response),
128
- ]
129
- assert utils.simplify_events(events) == [('root_agent', 'response4')]
130
-
131
- # At the end, function_called should still be 1.
132
- assert function_called == 1
133
-
134
-
135
- def test_async_function_with_none_response():
136
- responses = [
137
- Part.from_function_call(name='increase_by_one', args={'x': 1}),
138
- 'response1',
139
- 'response2',
140
- 'response3',
141
- 'response4',
142
- ]
143
- mockModel = utils.MockModel.create(responses=responses)
144
- function_called = 0
145
-
146
- def increase_by_one(x: int, tool_context: ToolContext) -> int:
147
- nonlocal function_called
148
- function_called += 1
149
- return 'pending'
150
-
151
- # Calls the first time.
152
- agent = Agent(
153
- name='root_agent',
154
- model=mockModel,
155
- tools=[LongRunningFunctionTool(func=increase_by_one)],
156
- )
157
- runner = utils.InMemoryRunner(agent)
158
- events = runner.run('test1')
159
-
160
- # Asserts the requests.
161
- assert len(mockModel.requests) == 2
162
- # 1 item: user content
163
- assert mockModel.requests[0].contents == [
164
- utils.UserContent('test1'),
165
- ]
166
- increase_by_one_call = Part.from_function_call(
167
- name='increase_by_one', args={'x': 1}
168
- )
169
-
170
- assert utils.simplify_contents(mockModel.requests[1].contents) == [
171
- ('user', 'test1'),
172
- ('model', increase_by_one_call),
173
- (
174
- 'user',
175
- Part.from_function_response(
176
- name='increase_by_one', response={'result': 'pending'}
177
- ),
178
- ),
179
- ]
180
-
181
- # Asserts the function calls.
182
- assert function_called == 1
183
-
184
- # Asserts the responses.
185
- assert utils.simplify_events(events) == [
186
- (
187
- 'root_agent',
188
- Part.from_function_call(name='increase_by_one', args={'x': 1}),
189
- ),
190
- (
191
- 'root_agent',
192
- Part.from_function_response(
193
- name='increase_by_one', response={'result': 'pending'}
194
- ),
195
- ),
196
- ('root_agent', 'response1'),
197
- ]
198
-
199
- # Updates with another pending progress.
200
- still_waiting_response = Part.from_function_response(
201
- name='increase_by_one', response={'status': 'still waiting'}
202
- )
203
- events = runner.run(utils.UserContent(still_waiting_response))
204
- # We have one new request.
205
- assert len(mockModel.requests) == 3
206
- assert utils.simplify_contents(mockModel.requests[2].contents) == [
207
- ('user', 'test1'),
208
- ('model', increase_by_one_call),
209
- ('user', still_waiting_response),
210
- ]
211
-
212
- assert utils.simplify_events(events) == [('root_agent', 'response2')]
213
-
214
- # Calls when the result is ready.
215
- result_response = Part.from_function_response(
216
- name='increase_by_one', response={'result': 2}
217
- )
218
- events = runner.run(utils.UserContent(result_response))
219
- # We have one new request.
220
- assert len(mockModel.requests) == 4
221
- assert utils.simplify_contents(mockModel.requests[3].contents) == [
222
- ('user', 'test1'),
223
- ('model', increase_by_one_call),
224
- ('user', result_response),
225
- ]
226
- assert utils.simplify_events(events) == [('root_agent', 'response3')]
227
-
228
- # Calls when the result is ready. Here we still accept the result and do
229
- # another summarization. Whether this is the right behavior is TBD.
230
- another_result_response = Part.from_function_response(
231
- name='increase_by_one', response={'result': 3}
232
- )
233
- events = runner.run(utils.UserContent(another_result_response))
234
- # We have one new request.
235
- assert len(mockModel.requests) == 5
236
- assert utils.simplify_contents(mockModel.requests[4].contents) == [
237
- ('user', 'test1'),
238
- ('model', increase_by_one_call),
239
- ('user', another_result_response),
240
- ]
241
- assert utils.simplify_events(events) == [('root_agent', 'response4')]
242
-
243
- # At the end, function_called should still be 1.
244
- assert function_called == 1