adk-python/tests/unittests/flows/llm_flows/test_functions_long_running.py
Selcuk Gun 41b33d4a0a Move public_utils to utils in tests
Renamed conflicting utils.py as testing_utils.py

PiperOrigin-RevId: 761715808
2025-05-21 16:35:11 -07:00

245 lines
7.8 KiB
Python

# Copyright 2025 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from google.adk.agents import Agent
from google.adk.tools import ToolContext
from google.adk.tools.long_running_tool import LongRunningFunctionTool
from google.genai.types import Part
from ... import testing_utils
def test_async_function():
responses = [
Part.from_function_call(name='increase_by_one', args={'x': 1}),
'response1',
'response2',
'response3',
'response4',
]
mockModel = testing_utils.MockModel.create(responses=responses)
function_called = 0
def increase_by_one(x: int, tool_context: ToolContext) -> int:
nonlocal function_called
function_called += 1
return {'status': 'pending'}
# Calls the first time.
agent = Agent(
name='root_agent',
model=mockModel,
tools=[LongRunningFunctionTool(func=increase_by_one)],
)
runner = testing_utils.InMemoryRunner(agent)
events = runner.run('test1')
# Asserts the requests.
assert len(mockModel.requests) == 2
# 1 item: user content
assert mockModel.requests[0].contents == [
testing_utils.UserContent('test1'),
]
increase_by_one_call = Part.from_function_call(
name='increase_by_one', args={'x': 1}
)
pending_response = Part.from_function_response(
name='increase_by_one', response={'status': 'pending'}
)
assert testing_utils.simplify_contents(mockModel.requests[1].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', pending_response),
]
# Asserts the function calls.
assert function_called == 1
# Asserts the responses.
assert testing_utils.simplify_events(events) == [
(
'root_agent',
Part.from_function_call(name='increase_by_one', args={'x': 1}),
),
(
'root_agent',
Part.from_function_response(
name='increase_by_one', response={'status': 'pending'}
),
),
('root_agent', 'response1'),
]
assert events[0].long_running_tool_ids
# Updates with another pending progress.
still_waiting_response = Part.from_function_response(
name='increase_by_one', response={'status': 'still waiting'}
)
events = runner.run(testing_utils.UserContent(still_waiting_response))
# We have one new request.
assert len(mockModel.requests) == 3
assert testing_utils.simplify_contents(mockModel.requests[2].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', still_waiting_response),
]
assert testing_utils.simplify_events(events) == [('root_agent', 'response2')]
# Calls when the result is ready.
result_response = Part.from_function_response(
name='increase_by_one', response={'result': 2}
)
events = runner.run(testing_utils.UserContent(result_response))
# We have one new request.
assert len(mockModel.requests) == 4
assert testing_utils.simplify_contents(mockModel.requests[3].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', result_response),
]
assert testing_utils.simplify_events(events) == [('root_agent', 'response3')]
# Calls when the result is ready. Here we still accept the result and do
# another summarization. Whether this is the right behavior is TBD.
another_result_response = Part.from_function_response(
name='increase_by_one', response={'result': 3}
)
events = runner.run(testing_utils.UserContent(another_result_response))
# We have one new request.
assert len(mockModel.requests) == 5
assert testing_utils.simplify_contents(mockModel.requests[4].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', another_result_response),
]
assert testing_utils.simplify_events(events) == [('root_agent', 'response4')]
# At the end, function_called should still be 1.
assert function_called == 1
def test_async_function_with_none_response():
responses = [
Part.from_function_call(name='increase_by_one', args={'x': 1}),
'response1',
'response2',
'response3',
'response4',
]
mockModel = testing_utils.MockModel.create(responses=responses)
function_called = 0
def increase_by_one(x: int, tool_context: ToolContext) -> int:
nonlocal function_called
function_called += 1
return 'pending'
# Calls the first time.
agent = Agent(
name='root_agent',
model=mockModel,
tools=[LongRunningFunctionTool(func=increase_by_one)],
)
runner = testing_utils.InMemoryRunner(agent)
events = runner.run('test1')
# Asserts the requests.
assert len(mockModel.requests) == 2
# 1 item: user content
assert mockModel.requests[0].contents == [
testing_utils.UserContent('test1'),
]
increase_by_one_call = Part.from_function_call(
name='increase_by_one', args={'x': 1}
)
assert testing_utils.simplify_contents(mockModel.requests[1].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
(
'user',
Part.from_function_response(
name='increase_by_one', response={'result': 'pending'}
),
),
]
# Asserts the function calls.
assert function_called == 1
# Asserts the responses.
assert testing_utils.simplify_events(events) == [
(
'root_agent',
Part.from_function_call(name='increase_by_one', args={'x': 1}),
),
(
'root_agent',
Part.from_function_response(
name='increase_by_one', response={'result': 'pending'}
),
),
('root_agent', 'response1'),
]
# Updates with another pending progress.
still_waiting_response = Part.from_function_response(
name='increase_by_one', response={'status': 'still waiting'}
)
events = runner.run(testing_utils.UserContent(still_waiting_response))
# We have one new request.
assert len(mockModel.requests) == 3
assert testing_utils.simplify_contents(mockModel.requests[2].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', still_waiting_response),
]
assert testing_utils.simplify_events(events) == [('root_agent', 'response2')]
# Calls when the result is ready.
result_response = Part.from_function_response(
name='increase_by_one', response={'result': 2}
)
events = runner.run(testing_utils.UserContent(result_response))
# We have one new request.
assert len(mockModel.requests) == 4
assert testing_utils.simplify_contents(mockModel.requests[3].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', result_response),
]
assert testing_utils.simplify_events(events) == [('root_agent', 'response3')]
# Calls when the result is ready. Here we still accept the result and do
# another summarization. Whether this is the right behavior is TBD.
another_result_response = Part.from_function_response(
name='increase_by_one', response={'result': 3}
)
events = runner.run(testing_utils.UserContent(another_result_response))
# We have one new request.
assert len(mockModel.requests) == 5
assert testing_utils.simplify_contents(mockModel.requests[4].contents) == [
('user', 'test1'),
('model', increase_by_one_call),
('user', another_result_response),
]
assert testing_utils.simplify_events(events) == [('root_agent', 'response4')]
# At the end, function_called should still be 1.
assert function_called == 1