google-adk 0.0.3__py3-none-any.whl → 0.0.4__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 (184) 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-CU22TRPI.js} +30 -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/contents.py +21 -1
  9. google/adk/flows/llm_flows/functions.py +3 -1
  10. google/adk/models/google_llm.py +0 -1
  11. google/adk/runners.py +13 -2
  12. google/adk/version.py +1 -1
  13. {google_adk-0.0.3.dist-info → google_adk-0.0.4.dist-info}/METADATA +4 -2
  14. google_adk-0.0.4.dist-info/RECORD +175 -0
  15. {google_adk-0.0.3.dist-info → google_adk-0.0.4.dist-info}/WHEEL +1 -1
  16. google_adk-0.0.4.dist-info/licenses/LICENSE +202 -0
  17. google/adk/._version.py +0 -0
  18. google/adk/docs/Makefile +0 -20
  19. google/adk/docs/build/doctrees/google-adk.doctree +0 -0
  20. google/adk/docs/build/html/_sources/google-adk.rst.txt +0 -98
  21. google/adk/docs/build/html/_sources/index.rst.txt +0 -7
  22. google/adk/docs/build/html/_static/autodoc_pydantic.css +0 -27
  23. google/adk/docs/build/html/_static/basic.css +0 -925
  24. google/adk/docs/build/html/_static/debug.css +0 -85
  25. google/adk/docs/build/html/_static/doctools.js +0 -156
  26. google/adk/docs/build/html/_static/documentation_options.js +0 -29
  27. google/adk/docs/build/html/_static/file.png +0 -0
  28. google/adk/docs/build/html/_static/language_data.js +0 -199
  29. google/adk/docs/build/html/_static/minus.png +0 -0
  30. google/adk/docs/build/html/_static/plus.png +0 -0
  31. google/adk/docs/build/html/_static/pygments.css +0 -274
  32. google/adk/docs/build/html/_static/scripts/furo-extensions.js +0 -16
  33. google/adk/docs/build/html/_static/scripts/furo.js +0 -19
  34. google/adk/docs/build/html/_static/scripts/furo.js.LICENSE.txt +0 -7
  35. google/adk/docs/build/html/_static/scripts/furo.js.map +0 -1
  36. google/adk/docs/build/html/_static/searchtools.js +0 -620
  37. google/adk/docs/build/html/_static/skeleton.css +0 -312
  38. google/adk/docs/build/html/_static/sphinx_highlight.js +0 -170
  39. google/adk/docs/build/html/_static/styles/furo-extensions.css +0 -18
  40. google/adk/docs/build/html/_static/styles/furo-extensions.css.map +0 -1
  41. google/adk/docs/build/html/_static/styles/furo.css +0 -18
  42. google/adk/docs/build/html/_static/styles/furo.css.map +0 -1
  43. google/adk/docs/build/html/genindex.html +0 -861
  44. google/adk/docs/build/html/google-adk.html +0 -5461
  45. google/adk/docs/build/html/index.html +0 -567
  46. google/adk/docs/build/html/objects.inv +0 -0
  47. google/adk/docs/build/html/py-modindex.html +0 -373
  48. google/adk/docs/build/html/search.html +0 -333
  49. google/adk/docs/build/html/searchindex.js +0 -17
  50. google/adk/docs/source/conf.py +0 -133
  51. google/adk/docs/source/google-adk.rst +0 -98
  52. google/adk/docs/source/index.rst +0 -7
  53. google/adk/tests/__init__.py +0 -14
  54. google/adk/tests/integration/.env.example +0 -10
  55. google/adk/tests/integration/__init__.py +0 -18
  56. google/adk/tests/integration/conftest.py +0 -119
  57. google/adk/tests/integration/fixture/__init__.py +0 -14
  58. google/adk/tests/integration/fixture/agent_with_config/__init__.py +0 -15
  59. google/adk/tests/integration/fixture/agent_with_config/agent.py +0 -88
  60. google/adk/tests/integration/fixture/callback_agent/__init__.py +0 -15
  61. google/adk/tests/integration/fixture/callback_agent/agent.py +0 -105
  62. google/adk/tests/integration/fixture/context_update_test/OWNERS +0 -1
  63. google/adk/tests/integration/fixture/context_update_test/__init__.py +0 -15
  64. google/adk/tests/integration/fixture/context_update_test/agent.py +0 -43
  65. google/adk/tests/integration/fixture/context_update_test/successful_test.session.json +0 -582
  66. google/adk/tests/integration/fixture/context_variable_agent/__init__.py +0 -15
  67. google/adk/tests/integration/fixture/context_variable_agent/agent.py +0 -115
  68. google/adk/tests/integration/fixture/customer_support_ma/__init__.py +0 -15
  69. google/adk/tests/integration/fixture/customer_support_ma/agent.py +0 -172
  70. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/__init__.py +0 -15
  71. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/agent.py +0 -338
  72. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/order_query.test.json +0 -69
  73. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/test_config.json +0 -6
  74. google/adk/tests/integration/fixture/flow_complex_spark/__init__.py +0 -15
  75. google/adk/tests/integration/fixture/flow_complex_spark/agent.py +0 -182
  76. google/adk/tests/integration/fixture/flow_complex_spark/sample.session.json +0 -190
  77. google/adk/tests/integration/fixture/hello_world_agent/__init__.py +0 -15
  78. google/adk/tests/integration/fixture/hello_world_agent/agent.py +0 -95
  79. google/adk/tests/integration/fixture/hello_world_agent/roll_die.test.json +0 -24
  80. google/adk/tests/integration/fixture/hello_world_agent/test_config.json +0 -6
  81. google/adk/tests/integration/fixture/home_automation_agent/__init__.py +0 -15
  82. google/adk/tests/integration/fixture/home_automation_agent/agent.py +0 -304
  83. google/adk/tests/integration/fixture/home_automation_agent/simple_test.test.json +0 -5
  84. google/adk/tests/integration/fixture/home_automation_agent/simple_test2.test.json +0 -5
  85. google/adk/tests/integration/fixture/home_automation_agent/test_config.json +0 -5
  86. google/adk/tests/integration/fixture/home_automation_agent/test_files/dependent_tool_calls.test.json +0 -18
  87. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/eval_data.test.json +0 -17
  88. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/test_config.json +0 -6
  89. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_multi_turn_conversation.test.json +0 -18
  90. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test.test.json +0 -17
  91. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test2.test.json +0 -5
  92. google/adk/tests/integration/fixture/home_automation_agent/test_files/test_config.json +0 -5
  93. google/adk/tests/integration/fixture/tool_agent/__init__.py +0 -15
  94. google/adk/tests/integration/fixture/tool_agent/agent.py +0 -218
  95. google/adk/tests/integration/fixture/tool_agent/files/Agent_test_plan.pdf +0 -0
  96. google/adk/tests/integration/fixture/trip_planner_agent/__init__.py +0 -15
  97. google/adk/tests/integration/fixture/trip_planner_agent/agent.py +0 -110
  98. google/adk/tests/integration/fixture/trip_planner_agent/initial.session.json +0 -13
  99. google/adk/tests/integration/fixture/trip_planner_agent/test_config.json +0 -5
  100. google/adk/tests/integration/fixture/trip_planner_agent/test_files/initial.session.json +0 -13
  101. google/adk/tests/integration/fixture/trip_planner_agent/test_files/test_config.json +0 -5
  102. google/adk/tests/integration/fixture/trip_planner_agent/test_files/trip_inquiry_sub_agent.test.json +0 -7
  103. google/adk/tests/integration/fixture/trip_planner_agent/trip_inquiry.test.json +0 -19
  104. google/adk/tests/integration/models/__init__.py +0 -14
  105. google/adk/tests/integration/models/test_google_llm.py +0 -65
  106. google/adk/tests/integration/test_callback.py +0 -70
  107. google/adk/tests/integration/test_context_variable.py +0 -67
  108. google/adk/tests/integration/test_evalute_agent_in_fixture.py +0 -76
  109. google/adk/tests/integration/test_multi_agent.py +0 -28
  110. google/adk/tests/integration/test_multi_turn.py +0 -42
  111. google/adk/tests/integration/test_single_agent.py +0 -23
  112. google/adk/tests/integration/test_sub_agent.py +0 -26
  113. google/adk/tests/integration/test_system_instruction.py +0 -177
  114. google/adk/tests/integration/test_tools.py +0 -287
  115. google/adk/tests/integration/test_with_test_file.py +0 -34
  116. google/adk/tests/integration/tools/__init__.py +0 -14
  117. google/adk/tests/integration/utils/__init__.py +0 -16
  118. google/adk/tests/integration/utils/asserts.py +0 -75
  119. google/adk/tests/integration/utils/test_runner.py +0 -97
  120. google/adk/tests/unittests/__init__.py +0 -14
  121. google/adk/tests/unittests/agents/__init__.py +0 -14
  122. google/adk/tests/unittests/agents/test_base_agent.py +0 -407
  123. google/adk/tests/unittests/agents/test_langgraph_agent.py +0 -191
  124. google/adk/tests/unittests/agents/test_llm_agent_callbacks.py +0 -138
  125. google/adk/tests/unittests/agents/test_llm_agent_fields.py +0 -231
  126. google/adk/tests/unittests/agents/test_loop_agent.py +0 -136
  127. google/adk/tests/unittests/agents/test_parallel_agent.py +0 -92
  128. google/adk/tests/unittests/agents/test_sequential_agent.py +0 -114
  129. google/adk/tests/unittests/artifacts/__init__.py +0 -14
  130. google/adk/tests/unittests/artifacts/test_artifact_service.py +0 -276
  131. google/adk/tests/unittests/auth/test_auth_handler.py +0 -575
  132. google/adk/tests/unittests/conftest.py +0 -73
  133. google/adk/tests/unittests/fast_api/__init__.py +0 -14
  134. google/adk/tests/unittests/fast_api/test_fast_api.py +0 -269
  135. google/adk/tests/unittests/flows/__init__.py +0 -14
  136. google/adk/tests/unittests/flows/llm_flows/__init__.py +0 -14
  137. google/adk/tests/unittests/flows/llm_flows/_test_examples.py +0 -142
  138. google/adk/tests/unittests/flows/llm_flows/test_agent_transfer.py +0 -311
  139. google/adk/tests/unittests/flows/llm_flows/test_functions_long_running.py +0 -244
  140. google/adk/tests/unittests/flows/llm_flows/test_functions_request_euc.py +0 -346
  141. google/adk/tests/unittests/flows/llm_flows/test_functions_sequential.py +0 -93
  142. google/adk/tests/unittests/flows/llm_flows/test_functions_simple.py +0 -258
  143. google/adk/tests/unittests/flows/llm_flows/test_identity.py +0 -66
  144. google/adk/tests/unittests/flows/llm_flows/test_instructions.py +0 -164
  145. google/adk/tests/unittests/flows/llm_flows/test_model_callbacks.py +0 -142
  146. google/adk/tests/unittests/flows/llm_flows/test_other_configs.py +0 -46
  147. google/adk/tests/unittests/flows/llm_flows/test_tool_callbacks.py +0 -269
  148. google/adk/tests/unittests/models/__init__.py +0 -14
  149. google/adk/tests/unittests/models/test_google_llm.py +0 -224
  150. google/adk/tests/unittests/models/test_litellm.py +0 -804
  151. google/adk/tests/unittests/models/test_models.py +0 -60
  152. google/adk/tests/unittests/sessions/__init__.py +0 -14
  153. google/adk/tests/unittests/sessions/test_session_service.py +0 -227
  154. google/adk/tests/unittests/sessions/test_vertex_ai_session_service.py +0 -246
  155. google/adk/tests/unittests/streaming/__init__.py +0 -14
  156. google/adk/tests/unittests/streaming/test_streaming.py +0 -50
  157. google/adk/tests/unittests/tools/__init__.py +0 -14
  158. google/adk/tests/unittests/tools/apihub_tool/clients/test_apihub_client.py +0 -499
  159. google/adk/tests/unittests/tools/apihub_tool/test_apihub_toolset.py +0 -204
  160. google/adk/tests/unittests/tools/application_integration_tool/clients/test_connections_client.py +0 -600
  161. google/adk/tests/unittests/tools/application_integration_tool/clients/test_integration_client.py +0 -630
  162. google/adk/tests/unittests/tools/application_integration_tool/test_application_integration_toolset.py +0 -345
  163. google/adk/tests/unittests/tools/google_api_tool/__init__.py +0 -13
  164. google/adk/tests/unittests/tools/google_api_tool/test_googleapi_to_openapi_converter.py +0 -657
  165. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_auto_auth_credential_exchanger.py +0 -145
  166. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_base_auth_credential_exchanger.py +0 -68
  167. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_oauth2_exchanger.py +0 -153
  168. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_service_account_exchanger.py +0 -196
  169. google/adk/tests/unittests/tools/openapi_tool/auth/test_auth_helper.py +0 -573
  170. google/adk/tests/unittests/tools/openapi_tool/common/test_common.py +0 -436
  171. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test.yaml +0 -1367
  172. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_spec_parser.py +0 -628
  173. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_toolset.py +0 -139
  174. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_operation_parser.py +0 -406
  175. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_rest_api_tool.py +0 -966
  176. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_tool_auth_handler.py +0 -201
  177. google/adk/tests/unittests/tools/retrieval/__init__.py +0 -14
  178. google/adk/tests/unittests/tools/retrieval/test_vertex_ai_rag_retrieval.py +0 -147
  179. google/adk/tests/unittests/tools/test_agent_tool.py +0 -167
  180. google/adk/tests/unittests/tools/test_base_tool.py +0 -141
  181. google/adk/tests/unittests/tools/test_build_function_declaration.py +0 -277
  182. google/adk/tests/unittests/utils.py +0 -304
  183. google_adk-0.0.3.dist-info/RECORD +0 -340
  184. {google_adk-0.0.3.dist-info → google_adk-0.0.4.dist-info}/entry_points.txt +0 -0
@@ -1,60 +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 import models
16
- from google.adk.models.anthropic_llm import Claude
17
- from google.adk.models.google_llm import Gemini
18
- from google.adk.models.registry import LLMRegistry
19
- import pytest
20
-
21
-
22
- @pytest.mark.parametrize(
23
- 'model_name',
24
- [
25
- 'gemini-1.5-flash',
26
- 'gemini-1.5-flash-001',
27
- 'gemini-1.5-flash-002',
28
- 'gemini-1.5-pro',
29
- 'gemini-1.5-pro-001',
30
- 'gemini-1.5-pro-002',
31
- 'gemini-2.0-flash-exp',
32
- 'projects/123456/locations/us-central1/endpoints/123456', # finetuned vertex gemini endpoint
33
- 'projects/123456/locations/us-central1/publishers/google/models/gemini-2.0-flash-exp', # vertex gemini long name
34
- ],
35
- )
36
- def test_match_gemini_family(model_name):
37
- assert models.LLMRegistry.resolve(model_name) is Gemini
38
-
39
-
40
- @pytest.mark.parametrize(
41
- 'model_name',
42
- [
43
- 'claude-3-5-haiku@20241022',
44
- 'claude-3-5-sonnet-v2@20241022',
45
- 'claude-3-5-sonnet@20240620',
46
- 'claude-3-haiku@20240307',
47
- 'claude-3-opus@20240229',
48
- 'claude-3-sonnet@20240229',
49
- ],
50
- )
51
- def test_match_claude_family(model_name):
52
- LLMRegistry.register(Claude)
53
-
54
- assert models.LLMRegistry.resolve(model_name) is Claude
55
-
56
-
57
- def test_non_exist_model():
58
- with pytest.raises(ValueError) as e_info:
59
- models.LLMRegistry.resolve('non-exist-model')
60
- assert 'Model non-exist-model not found.' in str(e_info.value)
@@ -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,227 +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 enum
16
- import pytest
17
-
18
- from google.adk.events import Event
19
- from google.adk.events import EventActions
20
- from google.adk.sessions import DatabaseSessionService
21
- from google.adk.sessions import InMemorySessionService
22
- from google.genai import types
23
-
24
-
25
- class SessionServiceType(enum.Enum):
26
- IN_MEMORY = 'IN_MEMORY'
27
- DATABASE = 'DATABASE'
28
-
29
-
30
- def get_session_service(
31
- service_type: SessionServiceType = SessionServiceType.IN_MEMORY,
32
- ):
33
- """Creates a session service for testing."""
34
- if service_type == SessionServiceType.DATABASE:
35
- return DatabaseSessionService('sqlite:///:memory:')
36
- return InMemorySessionService()
37
-
38
-
39
- @pytest.mark.parametrize(
40
- 'service_type', [SessionServiceType.IN_MEMORY, SessionServiceType.DATABASE]
41
- )
42
- def test_get_empty_session(service_type):
43
- session_service = get_session_service(service_type)
44
- assert not session_service.get_session(
45
- app_name='my_app', user_id='test_user', session_id='123'
46
- )
47
-
48
-
49
- @pytest.mark.parametrize(
50
- 'service_type', [SessionServiceType.IN_MEMORY, SessionServiceType.DATABASE]
51
- )
52
- def test_create_get_session(service_type):
53
- session_service = get_session_service(service_type)
54
- app_name = 'my_app'
55
- user_id = 'test_user'
56
- state = {'key': 'value'}
57
-
58
- session = session_service.create_session(
59
- app_name=app_name, user_id=user_id, state=state
60
- )
61
- assert session.app_name == app_name
62
- assert session.user_id == user_id
63
- assert session.id
64
- assert session.state == state
65
- assert (
66
- session_service.get_session(
67
- app_name=app_name, user_id=user_id, session_id=session.id
68
- )
69
- == session
70
- )
71
-
72
- session_id = session.id
73
- session_service.delete_session(
74
- app_name=app_name, user_id=user_id, session_id=session_id
75
- )
76
-
77
- assert (
78
- not session_service.get_session(
79
- app_name=app_name, user_id=user_id, session_id=session.id
80
- )
81
- == session
82
- )
83
-
84
-
85
- @pytest.mark.parametrize(
86
- 'service_type', [SessionServiceType.IN_MEMORY, SessionServiceType.DATABASE]
87
- )
88
- def test_create_and_list_sessions(service_type):
89
- session_service = get_session_service(service_type)
90
- app_name = 'my_app'
91
- user_id = 'test_user'
92
-
93
- session_ids = ['session' + str(i) for i in range(5)]
94
- for session_id in session_ids:
95
- session_service.create_session(
96
- app_name=app_name, user_id=user_id, session_id=session_id
97
- )
98
-
99
- sessions = session_service.list_sessions(
100
- app_name=app_name, user_id=user_id
101
- ).sessions
102
- for i in range(len(sessions)):
103
- assert sessions[i].id == session_ids[i]
104
-
105
-
106
- @pytest.mark.parametrize(
107
- 'service_type', [SessionServiceType.IN_MEMORY, SessionServiceType.DATABASE]
108
- )
109
- def test_session_state(service_type):
110
- session_service = get_session_service(service_type)
111
- app_name = 'my_app'
112
- user_id_1 = 'user1'
113
- user_id_2 = 'user2'
114
- session_id_11 = 'session11'
115
- session_id_12 = 'session12'
116
- session_id_2 = 'session2'
117
- state_11 = {'key11': 'value11'}
118
- state_12 = {'key12': 'value12'}
119
-
120
- session_11 = session_service.create_session(
121
- app_name=app_name,
122
- user_id=user_id_1,
123
- state=state_11,
124
- session_id=session_id_11,
125
- )
126
- session_service.create_session(
127
- app_name=app_name,
128
- user_id=user_id_1,
129
- state=state_12,
130
- session_id=session_id_12,
131
- )
132
- session_service.create_session(
133
- app_name=app_name, user_id=user_id_2, session_id=session_id_2
134
- )
135
-
136
- assert session_11.state.get('key11') == 'value11'
137
-
138
- event = Event(
139
- invocation_id='invocation',
140
- author='user',
141
- content=types.Content(role='user', parts=[types.Part(text='text')]),
142
- actions=EventActions(
143
- state_delta={
144
- 'app:key': 'value',
145
- 'user:key1': 'value1',
146
- 'temp:key': 'temp',
147
- 'key11': 'value11_new',
148
- }
149
- ),
150
- )
151
- session_service.append_event(session=session_11, event=event)
152
-
153
- # User and app state is stored, temp state is filtered.
154
- assert session_11.state.get('app:key') == 'value'
155
- assert session_11.state.get('key11') == 'value11_new'
156
- assert session_11.state.get('user:key1') == 'value1'
157
- assert not session_11.state.get('temp:key')
158
-
159
- session_12 = session_service.get_session(
160
- app_name=app_name, user_id=user_id_1, session_id=session_id_12
161
- )
162
- # After getting a new instance, the session_12 got the user and app state,
163
- # even append_event is not applied to it, temp state has no effect
164
- assert session_12.state.get('key12') == 'value12'
165
- assert not session_12.state.get('temp:key')
166
-
167
- # The user1's state is not visible to user2, app state is visible
168
- session_2 = session_service.get_session(
169
- app_name=app_name, user_id=user_id_2, session_id=session_id_2
170
- )
171
- assert session_2.state.get('app:key') == 'value'
172
- assert not session_2.state.get('user:key1')
173
-
174
- assert not session_2.state.get('user:key1')
175
-
176
- # The change to session_11 is persisted
177
- session_11 = session_service.get_session(
178
- app_name=app_name, user_id=user_id_1, session_id=session_id_11
179
- )
180
- assert session_11.state.get('key11') == 'value11_new'
181
- assert session_11.state.get('user:key1') == 'value1'
182
- assert not session_11.state.get('temp:key')
183
-
184
-
185
- @pytest.mark.parametrize(
186
- "service_type", [SessionServiceType.IN_MEMORY, SessionServiceType.DATABASE]
187
- )
188
- def test_create_new_session_will_merge_states(service_type):
189
- session_service = get_session_service(service_type)
190
- app_name = 'my_app'
191
- user_id = 'user'
192
- session_id_1 = 'session1'
193
- session_id_2 = 'session2'
194
- state_1 = {'key1': 'value1'}
195
-
196
- session_1 = session_service.create_session(
197
- app_name=app_name, user_id=user_id, state=state_1, session_id=session_id_1
198
- )
199
-
200
- event = Event(
201
- invocation_id='invocation',
202
- author='user',
203
- content=types.Content(role='user', parts=[types.Part(text='text')]),
204
- actions=EventActions(
205
- state_delta={
206
- 'app:key': 'value',
207
- 'user:key1': 'value1',
208
- 'temp:key': 'temp',
209
- }
210
- ),
211
- )
212
- session_service.append_event(session=session_1, event=event)
213
-
214
- # User and app state is stored, temp state is filtered.
215
- assert session_1.state.get('app:key') == 'value'
216
- assert session_1.state.get('key1') == 'value1'
217
- assert session_1.state.get('user:key1') == 'value1'
218
- assert not session_1.state.get('temp:key')
219
-
220
- session_2 = session_service.create_session(
221
- app_name=app_name, user_id=user_id, state={}, session_id=session_id_2
222
- )
223
- # Session 2 has the persisted states
224
- assert session_2.state.get('app:key') == 'value'
225
- assert session_2.state.get('user:key1') == 'value1'
226
- assert not session_2.state.get('key1')
227
- assert not session_2.state.get('temp:key')
@@ -1,246 +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 re
16
- import this
17
- from typing import Any
18
- import uuid
19
- from dateutil.parser import isoparse
20
- from google.adk.events import Event
21
- from google.adk.events import EventActions
22
- from google.adk.sessions import Session
23
- from google.adk.sessions import VertexAiSessionService
24
- from google.genai import types
25
- import pytest
26
-
27
-
28
- MOCK_SESSION_JSON_1 = {
29
- 'name': (
30
- 'projects/test-project/locations/test-location/'
31
- 'reasoningEngines/123/sessions/1'
32
- ),
33
- 'createTime': '2024-12-12T12:12:12.123456Z',
34
- 'updateTime': '2024-12-12T12:12:12.123456Z',
35
- 'sessionState': {
36
- 'key': {'value': 'test_value'},
37
- },
38
- 'userId': 'user',
39
- }
40
- MOCK_SESSION_JSON_2 = {
41
- 'name': (
42
- 'projects/test-project/locations/test-location/'
43
- 'reasoningEngines/123/sessions/2'
44
- ),
45
- 'updateTime': '2024-12-13T12:12:12.123456Z',
46
- 'userId': 'user',
47
- }
48
- MOCK_SESSION_JSON_3 = {
49
- 'name': (
50
- 'projects/test-project/locations/test-location/'
51
- 'reasoningEngines/123/sessions/3'
52
- ),
53
- 'updateTime': '2024-12-14T12:12:12.123456Z',
54
- 'userId': 'user2',
55
- }
56
- MOCK_EVENT_JSON = [
57
- {
58
- 'name': (
59
- 'projects/test-project/locations/test-location/'
60
- 'reasoningEngines/test_engine/sessions/1/events/123'
61
- ),
62
- 'invocationId': '123',
63
- 'author': 'user',
64
- 'timestamp': '2024-12-12T12:12:12.123456Z',
65
- 'content': {
66
- 'parts': [
67
- {'text': 'test_content'},
68
- ],
69
- },
70
- 'actions': {
71
- 'stateDelta': {
72
- 'key': {'value': 'test_value'},
73
- },
74
- 'transferAgent': 'agent',
75
- },
76
- 'eventMetadata': {
77
- 'partial': False,
78
- 'turnComplete': True,
79
- 'interrupted': False,
80
- 'branch': '',
81
- 'longRunningToolIds': ['tool1'],
82
- },
83
- },
84
- ]
85
-
86
- MOCK_SESSION = Session(
87
- app_name='123',
88
- user_id='user',
89
- id='1',
90
- state=MOCK_SESSION_JSON_1['sessionState'],
91
- last_update_time=isoparse(MOCK_SESSION_JSON_1['updateTime']).timestamp(),
92
- events=[
93
- Event(
94
- id='123',
95
- invocation_id='123',
96
- author='user',
97
- timestamp=isoparse(MOCK_EVENT_JSON[0]['timestamp']).timestamp(),
98
- content=types.Content(parts=[types.Part(text='test_content')]),
99
- actions=EventActions(
100
- transfer_to_agent='agent',
101
- state_delta={'key': {'value': 'test_value'}},
102
- ),
103
- partial=False,
104
- turn_complete=True,
105
- interrupted=False,
106
- branch='',
107
- long_running_tool_ids={'tool1'},
108
- ),
109
- ],
110
- )
111
-
112
-
113
- SESSION_REGEX = r'^reasoningEngines/([^/]+)/sessions/([^/]+)$'
114
- SESSIONS_REGEX = r'^reasoningEngines/([^/]+)/sessions$'
115
- EVENTS_REGEX = r'^reasoningEngines/([^/]+)/sessions/([^/]+)/events$'
116
- LRO_REGEX = r'^operations/([^/]+)$'
117
-
118
-
119
- class MockApiClient:
120
- """Mocks the API Client."""
121
-
122
- def __init__(self) -> None:
123
- """Initializes MockClient."""
124
- this.session_dict: dict[str, Any] = {}
125
- this.event_dict: dict[str, list[Any]] = {}
126
-
127
- def request(self, http_method: str, path: str, request_dict: dict[str, Any]):
128
- """Mocks the API Client request method."""
129
- if http_method == 'GET':
130
- if re.match(SESSION_REGEX, path):
131
- match = re.match(SESSION_REGEX, path)
132
- if match:
133
- session_id = match.group(2)
134
- if session_id in self.session_dict:
135
- return self.session_dict[session_id]
136
- else:
137
- raise ValueError(f'Session not found: {session_id}')
138
- elif re.match(SESSIONS_REGEX, path):
139
- return {
140
- 'sessions': self.session_dict.values(),
141
- }
142
- elif re.match(EVENTS_REGEX, path):
143
- match = re.match(EVENTS_REGEX, path)
144
- if match:
145
- return {'sessionEvents': self.event_dict[match.group(2)]}
146
- elif re.match(LRO_REGEX, path):
147
- return {
148
- 'name': (
149
- 'projects/test-project/locations/test-location/'
150
- 'reasoningEngines/123/sessions/123'
151
- ),
152
- 'done': True,
153
- }
154
- else:
155
- raise ValueError(f'Unsupported path: {path}')
156
- elif http_method == 'POST':
157
- id = str(uuid.uuid4())
158
- self.session_dict[id] = {
159
- 'name': (
160
- 'projects/test-project/locations/test-location/'
161
- 'reasoningEngines/123/sessions/'
162
- + id
163
- ),
164
- 'userId': request_dict['user_id'],
165
- 'sessionState': request_dict.get('sessionState', {}),
166
- 'updateTime': '2024-12-12T12:12:12.123456Z',
167
- }
168
- return {
169
- 'name': (
170
- 'projects/test_project/locations/test_location/'
171
- 'reasoningEngines/test_engine/sessions/123'
172
- ),
173
- 'done': False,
174
- }
175
- elif http_method == 'DELETE':
176
- match = re.match(SESSION_REGEX, path)
177
- if match:
178
- self.session_dict.pop(match.group(2))
179
- else:
180
- raise ValueError(f'Unsupported http method: {http_method}')
181
-
182
-
183
- def mock_vertex_ai_session_service():
184
- """Creates a mock Vertex AI Session service for testing."""
185
- service = VertexAiSessionService(
186
- project='test-project', location='test-location'
187
- )
188
- service.api_client = MockApiClient()
189
- service.api_client.session_dict = {
190
- '1': MOCK_SESSION_JSON_1,
191
- '2': MOCK_SESSION_JSON_2,
192
- '3': MOCK_SESSION_JSON_3,
193
- }
194
- service.api_client.event_dict = {
195
- '1': MOCK_EVENT_JSON,
196
- }
197
- return service
198
-
199
-
200
- def test_get_empty_session():
201
- session_service = mock_vertex_ai_session_service()
202
- with pytest.raises(ValueError) as excinfo:
203
- assert session_service.get_session(
204
- app_name='123', user_id='user', session_id='0'
205
- )
206
- assert str(excinfo.value) == 'Session not found: 0'
207
-
208
-
209
- def test_get_and_delete_session():
210
- session_service = mock_vertex_ai_session_service()
211
-
212
- assert (
213
- session_service.get_session(
214
- app_name='123', user_id='user', session_id='1'
215
- )
216
- == MOCK_SESSION
217
- )
218
-
219
- session_service.delete_session(app_name='123', user_id='user', session_id='1')
220
- with pytest.raises(ValueError) as excinfo:
221
- assert session_service.get_session(
222
- app_name='123', user_id='user', session_id='1'
223
- )
224
- assert str(excinfo.value) == 'Session not found: 1'
225
-
226
- def test_list_sessions():
227
- session_service = mock_vertex_ai_session_service()
228
- sessions = session_service.list_sessions(app_name='123', user_id='user')
229
- assert len(sessions.sessions) == 2
230
- assert sessions.sessions[0].id == '1'
231
- assert sessions.sessions[1].id == '2'
232
-
233
- def test_create_session():
234
- session_service = mock_vertex_ai_session_service()
235
- session = session_service.create_session(
236
- app_name='123', user_id='user', state={'key': 'value'}
237
- )
238
- assert session.state == {'key': 'value'}
239
- assert session.app_name == '123'
240
- assert session.user_id == 'user'
241
- assert session.last_update_time is not None
242
-
243
- session_id = session.id
244
- assert session == session_service.get_session(
245
- app_name='123', user_id='user', session_id=session_id
246
- )
@@ -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,50 +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.agents import LiveRequestQueue
17
- from google.adk.models import LlmResponse
18
- from google.genai import types
19
- import pytest
20
-
21
- from .. import utils
22
-
23
-
24
- @pytest.mark.skip(reason='Streaming is hanging.')
25
- def test_streaming():
26
- response1 = LlmResponse(
27
- turn_complete=True,
28
- )
29
-
30
- mock_model = utils.MockModel.create([response1])
31
-
32
- root_agent = Agent(
33
- name='root_agent',
34
- model=mock_model,
35
- tools=[],
36
- )
37
-
38
- runner = utils.InMemoryRunner(
39
- root_agent=root_agent, response_modalities=['AUDIO']
40
- )
41
- live_request_queue = LiveRequestQueue()
42
- live_request_queue.send_realtime(
43
- blob=types.Blob(data=b'\x00\xFF', mime_type='audio/pcm')
44
- )
45
- res_events = runner.run_live(live_request_queue)
46
-
47
- assert res_events is not None, 'Expected a list of events, got None.'
48
- assert (
49
- len(res_events) > 0
50
- ), 'Expected at least one response, but got an empty list.'
@@ -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
-