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.
- google/adk/agents/run_config.py +4 -0
- google/adk/auth/auth_preprocessor.py +19 -16
- google/adk/cli/agent_graph.py +31 -5
- google/adk/cli/browser/index.html +1 -1
- google/adk/cli/browser/{main-XUU6OGCC.js → main-CU22TRPI.js} +30 -30
- google/adk/cli/cli.py +8 -8
- google/adk/cli/cli_deploy.py +2 -4
- google/adk/cli/cli_tools_click.py +57 -12
- google/adk/cli/fast_api.py +19 -9
- google/adk/flows/llm_flows/contents.py +21 -1
- google/adk/flows/llm_flows/functions.py +3 -1
- google/adk/models/google_llm.py +0 -1
- google/adk/runners.py +13 -2
- google/adk/version.py +1 -1
- {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info}/METADATA +4 -2
- google_adk-0.0.4.dist-info/RECORD +175 -0
- {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info}/WHEEL +1 -1
- google/adk/cli/media_streamer/__init__.py +0 -19
- google/adk/cli/media_streamer/index.html +0 -228
- google/adk/tests/__init__.py +0 -14
- google/adk/tests/integration/.env.example +0 -10
- google/adk/tests/integration/__init__.py +0 -18
- google/adk/tests/integration/conftest.py +0 -119
- google/adk/tests/integration/fixture/__init__.py +0 -14
- google/adk/tests/integration/fixture/agent_with_config/__init__.py +0 -15
- google/adk/tests/integration/fixture/agent_with_config/agent.py +0 -88
- google/adk/tests/integration/fixture/callback_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/callback_agent/agent.py +0 -105
- google/adk/tests/integration/fixture/context_update_test/OWNERS +0 -1
- google/adk/tests/integration/fixture/context_update_test/__init__.py +0 -15
- google/adk/tests/integration/fixture/context_update_test/agent.py +0 -43
- google/adk/tests/integration/fixture/context_update_test/successful_test.session.json +0 -582
- google/adk/tests/integration/fixture/context_variable_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/context_variable_agent/agent.py +0 -115
- google/adk/tests/integration/fixture/customer_support_ma/__init__.py +0 -15
- google/adk/tests/integration/fixture/customer_support_ma/agent.py +0 -172
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/agent.py +0 -338
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/order_query.test.json +0 -69
- google/adk/tests/integration/fixture/ecommerce_customer_service_agent/test_config.json +0 -6
- google/adk/tests/integration/fixture/flow_complex_spark/__init__.py +0 -15
- google/adk/tests/integration/fixture/flow_complex_spark/agent.py +0 -182
- google/adk/tests/integration/fixture/flow_complex_spark/sample.debug.log +0 -243
- google/adk/tests/integration/fixture/flow_complex_spark/sample.session.json +0 -190
- google/adk/tests/integration/fixture/hello_world_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/hello_world_agent/agent.py +0 -95
- google/adk/tests/integration/fixture/hello_world_agent/roll_die.test.json +0 -24
- google/adk/tests/integration/fixture/hello_world_agent/test_config.json +0 -6
- google/adk/tests/integration/fixture/home_automation_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/home_automation_agent/agent.py +0 -304
- google/adk/tests/integration/fixture/home_automation_agent/simple_test.test.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/simple_test2.test.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/test_config.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/test_files/dependent_tool_calls.test.json +0 -18
- google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/eval_data.test.json +0 -17
- google/adk/tests/integration/fixture/home_automation_agent/test_files/memorizing_past_events/test_config.json +0 -6
- google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_multi_turn_conversation.test.json +0 -18
- google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test.test.json +0 -17
- google/adk/tests/integration/fixture/home_automation_agent/test_files/simple_test2.test.json +0 -5
- google/adk/tests/integration/fixture/home_automation_agent/test_files/test_config.json +0 -5
- google/adk/tests/integration/fixture/tool_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/tool_agent/agent.py +0 -218
- google/adk/tests/integration/fixture/tool_agent/files/Agent_test_plan.pdf +0 -0
- google/adk/tests/integration/fixture/trip_planner_agent/__init__.py +0 -15
- google/adk/tests/integration/fixture/trip_planner_agent/agent.py +0 -110
- google/adk/tests/integration/fixture/trip_planner_agent/initial.session.json +0 -13
- google/adk/tests/integration/fixture/trip_planner_agent/test_config.json +0 -5
- google/adk/tests/integration/fixture/trip_planner_agent/test_files/initial.session.json +0 -13
- google/adk/tests/integration/fixture/trip_planner_agent/test_files/test_config.json +0 -5
- google/adk/tests/integration/fixture/trip_planner_agent/test_files/trip_inquiry_sub_agent.test.json +0 -7
- google/adk/tests/integration/fixture/trip_planner_agent/trip_inquiry.test.json +0 -19
- google/adk/tests/integration/models/__init__.py +0 -14
- google/adk/tests/integration/models/test_google_llm.py +0 -65
- google/adk/tests/integration/test_callback.py +0 -70
- google/adk/tests/integration/test_context_variable.py +0 -67
- google/adk/tests/integration/test_evalute_agent_in_fixture.py +0 -76
- google/adk/tests/integration/test_multi_agent.py +0 -28
- google/adk/tests/integration/test_multi_turn.py +0 -42
- google/adk/tests/integration/test_single_agent.py +0 -23
- google/adk/tests/integration/test_sub_agent.py +0 -26
- google/adk/tests/integration/test_system_instruction.py +0 -177
- google/adk/tests/integration/test_tools.py +0 -287
- google/adk/tests/integration/test_with_test_file.py +0 -34
- google/adk/tests/integration/tools/__init__.py +0 -14
- google/adk/tests/integration/utils/__init__.py +0 -16
- google/adk/tests/integration/utils/asserts.py +0 -75
- google/adk/tests/integration/utils/test_runner.py +0 -97
- google/adk/tests/unittests/__init__.py +0 -14
- google/adk/tests/unittests/agents/__init__.py +0 -14
- google/adk/tests/unittests/agents/test_base_agent.py +0 -407
- google/adk/tests/unittests/agents/test_langgraph_agent.py +0 -191
- google/adk/tests/unittests/agents/test_llm_agent_callbacks.py +0 -138
- google/adk/tests/unittests/agents/test_llm_agent_fields.py +0 -231
- google/adk/tests/unittests/agents/test_loop_agent.py +0 -136
- google/adk/tests/unittests/agents/test_parallel_agent.py +0 -92
- google/adk/tests/unittests/agents/test_sequential_agent.py +0 -114
- google/adk/tests/unittests/artifacts/__init__.py +0 -14
- google/adk/tests/unittests/artifacts/test_artifact_service.py +0 -276
- google/adk/tests/unittests/auth/test_auth_handler.py +0 -575
- google/adk/tests/unittests/conftest.py +0 -73
- google/adk/tests/unittests/fast_api/__init__.py +0 -14
- google/adk/tests/unittests/fast_api/test_fast_api.py +0 -269
- google/adk/tests/unittests/flows/__init__.py +0 -14
- google/adk/tests/unittests/flows/llm_flows/__init__.py +0 -14
- google/adk/tests/unittests/flows/llm_flows/_test_examples.py +0 -142
- google/adk/tests/unittests/flows/llm_flows/test_agent_transfer.py +0 -311
- google/adk/tests/unittests/flows/llm_flows/test_functions_long_running.py +0 -244
- google/adk/tests/unittests/flows/llm_flows/test_functions_request_euc.py +0 -346
- google/adk/tests/unittests/flows/llm_flows/test_functions_sequential.py +0 -93
- google/adk/tests/unittests/flows/llm_flows/test_functions_simple.py +0 -258
- google/adk/tests/unittests/flows/llm_flows/test_identity.py +0 -66
- google/adk/tests/unittests/flows/llm_flows/test_instructions.py +0 -164
- google/adk/tests/unittests/flows/llm_flows/test_model_callbacks.py +0 -142
- google/adk/tests/unittests/flows/llm_flows/test_other_configs.py +0 -46
- google/adk/tests/unittests/flows/llm_flows/test_tool_callbacks.py +0 -269
- google/adk/tests/unittests/models/__init__.py +0 -14
- google/adk/tests/unittests/models/test_google_llm.py +0 -224
- google/adk/tests/unittests/models/test_litellm.py +0 -804
- google/adk/tests/unittests/models/test_models.py +0 -60
- google/adk/tests/unittests/sessions/__init__.py +0 -14
- google/adk/tests/unittests/sessions/test_session_service.py +0 -227
- google/adk/tests/unittests/sessions/test_vertex_ai_session_service.py +0 -246
- google/adk/tests/unittests/streaming/__init__.py +0 -14
- google/adk/tests/unittests/streaming/test_streaming.py +0 -50
- google/adk/tests/unittests/tools/__init__.py +0 -14
- google/adk/tests/unittests/tools/apihub_tool/clients/test_apihub_client.py +0 -499
- google/adk/tests/unittests/tools/apihub_tool/test_apihub_toolset.py +0 -204
- google/adk/tests/unittests/tools/application_integration_tool/clients/test_connections_client.py +0 -600
- google/adk/tests/unittests/tools/application_integration_tool/clients/test_integration_client.py +0 -630
- google/adk/tests/unittests/tools/application_integration_tool/test_application_integration_toolset.py +0 -345
- google/adk/tests/unittests/tools/google_api_tool/__init__.py +0 -13
- google/adk/tests/unittests/tools/google_api_tool/test_googleapi_to_openapi_converter.py +0 -657
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_auto_auth_credential_exchanger.py +0 -145
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_base_auth_credential_exchanger.py +0 -68
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_oauth2_exchanger.py +0 -153
- google/adk/tests/unittests/tools/openapi_tool/auth/credential_exchangers/test_service_account_exchanger.py +0 -196
- google/adk/tests/unittests/tools/openapi_tool/auth/test_auth_helper.py +0 -573
- google/adk/tests/unittests/tools/openapi_tool/common/test_common.py +0 -436
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test.yaml +0 -1367
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_spec_parser.py +0 -628
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_openapi_toolset.py +0 -139
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_operation_parser.py +0 -406
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_rest_api_tool.py +0 -966
- google/adk/tests/unittests/tools/openapi_tool/openapi_spec_parser/test_tool_auth_handler.py +0 -201
- google/adk/tests/unittests/tools/retrieval/__init__.py +0 -14
- google/adk/tests/unittests/tools/retrieval/test_vertex_ai_rag_retrieval.py +0 -147
- google/adk/tests/unittests/tools/test_agent_tool.py +0 -167
- google/adk/tests/unittests/tools/test_base_tool.py +0 -141
- google/adk/tests/unittests/tools/test_build_function_declaration.py +0 -277
- google/adk/tests/unittests/utils.py +0 -304
- google_adk-0.0.2.dist-info/RECORD +0 -308
- {google_adk-0.0.2.dist-info → google_adk-0.0.4.dist-info}/entry_points.txt +0 -0
- {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 & 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>
|
google/adk/tests/__init__.py
DELETED
@@ -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 +0,0 @@
|
|
1
|
-
gkcng
|
@@ -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
|