google-adk 0.0.2__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 (153) hide show
  1. google/adk/agents/run_config.py +4 -0
  2. google/adk/auth/auth_preprocessor.py +19 -16
  3. google/adk/cli/agent_graph.py +31 -5
  4. google/adk/cli/browser/index.html +1 -1
  5. google/adk/cli/browser/{main-XUU6OGCC.js → main-CU22TRPI.js} +30 -30
  6. google/adk/cli/cli.py +8 -8
  7. google/adk/cli/cli_deploy.py +2 -4
  8. google/adk/cli/cli_tools_click.py +57 -12
  9. google/adk/cli/fast_api.py +19 -9
  10. google/adk/flows/llm_flows/contents.py +21 -1
  11. google/adk/flows/llm_flows/functions.py +3 -1
  12. google/adk/models/google_llm.py +0 -1
  13. google/adk/runners.py +13 -2
  14. google/adk/version.py +1 -1
  15. {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info}/METADATA +4 -2
  16. google_adk-0.0.4.dist-info/RECORD +175 -0
  17. {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info}/WHEEL +1 -1
  18. google/adk/cli/media_streamer/__init__.py +0 -19
  19. google/adk/cli/media_streamer/index.html +0 -228
  20. google/adk/tests/__init__.py +0 -14
  21. google/adk/tests/integration/.env.example +0 -10
  22. google/adk/tests/integration/__init__.py +0 -18
  23. google/adk/tests/integration/conftest.py +0 -119
  24. google/adk/tests/integration/fixture/__init__.py +0 -14
  25. google/adk/tests/integration/fixture/agent_with_config/__init__.py +0 -15
  26. google/adk/tests/integration/fixture/agent_with_config/agent.py +0 -88
  27. google/adk/tests/integration/fixture/callback_agent/__init__.py +0 -15
  28. google/adk/tests/integration/fixture/callback_agent/agent.py +0 -105
  29. google/adk/tests/integration/fixture/context_update_test/OWNERS +0 -1
  30. google/adk/tests/integration/fixture/context_update_test/__init__.py +0 -15
  31. google/adk/tests/integration/fixture/context_update_test/agent.py +0 -43
  32. google/adk/tests/integration/fixture/context_update_test/successful_test.session.json +0 -582
  33. google/adk/tests/integration/fixture/context_variable_agent/__init__.py +0 -15
  34. google/adk/tests/integration/fixture/context_variable_agent/agent.py +0 -115
  35. google/adk/tests/integration/fixture/customer_support_ma/__init__.py +0 -15
  36. google/adk/tests/integration/fixture/customer_support_ma/agent.py +0 -172
  37. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/__init__.py +0 -15
  38. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/agent.py +0 -338
  39. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/order_query.test.json +0 -69
  40. google/adk/tests/integration/fixture/ecommerce_customer_service_agent/test_config.json +0 -6
  41. google/adk/tests/integration/fixture/flow_complex_spark/__init__.py +0 -15
  42. google/adk/tests/integration/fixture/flow_complex_spark/agent.py +0 -182
  43. google/adk/tests/integration/fixture/flow_complex_spark/sample.debug.log +0 -243
  44. google/adk/tests/integration/fixture/flow_complex_spark/sample.session.json +0 -190
  45. google/adk/tests/integration/fixture/hello_world_agent/__init__.py +0 -15
  46. google/adk/tests/integration/fixture/hello_world_agent/agent.py +0 -95
  47. google/adk/tests/integration/fixture/hello_world_agent/roll_die.test.json +0 -24
  48. google/adk/tests/integration/fixture/hello_world_agent/test_config.json +0 -6
  49. google/adk/tests/integration/fixture/home_automation_agent/__init__.py +0 -15
  50. google/adk/tests/integration/fixture/home_automation_agent/agent.py +0 -304
  51. google/adk/tests/integration/fixture/home_automation_agent/simple_test.test.json +0 -5
  52. google/adk/tests/integration/fixture/home_automation_agent/simple_test2.test.json +0 -5
  53. google/adk/tests/integration/fixture/home_automation_agent/test_config.json +0 -5
  54. google/adk/tests/integration/fixture/home_automation_agent/test_files/dependent_tool_calls.test.json +0 -18
  55. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/eval_data.test.json +0 -17
  56. google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/test_config.json +0 -6
  57. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_multi_turn_conversation.test.json +0 -18
  58. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test.test.json +0 -17
  59. google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test2.test.json +0 -5
  60. google/adk/tests/integration/fixture/home_automation_agent/test_files/test_config.json +0 -5
  61. google/adk/tests/integration/fixture/tool_agent/__init__.py +0 -15
  62. google/adk/tests/integration/fixture/tool_agent/agent.py +0 -218
  63. google/adk/tests/integration/fixture/tool_agent/files/Agent_test_plan.pdf +0 -0
  64. google/adk/tests/integration/fixture/trip_planner_agent/__init__.py +0 -15
  65. google/adk/tests/integration/fixture/trip_planner_agent/agent.py +0 -110
  66. google/adk/tests/integration/fixture/trip_planner_agent/initial.session.json +0 -13
  67. google/adk/tests/integration/fixture/trip_planner_agent/test_config.json +0 -5
  68. google/adk/tests/integration/fixture/trip_planner_agent/test_files/initial.session.json +0 -13
  69. google/adk/tests/integration/fixture/trip_planner_agent/test_files/test_config.json +0 -5
  70. google/adk/tests/integration/fixture/trip_planner_agent/test_files/trip_inquiry_sub_agent.test.json +0 -7
  71. google/adk/tests/integration/fixture/trip_planner_agent/trip_inquiry.test.json +0 -19
  72. google/adk/tests/integration/models/__init__.py +0 -14
  73. google/adk/tests/integration/models/test_google_llm.py +0 -65
  74. google/adk/tests/integration/test_callback.py +0 -70
  75. google/adk/tests/integration/test_context_variable.py +0 -67
  76. google/adk/tests/integration/test_evalute_agent_in_fixture.py +0 -76
  77. google/adk/tests/integration/test_multi_agent.py +0 -28
  78. google/adk/tests/integration/test_multi_turn.py +0 -42
  79. google/adk/tests/integration/test_single_agent.py +0 -23
  80. google/adk/tests/integration/test_sub_agent.py +0 -26
  81. google/adk/tests/integration/test_system_instruction.py +0 -177
  82. google/adk/tests/integration/test_tools.py +0 -287
  83. google/adk/tests/integration/test_with_test_file.py +0 -34
  84. google/adk/tests/integration/tools/__init__.py +0 -14
  85. google/adk/tests/integration/utils/__init__.py +0 -16
  86. google/adk/tests/integration/utils/asserts.py +0 -75
  87. google/adk/tests/integration/utils/test_runner.py +0 -97
  88. google/adk/tests/unittests/__init__.py +0 -14
  89. google/adk/tests/unittests/agents/__init__.py +0 -14
  90. google/adk/tests/unittests/agents/test_base_agent.py +0 -407
  91. google/adk/tests/unittests/agents/test_langgraph_agent.py +0 -191
  92. google/adk/tests/unittests/agents/test_llm_agent_callbacks.py +0 -138
  93. google/adk/tests/unittests/agents/test_llm_agent_fields.py +0 -231
  94. google/adk/tests/unittests/agents/test_loop_agent.py +0 -136
  95. google/adk/tests/unittests/agents/test_parallel_agent.py +0 -92
  96. google/adk/tests/unittests/agents/test_sequential_agent.py +0 -114
  97. google/adk/tests/unittests/artifacts/__init__.py +0 -14
  98. google/adk/tests/unittests/artifacts/test_artifact_service.py +0 -276
  99. google/adk/tests/unittests/auth/test_auth_handler.py +0 -575
  100. google/adk/tests/unittests/conftest.py +0 -73
  101. google/adk/tests/unittests/fast_api/__init__.py +0 -14
  102. google/adk/tests/unittests/fast_api/test_fast_api.py +0 -269
  103. google/adk/tests/unittests/flows/__init__.py +0 -14
  104. google/adk/tests/unittests/flows/llm_flows/__init__.py +0 -14
  105. google/adk/tests/unittests/flows/llm_flows/_test_examples.py +0 -142
  106. google/adk/tests/unittests/flows/llm_flows/test_agent_transfer.py +0 -311
  107. google/adk/tests/unittests/flows/llm_flows/test_functions_long_running.py +0 -244
  108. google/adk/tests/unittests/flows/llm_flows/test_functions_request_euc.py +0 -346
  109. google/adk/tests/unittests/flows/llm_flows/test_functions_sequential.py +0 -93
  110. google/adk/tests/unittests/flows/llm_flows/test_functions_simple.py +0 -258
  111. google/adk/tests/unittests/flows/llm_flows/test_identity.py +0 -66
  112. google/adk/tests/unittests/flows/llm_flows/test_instructions.py +0 -164
  113. google/adk/tests/unittests/flows/llm_flows/test_model_callbacks.py +0 -142
  114. google/adk/tests/unittests/flows/llm_flows/test_other_configs.py +0 -46
  115. google/adk/tests/unittests/flows/llm_flows/test_tool_callbacks.py +0 -269
  116. google/adk/tests/unittests/models/__init__.py +0 -14
  117. google/adk/tests/unittests/models/test_google_llm.py +0 -224
  118. google/adk/tests/unittests/models/test_litellm.py +0 -804
  119. google/adk/tests/unittests/models/test_models.py +0 -60
  120. google/adk/tests/unittests/sessions/__init__.py +0 -14
  121. google/adk/tests/unittests/sessions/test_session_service.py +0 -227
  122. google/adk/tests/unittests/sessions/test_vertex_ai_session_service.py +0 -246
  123. google/adk/tests/unittests/streaming/__init__.py +0 -14
  124. google/adk/tests/unittests/streaming/test_streaming.py +0 -50
  125. google/adk/tests/unittests/tools/__init__.py +0 -14
  126. google/adk/tests/unittests/tools/apihub_tool/clients/test_apihub_client.py +0 -499
  127. google/adk/tests/unittests/tools/apihub_tool/test_apihub_toolset.py +0 -204
  128. google/adk/tests/unittests/tools/application_integration_tool/clients/test_connections_client.py +0 -600
  129. google/adk/tests/unittests/tools/application_integration_tool/clients/test_integration_client.py +0 -630
  130. google/adk/tests/unittests/tools/application_integration_tool/test_application_integration_toolset.py +0 -345
  131. google/adk/tests/unittests/tools/google_api_tool/__init__.py +0 -13
  132. google/adk/tests/unittests/tools/google_api_tool/test_googleapi_to_openapi_converter.py +0 -657
  133. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_auto_auth_credential_exchanger.py +0 -145
  134. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_base_auth_credential_exchanger.py +0 -68
  135. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_oauth2_exchanger.py +0 -153
  136. google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_service_account_exchanger.py +0 -196
  137. google/adk/tests/unittests/tools/openapi_tool/auth/test_auth_helper.py +0 -573
  138. google/adk/tests/unittests/tools/openapi_tool/common/test_common.py +0 -436
  139. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test.yaml +0 -1367
  140. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_spec_parser.py +0 -628
  141. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_toolset.py +0 -139
  142. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_operation_parser.py +0 -406
  143. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_rest_api_tool.py +0 -966
  144. google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_tool_auth_handler.py +0 -201
  145. google/adk/tests/unittests/tools/retrieval/__init__.py +0 -14
  146. google/adk/tests/unittests/tools/retrieval/test_vertex_ai_rag_retrieval.py +0 -147
  147. google/adk/tests/unittests/tools/test_agent_tool.py +0 -167
  148. google/adk/tests/unittests/tools/test_base_tool.py +0 -141
  149. google/adk/tests/unittests/tools/test_build_function_declaration.py +0 -277
  150. google/adk/tests/unittests/utils.py +0 -304
  151. google_adk-0.0.2.dist-info/RECORD +0 -308
  152. {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info}/entry_points.txt +0 -0
  153. {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info/licenses}/LICENSE +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
-