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,228 +0,0 @@
1
- <!--
2
- Copyright 2025 Google LLC
3
-
4
- Licensed under the Apache License, Version 2.0 (the "License");
5
- you may not use this file except in compliance with the License.
6
- You may obtain a copy of the License at
7
-
8
- http://www.apache.org/licenses/LICENSE-2.0
9
-
10
- Unless required by applicable law or agreed to in writing, software
11
- distributed under the License is distributed on an "AS IS" BASIS,
12
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
- See the License for the specific language governing permissions and
14
- limitations under the License.
15
- -->
16
-
17
- <html>
18
- <head>
19
- <style type="text/css">
20
- body {
21
- font-family: sans-serif;
22
- padding: 10px;
23
- }
24
- button {
25
- font-size: 16px;
26
- padding: 10px 15px;
27
- margin-right: 10px;
28
- }
29
- #status {
30
- font-weight: bold;
31
- }
32
- /* Mirror the preview video */
33
- #videoPreview {
34
- border: 1px solid #ccc;
35
- transform: scaleX(-1);
36
- }
37
- </style>
38
- </head>
39
- <body>
40
- <h2>Audio &amp; Video Recorder Component</h2>
41
- <!-- Video preview -->
42
- <video id="videoPreview" autoplay muted playsinline width="320" height="240"></video>
43
- <br/>
44
- <!-- Unified start/stop buttons -->
45
- <button id="startButton" onclick="startRecording()">Start Streaming</button>
46
- <button id="stopButton" onclick="stopRecording()" disabled>Stop Streaming</button>
47
- <span id="status">Idle</span>
48
-
49
- <script type="text/javascript">
50
- // --- Streamlit component lifecycle constants ---
51
- const SET_COMPONENT_VALUE = "streamlit:setComponentValue";
52
- const COMPONENT_READY = "streamlit:componentReady";
53
- const SET_FRAME_HEIGHT = "streamlit:setFrameHeight";
54
- const RENDER = "streamlit:render";
55
-
56
- // Helper function to send messages to the Streamlit host.
57
- function _sendMessage(type, data) {
58
- const outboundData = Object.assign({ isStreamlitMessage: true, type: type }, data);
59
- window.parent.postMessage(outboundData, "*");
60
- }
61
- function setFrameHeight(height) {
62
- _sendMessage(SET_FRAME_HEIGHT, { height: height });
63
- }
64
- // This function sends the combined data.
65
- function notifyHost(data) {
66
- _sendMessage(SET_COMPONENT_VALUE, data);
67
- }
68
- function initialize() {
69
- _sendMessage(COMPONENT_READY, { apiVersion: 1 });
70
- window.addEventListener("load", function() {
71
- setTimeout(() => setFrameHeight(document.documentElement.clientHeight), 0);
72
- });
73
- }
74
- initialize();
75
-
76
- // --- Global Variables ---
77
- let isRecording = false;
78
- let audioContext;
79
- let scriptProcessor;
80
- let audioAccumulatedChunks = [];
81
- let audioFlushIntervalId = null;
82
- let videoCaptureIntervalId = null;
83
- const SAMPLE_RATE = 16000;
84
- const BUFFER_SIZE = 4096;
85
- let mediaStream = null; // for video (and optionally audio if needed)
86
-
87
- // Global object to hold combined recording data.
88
- let recordingData = {};
89
-
90
- // Instead of calling notifyHost directly, update our combined data and send.
91
- function sendCombinedUpdate(key, value) {
92
- recordingData[key] = value;
93
- notifyHost({
94
- value: recordingData,
95
- dataType: "json"
96
- });
97
- }
98
-
99
- const videoElement = document.getElementById("videoPreview");
100
- const statusSpan = document.getElementById("status");
101
- const startButton = document.getElementById("startButton");
102
- const stopButton = document.getElementById("stopButton");
103
-
104
- // --- Start Recording: Capture both audio and video ---
105
- async function startRecording() {
106
- if (isRecording) return;
107
- isRecording = true;
108
- startButton.disabled = true;
109
- stopButton.disabled = false;
110
- statusSpan.innerText = "Streaming...";
111
-
112
- // --- Setup audio recording ---
113
- try {
114
- audioContext = new (window.AudioContext || window.webkitAudioContext)({ sampleRate: SAMPLE_RATE });
115
- let audioStream = await navigator.mediaDevices.getUserMedia({ audio: true });
116
- let mediaStreamSource = audioContext.createMediaStreamSource(audioStream);
117
- scriptProcessor = audioContext.createScriptProcessor(BUFFER_SIZE, 1, 1);
118
- scriptProcessor.onaudioprocess = function(event) {
119
- let inputBuffer = event.inputBuffer;
120
- let inputData = inputBuffer.getChannelData(0);
121
- let pcmData = new Int16Array(inputData.length);
122
- for (let i = 0; i < inputData.length; i++) {
123
- let s = Math.max(-1, Math.min(1, inputData[i]));
124
- pcmData[i] = s < 0 ? s * 0x8000 : s * 0x7FFF;
125
- }
126
- audioAccumulatedChunks.push(pcmData);
127
- };
128
- mediaStreamSource.connect(scriptProcessor);
129
- scriptProcessor.connect(audioContext.destination);
130
- } catch (err) {
131
- console.error("Error accessing microphone:", err);
132
- statusSpan.innerText = "Audio Error: " + err;
133
- }
134
-
135
- // --- Setup video recording ---
136
- try {
137
- let videoStream = await navigator.mediaDevices.getUserMedia({ video: true });
138
- videoElement.srcObject = videoStream;
139
- // Save the video stream so we can stop it later.
140
- mediaStream = videoStream;
141
- } catch (err) {
142
- console.error("Error accessing camera:", err);
143
- statusSpan.innerText = "Video Error: " + err;
144
- }
145
-
146
- // --- Start interval timers for flushing audio and capturing video ---
147
- videoCaptureIntervalId = setInterval(captureVideoFrame, 250); // every 250ms
148
- audioFlushIntervalId = setInterval(flushAudioChunks, 250); // every 250ms
149
- }
150
-
151
- // --- Stop Recording ---
152
- function stopRecording() {
153
- if (!isRecording) return;
154
- isRecording = false;
155
- startButton.disabled = false;
156
- stopButton.disabled = true;
157
- statusSpan.innerText = "Stopped.";
158
-
159
- // Stop audio processing
160
- if (scriptProcessor) {
161
- scriptProcessor.disconnect();
162
- scriptProcessor = null;
163
- }
164
- if (audioContext) {
165
- audioContext.close();
166
- audioContext = null;
167
- }
168
- if (audioFlushIntervalId) {
169
- clearInterval(audioFlushIntervalId);
170
- audioFlushIntervalId = null;
171
- }
172
-
173
- // Stop video stream
174
- if (mediaStream) {
175
- mediaStream.getTracks().forEach(track => track.stop());
176
- mediaStream = null;
177
- }
178
- if (videoCaptureIntervalId) {
179
- clearInterval(videoCaptureIntervalId);
180
- videoCaptureIntervalId = null;
181
- }
182
-
183
- // Flush any remaining audio chunks
184
- flushAudioChunks();
185
-
186
- // Final notification that recording has stopped.
187
- sendCombinedUpdate("message", "Recording stopped");
188
- sendCombinedUpdate("isRecording", false);
189
- }
190
-
191
- // --- Flush accumulated audio chunks ---
192
- function flushAudioChunks() {
193
- if (audioAccumulatedChunks.length === 0) return;
194
- // Combine all audio chunks into one array.
195
- let totalLength = audioAccumulatedChunks.reduce((sum, chunk) => sum + chunk.length, 0);
196
- let combined = new Int16Array(totalLength);
197
- let offset = 0;
198
- for (let chunk of audioAccumulatedChunks) {
199
- combined.set(chunk, offset);
200
- offset += chunk.length;
201
- }
202
- audioAccumulatedChunks = []; // clear accumulator
203
-
204
- // Convert combined PCM data to a Blob.
205
- let blob = new Blob([combined.buffer], { type: 'application/octet-stream' });
206
- let reader = new FileReader();
207
- reader.onloadend = function() {
208
- // Instead of sending only audio, update the combined data.
209
- sendCombinedUpdate("audioChunk", reader.result);
210
- };
211
- reader.readAsDataURL(blob);
212
- }
213
-
214
- // --- Capture a single video frame ---
215
- function captureVideoFrame() {
216
- if (!videoElement.srcObject) return;
217
- const canvas = document.createElement("canvas");
218
- canvas.width = videoElement.videoWidth || 320;
219
- canvas.height = videoElement.videoHeight || 240;
220
- const ctx = canvas.getContext("2d");
221
- ctx.drawImage(videoElement, 0, 0, canvas.width, canvas.height);
222
- let dataUrl = canvas.toDataURL("image/jpeg");
223
- // Instead of a separate notifyHost call, update the combined data.
224
- sendCombinedUpdate("videoFrame", dataUrl);
225
- }
226
- </script>
227
- </body>
228
- </html>
@@ -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,10 +0,0 @@
1
- # Copy as .env file and fill your values below to run integration tests.
2
-
3
- # Choose Backend: GOOGLE_AI_ONLY | VERTEX_ONLY | BOTH (default)
4
- TEST_BACKEND=BOTH
5
-
6
- # ML Dev backend config
7
- GOOGLE_API_KEY=YOUR_VALUE_HERE
8
- # Vertex backend config
9
- GOOGLE_CLOUD_PROJECT=YOUR_VALUE_HERE
10
- GOOGLE_CLOUD_LOCATION=YOUR_VALUE_HERE
@@ -1,18 +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
- # This allows pytest to show the values of the asserts.
18
- pytest.register_assert_rewrite('tests.integration.utils')
@@ -1,119 +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 logging
16
- import os
17
- from typing import Literal
18
- import warnings
19
-
20
- from dotenv import load_dotenv
21
- from google.adk import Agent
22
- from pytest import fixture
23
- from pytest import FixtureRequest
24
- from pytest import hookimpl
25
- from pytest import Metafunc
26
-
27
- from .utils import TestRunner
28
-
29
- logger = logging.getLogger(__name__)
30
-
31
-
32
- def load_env_for_tests():
33
- dotenv_path = os.path.join(os.path.dirname(__file__), '.env')
34
- if not os.path.exists(dotenv_path):
35
- warnings.warn(
36
- f'Missing .env file at {dotenv_path}. See dotenv.sample for an example.'
37
- )
38
- else:
39
- load_dotenv(dotenv_path, override=True, verbose=True)
40
- if 'GOOGLE_API_KEY' not in os.environ:
41
- warnings.warn(
42
- 'Missing GOOGLE_API_KEY in the environment variables. GOOGLE_AI backend'
43
- ' integration tests will fail.'
44
- )
45
- for env_var in [
46
- 'GOOGLE_CLOUD_PROJECT',
47
- 'GOOGLE_CLOUD_LOCATION',
48
- ]:
49
- if env_var not in os.environ:
50
- warnings.warn(
51
- f'Missing {env_var} in the environment variables. Vertex backend'
52
- ' integration tests will fail.'
53
- )
54
-
55
-
56
- load_env_for_tests()
57
-
58
- BackendType = Literal['GOOGLE_AI', 'VERTEX']
59
-
60
-
61
- @fixture
62
- def agent_runner(request: FixtureRequest) -> TestRunner:
63
- assert isinstance(request.param, dict)
64
-
65
- if 'agent' in request.param:
66
- assert isinstance(request.param['agent'], Agent)
67
- return TestRunner(request.param['agent'])
68
- elif 'agent_name' in request.param:
69
- assert isinstance(request.param['agent_name'], str)
70
- return TestRunner.from_agent_name(request.param['agent_name'])
71
-
72
- raise NotImplementedError('Must provide agent or agent_name.')
73
-
74
-
75
- @fixture(autouse=True)
76
- def llm_backend(request: FixtureRequest):
77
- # Set backend environment value.
78
- original_val = os.environ.get('GOOGLE_GENAI_USE_VERTEXAI')
79
- backend_type = request.param
80
- if backend_type == 'GOOGLE_AI':
81
- os.environ['GOOGLE_GENAI_USE_VERTEXAI'] = '0'
82
- else:
83
- os.environ['GOOGLE_GENAI_USE_VERTEXAI'] = '1'
84
-
85
- yield # Run the test
86
-
87
- # Restore the environment
88
- if original_val is None:
89
- os.environ.pop('GOOGLE_GENAI_USE_VERTEXAI', None)
90
- else:
91
- os.environ['GOOGLE_GENAI_USE_VERTEXAI'] = original_val
92
-
93
-
94
- @hookimpl(tryfirst=True)
95
- def pytest_generate_tests(metafunc: Metafunc):
96
- if llm_backend.__name__ in metafunc.fixturenames:
97
- if not _is_explicitly_marked(llm_backend.__name__, metafunc):
98
- test_backend = os.environ.get('TEST_BACKEND', 'BOTH')
99
- if test_backend == 'GOOGLE_AI_ONLY':
100
- metafunc.parametrize(llm_backend.__name__, ['GOOGLE_AI'], indirect=True)
101
- elif test_backend == 'VERTEX_ONLY':
102
- metafunc.parametrize(llm_backend.__name__, ['VERTEX'], indirect=True)
103
- elif test_backend == 'BOTH':
104
- metafunc.parametrize(
105
- llm_backend.__name__, ['GOOGLE_AI', 'VERTEX'], indirect=True
106
- )
107
- else:
108
- raise ValueError(
109
- f'Invalid TEST_BACKEND value: {test_backend}, should be one of'
110
- ' [GOOGLE_AI_ONLY, VERTEX_ONLY, BOTH]'
111
- )
112
-
113
-
114
- def _is_explicitly_marked(mark_name: str, metafunc: Metafunc) -> bool:
115
- if hasattr(metafunc.function, 'pytestmark'):
116
- for mark in metafunc.function.pytestmark:
117
- if mark.name == 'parametrize' and mark.args[0] == mark_name:
118
- return True
119
- return False
@@ -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,15 +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 . import agent
@@ -1,88 +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 Agent
16
- from google.genai import types
17
-
18
- new_message = types.Content(
19
- role="user",
20
- parts=[types.Part.from_text(text="Count a number")],
21
- )
22
-
23
- google_agent_1 = Agent(
24
- model="gemini-1.5-flash",
25
- name="agent_1",
26
- description="The first agent in the team.",
27
- instruction="Just say 1",
28
- generate_content_config=types.GenerateContentConfig(
29
- temperature=0.1,
30
- ),
31
- )
32
-
33
- google_agent_2 = Agent(
34
- model="gemini-1.5-flash",
35
- name="agent_2",
36
- description="The second agent in the team.",
37
- instruction="Just say 2",
38
- generate_content_config=types.GenerateContentConfig(
39
- temperature=0.2,
40
- safety_settings=[{
41
- "category": "HARM_CATEGORY_HATE_SPEECH",
42
- "threshold": "BLOCK_ONLY_HIGH",
43
- }],
44
- ),
45
- )
46
-
47
- google_agent_3 = Agent(
48
- model="gemini-1.5-flash",
49
- name="agent_3",
50
- description="The third agent in the team.",
51
- instruction="Just say 3",
52
- generate_content_config=types.GenerateContentConfig(
53
- temperature=0.5,
54
- safety_settings=[{
55
- "category": "HARM_CATEGORY_DANGEROUS_CONTENT",
56
- "threshold": "BLOCK_NONE",
57
- }],
58
- ),
59
- )
60
-
61
- google_agent_with_instruction_in_config = Agent(
62
- model="gemini-1.5-flash",
63
- name="agent",
64
- generate_content_config=types.GenerateContentConfig(
65
- temperature=0.5, system_instruction="Count 1"
66
- ),
67
- )
68
-
69
-
70
- def function():
71
- pass
72
-
73
-
74
- google_agent_with_tools_in_config = Agent(
75
- model="gemini-1.5-flash",
76
- name="agent",
77
- generate_content_config=types.GenerateContentConfig(
78
- temperature=0.5, tools=[function]
79
- ),
80
- )
81
-
82
- google_agent_with_response_schema_in_config = Agent(
83
- model="gemini-1.5-flash",
84
- name="agent",
85
- generate_content_config=types.GenerateContentConfig(
86
- temperature=0.5, response_schema={"key": "value"}
87
- ),
88
- )
@@ -1,15 +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 . import agent
@@ -1,105 +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 Optional
16
-
17
- from google.adk import Agent
18
- from google.adk.agents.callback_context import CallbackContext
19
- from google.adk.agents.invocation_context import InvocationContext
20
- from google.adk.models import LlmRequest
21
- from google.adk.models import LlmResponse
22
- from google.genai import types
23
-
24
-
25
- def before_agent_call_end_invocation(
26
- callback_context: CallbackContext,
27
- ) -> types.Content:
28
- return types.Content(
29
- role='model',
30
- parts=[types.Part(text='End invocation event before agent call.')],
31
- )
32
-
33
-
34
- def before_agent_call(
35
- invocation_context: InvocationContext,
36
- ) -> types.Content:
37
- return types.Content(
38
- role='model',
39
- parts=[types.Part.from_text(text='Plain text event before agent call.')],
40
- )
41
-
42
-
43
- def before_model_call_end_invocation(
44
- callback_context: CallbackContext, llm_request: LlmRequest
45
- ) -> LlmResponse:
46
- return LlmResponse(
47
- content=types.Content(
48
- role='model',
49
- parts=[
50
- types.Part.from_text(
51
- text='End invocation event before model call.'
52
- )
53
- ],
54
- )
55
- )
56
-
57
-
58
- def before_model_call(
59
- invocation_context: InvocationContext, request: LlmRequest
60
- ) -> LlmResponse:
61
- request.config.system_instruction = 'Just return 999 as response.'
62
- return LlmResponse(
63
- content=types.Content(
64
- role='model',
65
- parts=[
66
- types.Part.from_text(
67
- text='Update request event before model call.'
68
- )
69
- ],
70
- )
71
- )
72
-
73
-
74
- def after_model_call(
75
- callback_context: CallbackContext,
76
- llm_response: LlmResponse,
77
- ) -> Optional[LlmResponse]:
78
- content = llm_response.content
79
- if not content or not content.parts or not content.parts[0].text:
80
- return
81
-
82
- content.parts[0].text += 'Update response event after model call.'
83
- return llm_response
84
-
85
-
86
- before_agent_callback_agent = Agent(
87
- model='gemini-1.5-flash',
88
- name='before_agent_callback_agent',
89
- instruction='echo 1',
90
- before_agent_callback=before_agent_call_end_invocation,
91
- )
92
-
93
- before_model_callback_agent = Agent(
94
- model='gemini-1.5-flash',
95
- name='before_model_callback_agent',
96
- instruction='echo 2',
97
- before_model_callback=before_model_call_end_invocation,
98
- )
99
-
100
- after_model_callback_agent = Agent(
101
- model='gemini-1.5-flash',
102
- name='after_model_callback_agent',
103
- instruction='Say hello',
104
- after_model_callback=after_model_call,
105
- )
@@ -1,15 +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 . import agent