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.
- google/adk/agents/run_config.py +4 -0
- google/adk/auth/auth_preprocessor.py +19 -16
- google/adk/cli/browser/index.html +1 -1
- google/adk/cli/browser/{main-SY2WYYGV.js → main-SLIAU2JL.js} +46 -30
- google/adk/cli/cli.py +8 -8
- google/adk/cli/cli_deploy.py +2 -4
- google/adk/cli/cli_tools_click.py +6 -6
- google/adk/flows/llm_flows/base_llm_flow.py +1 -1
- google/adk/flows/llm_flows/contents.py +21 -1
- google/adk/flows/llm_flows/functions.py +3 -1
- google/adk/models/google_llm.py +0 -1
- google/adk/runners.py +13 -2
- google/adk/version.py +1 -1
- {google_adk-0.0.3.dist-info → google_adk-0.0.5.dist-info}/METADATA +4 -2
- google_adk-0.0.5.dist-info/RECORD +175 -0
- {google_adk-0.0.3.dist-info → google_adk-0.0.5.dist-info}/WHEEL +1 -1
- google_adk-0.0.5.dist-info/licenses/LICENSE +202 -0
- google/adk/._version.py +0 -0
- google/adk/docs/Makefile +0 -20
- google/adk/docs/build/doctrees/google-adk.doctree +0 -0
- google/adk/docs/build/html/_sources/google-adk.rst.txt +0 -98
- google/adk/docs/build/html/_sources/index.rst.txt +0 -7
- google/adk/docs/build/html/_static/autodoc_pydantic.css +0 -27
- google/adk/docs/build/html/_static/basic.css +0 -925
- google/adk/docs/build/html/_static/debug.css +0 -85
- google/adk/docs/build/html/_static/doctools.js +0 -156
- google/adk/docs/build/html/_static/documentation_options.js +0 -29
- google/adk/docs/build/html/_static/file.png +0 -0
- google/adk/docs/build/html/_static/language_data.js +0 -199
- google/adk/docs/build/html/_static/minus.png +0 -0
- google/adk/docs/build/html/_static/plus.png +0 -0
- google/adk/docs/build/html/_static/pygments.css +0 -274
- google/adk/docs/build/html/_static/scripts/furo-extensions.js +0 -16
- google/adk/docs/build/html/_static/scripts/furo.js +0 -19
- google/adk/docs/build/html/_static/scripts/furo.js.LICENSE.txt +0 -7
- google/adk/docs/build/html/_static/scripts/furo.js.map +0 -1
- google/adk/docs/build/html/_static/searchtools.js +0 -620
- google/adk/docs/build/html/_static/skeleton.css +0 -312
- google/adk/docs/build/html/_static/sphinx_highlight.js +0 -170
- google/adk/docs/build/html/_static/styles/furo-extensions.css +0 -18
- google/adk/docs/build/html/_static/styles/furo-extensions.css.map +0 -1
- google/adk/docs/build/html/_static/styles/furo.css +0 -18
- google/adk/docs/build/html/_static/styles/furo.css.map +0 -1
- google/adk/docs/build/html/genindex.html +0 -861
- google/adk/docs/build/html/google-adk.html +0 -5461
- google/adk/docs/build/html/index.html +0 -567
- google/adk/docs/build/html/objects.inv +0 -0
- google/adk/docs/build/html/py-modindex.html +0 -373
- google/adk/docs/build/html/search.html +0 -333
- google/adk/docs/build/html/searchindex.js +0 -17
- google/adk/docs/source/conf.py +0 -133
- google/adk/docs/source/google-adk.rst +0 -98
- google/adk/docs/source/index.rst +0 -7
- google/adk/tests/__init__.py +0 -14
- google/adk/tests/integration/.env.example +0 -10
- google/adk/tests/integration/__init__.py +0 -18
- google/adk/tests/integration/conftest.py +0 -119
- google/adk/tests/integration/fixture/__init__.py +0 -14
- google/adk/tests/integration/fixture/agent_with_config/__init__.py +0 -15
- google/adk/tests/integration/fixture/agent_with_config/agent.py +0 -88
- google/adk/tests/integration/fixture/callback_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/callback_agent/agent.py +0 -105
- google/adk/tests/integration/fixture/context_update_test/OWNERS +0 -1
- google/adk/tests/integration/fixture/context_update_test/__init__.py +0 -15
- google/adk/tests/integration/fixture/context_update_test/agent.py +0 -43
- google/adk/tests/integration/fixture/context_update_test/successful_test.session.json +0 -582
- google/adk/tests/integration/fixture/context_variable_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/context_variable_agent/agent.py +0 -115
- google/adk/tests/integration/fixture/customer_support_ma/__init__.py +0 -15
- google/adk/tests/integration/fixture/customer_support_ma/agent.py +0 -172
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/agent.py +0 -338
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/order_query.test.json +0 -69
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/test_config.json +0 -6
- google/adk/tests/integration/fixture/flow_complex_spark/__init__.py +0 -15
- google/adk/tests/integration/fixture/flow_complex_spark/agent.py +0 -182
- google/adk/tests/integration/fixture/flow_complex_spark/sample.session.json +0 -190
- google/adk/tests/integration/fixture/hello_world_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/hello_world_agent/agent.py +0 -95
- google/adk/tests/integration/fixture/hello_world_agent/roll_die.test.json +0 -24
- google/adk/tests/integration/fixture/hello_world_agent/test_config.json +0 -6
- google/adk/tests/integration/fixture/home_automation_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/home_automation_agent/agent.py +0 -304
- google/adk/tests/integration/fixture/home_automation_agent/simple_test.test.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/simple_test2.test.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/test_config.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/test_files/dependent_tool_calls.test.json +0 -18
- google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/eval_data.test.json +0 -17
- google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/test_config.json +0 -6
- google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_multi_turn_conversation.test.json +0 -18
- google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test.test.json +0 -17
- google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test2.test.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/test_files/test_config.json +0 -5
- google/adk/tests/integration/fixture/tool_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/tool_agent/agent.py +0 -218
- google/adk/tests/integration/fixture/tool_agent/files/Agent_test_plan.pdf +0 -0
- google/adk/tests/integration/fixture/trip_planner_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/trip_planner_agent/agent.py +0 -110
- google/adk/tests/integration/fixture/trip_planner_agent/initial.session.json +0 -13
- google/adk/tests/integration/fixture/trip_planner_agent/test_config.json +0 -5
- google/adk/tests/integration/fixture/trip_planner_agent/test_files/initial.session.json +0 -13
- google/adk/tests/integration/fixture/trip_planner_agent/test_files/test_config.json +0 -5
- google/adk/tests/integration/fixture/trip_planner_agent/test_files/trip_inquiry_sub_agent.test.json +0 -7
- google/adk/tests/integration/fixture/trip_planner_agent/trip_inquiry.test.json +0 -19
- google/adk/tests/integration/models/__init__.py +0 -14
- google/adk/tests/integration/models/test_google_llm.py +0 -65
- google/adk/tests/integration/test_callback.py +0 -70
- google/adk/tests/integration/test_context_variable.py +0 -67
- google/adk/tests/integration/test_evalute_agent_in_fixture.py +0 -76
- google/adk/tests/integration/test_multi_agent.py +0 -28
- google/adk/tests/integration/test_multi_turn.py +0 -42
- google/adk/tests/integration/test_single_agent.py +0 -23
- google/adk/tests/integration/test_sub_agent.py +0 -26
- google/adk/tests/integration/test_system_instruction.py +0 -177
- google/adk/tests/integration/test_tools.py +0 -287
- google/adk/tests/integration/test_with_test_file.py +0 -34
- google/adk/tests/integration/tools/__init__.py +0 -14
- google/adk/tests/integration/utils/__init__.py +0 -16
- google/adk/tests/integration/utils/asserts.py +0 -75
- google/adk/tests/integration/utils/test_runner.py +0 -97
- google/adk/tests/unittests/__init__.py +0 -14
- google/adk/tests/unittests/agents/__init__.py +0 -14
- google/adk/tests/unittests/agents/test_base_agent.py +0 -407
- google/adk/tests/unittests/agents/test_langgraph_agent.py +0 -191
- google/adk/tests/unittests/agents/test_llm_agent_callbacks.py +0 -138
- google/adk/tests/unittests/agents/test_llm_agent_fields.py +0 -231
- google/adk/tests/unittests/agents/test_loop_agent.py +0 -136
- google/adk/tests/unittests/agents/test_parallel_agent.py +0 -92
- google/adk/tests/unittests/agents/test_sequential_agent.py +0 -114
- google/adk/tests/unittests/artifacts/__init__.py +0 -14
- google/adk/tests/unittests/artifacts/test_artifact_service.py +0 -276
- google/adk/tests/unittests/auth/test_auth_handler.py +0 -575
- google/adk/tests/unittests/conftest.py +0 -73
- google/adk/tests/unittests/fast_api/__init__.py +0 -14
- google/adk/tests/unittests/fast_api/test_fast_api.py +0 -269
- google/adk/tests/unittests/flows/__init__.py +0 -14
- google/adk/tests/unittests/flows/llm_flows/__init__.py +0 -14
- google/adk/tests/unittests/flows/llm_flows/_test_examples.py +0 -142
- google/adk/tests/unittests/flows/llm_flows/test_agent_transfer.py +0 -311
- google/adk/tests/unittests/flows/llm_flows/test_functions_long_running.py +0 -244
- google/adk/tests/unittests/flows/llm_flows/test_functions_request_euc.py +0 -346
- google/adk/tests/unittests/flows/llm_flows/test_functions_sequential.py +0 -93
- google/adk/tests/unittests/flows/llm_flows/test_functions_simple.py +0 -258
- google/adk/tests/unittests/flows/llm_flows/test_identity.py +0 -66
- google/adk/tests/unittests/flows/llm_flows/test_instructions.py +0 -164
- google/adk/tests/unittests/flows/llm_flows/test_model_callbacks.py +0 -142
- google/adk/tests/unittests/flows/llm_flows/test_other_configs.py +0 -46
- google/adk/tests/unittests/flows/llm_flows/test_tool_callbacks.py +0 -269
- google/adk/tests/unittests/models/__init__.py +0 -14
- google/adk/tests/unittests/models/test_google_llm.py +0 -224
- google/adk/tests/unittests/models/test_litellm.py +0 -804
- google/adk/tests/unittests/models/test_models.py +0 -60
- google/adk/tests/unittests/sessions/__init__.py +0 -14
- google/adk/tests/unittests/sessions/test_session_service.py +0 -227
- google/adk/tests/unittests/sessions/test_vertex_ai_session_service.py +0 -246
- google/adk/tests/unittests/streaming/__init__.py +0 -14
- google/adk/tests/unittests/streaming/test_streaming.py +0 -50
- google/adk/tests/unittests/tools/__init__.py +0 -14
- google/adk/tests/unittests/tools/apihub_tool/clients/test_apihub_client.py +0 -499
- google/adk/tests/unittests/tools/apihub_tool/test_apihub_toolset.py +0 -204
- google/adk/tests/unittests/tools/application_integration_tool/clients/test_connections_client.py +0 -600
- google/adk/tests/unittests/tools/application_integration_tool/clients/test_integration_client.py +0 -630
- google/adk/tests/unittests/tools/application_integration_tool/test_application_integration_toolset.py +0 -345
- google/adk/tests/unittests/tools/google_api_tool/__init__.py +0 -13
- google/adk/tests/unittests/tools/google_api_tool/test_googleapi_to_openapi_converter.py +0 -657
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_auto_auth_credential_exchanger.py +0 -145
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_base_auth_credential_exchanger.py +0 -68
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_oauth2_exchanger.py +0 -153
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_service_account_exchanger.py +0 -196
- google/adk/tests/unittests/tools/openapi_tool/auth/test_auth_helper.py +0 -573
- google/adk/tests/unittests/tools/openapi_tool/common/test_common.py +0 -436
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test.yaml +0 -1367
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_spec_parser.py +0 -628
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_toolset.py +0 -139
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_operation_parser.py +0 -406
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_rest_api_tool.py +0 -966
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_tool_auth_handler.py +0 -201
- google/adk/tests/unittests/tools/retrieval/__init__.py +0 -14
- google/adk/tests/unittests/tools/retrieval/test_vertex_ai_rag_retrieval.py +0 -147
- google/adk/tests/unittests/tools/test_agent_tool.py +0 -167
- google/adk/tests/unittests/tools/test_base_tool.py +0 -141
- google/adk/tests/unittests/tools/test_build_function_declaration.py +0 -277
- google/adk/tests/unittests/utils.py +0 -304
- google_adk-0.0.3.dist-info/RECORD +0 -340
- {google_adk-0.0.3.dist-info → google_adk-0.0.5.dist-info}/entry_points.txt +0 -0
@@ -1,177 +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
|
-
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
|
@@ -1,287 +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
|
-
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
|
-
)
|
@@ -1,34 +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.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
|
-
)
|
@@ -1,14 +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
|
-
|
@@ -1,16 +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 .asserts import *
|
16
|
-
from .test_runner import TestRunner
|
@@ -1,75 +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 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
|
@@ -1,97 +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
|
-
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
|