-
Notifications
You must be signed in to change notification settings - Fork 381
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Async Function Support for Tools Parameter in GenerativeModel #632
base: main
Are you sure you want to change the base?
Conversation
Thanks for your pull request! It looks like this may be your first contribution to a Google open source project. Before we can look at your pull request, you'll need to sign a Contributor License Agreement (CLA). View this failed invocation of the CLA check for more information. For the most up to date status, view the checks section at the bottom of the pull request. |
Thanks! Note we do need you to sign the CLA before we can move the PR farther along. |
appreciate it! yeah i saw the notification, signed it rightaway🫡 |
Hi! This is to remind that i have signed The CLA Form |
Thanks for the reminder! I'll review today. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Okay, I don't think this works yet.
I think there are two ways to fix this:
.1
I think we need to have the two types, sync and async, and then here (in the async function handler) we need to check the type of the callable and await
it, or not:
fr = tools_lib(fc) |
Or .2
await
it or not... the other option is use asyncio.to_thread
to make all functions awaitable in the async
function handler.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This code is probably insufficiently tested (to begin with, my fault, not yours), can you add a test, or try this out in a colab notebook and share that?
You should be able to install from your branch using pip install git+https://github.com/somwrks/generative-ai-python/tree/main
yes i agree with the first approach, i was essentially working with my project which is basically a discord bot working with different agents to automate actions. That is where i found this bug or thing that it won't allow async functions to work well. I'll update the changes and create another request from google collab with demo example aswellas my main project which is significantly larger. Does that sound good? |
approached this with different approach of handling async and sync functions in the beginning by checking the type and then running a separate nesting async loop function for each tool class CallableFunctionDeclaration(FunctionDeclaration):
def __init__(
self,
*,
name: str,
description: str,
parameters: dict[str, Any] | None = None,
function: Callable[..., Any],
):
super().__init__(name=name, description=description, parameters=parameters)
self.function = function
self.is_async = inspect.iscoroutinefunction(function)
def __call__(self, fc: protos.FunctionCall) -> protos.FunctionResponse:
"""Handles both sync and async function calls transparently"""
try:
# Get or create event loop
try:
loop = asyncio.get_running_loop()
except RuntimeError:
loop = asyncio.new_event_loop()
asyncio.set_event_loop(loop)
# Execute function based on type
if self.is_async:
result = loop.run_until_complete(self._run_async(fc))
else:
result = self.function(**fc.args)
# Format response
if not isinstance(result, dict):
result = {"result": result}
return protos.FunctionResponse(name=fc.name, response=result)
except Exception as e:
return protos.FunctionResponse(
name=fc.name,
response={"error": str(e), "type": type(e).__name__}
)
async def _run_async(self, fc: protos.FunctionCall):
"""Helper method to run async functions"""
return await self.function(**fc.args) class FunctionLibrary:
def __init__(self, tools: Iterable[ToolType]):
tools = _make_tools(tools)
self._tools = list(tools)
self._index = {}
for tool in self._tools:
for declaration in tool.function_declarations:
name = declaration.name
if name in self._index:
raise ValueError(
f"Invalid operation: A `FunctionDeclaration` named '{name}' is already defined. "
"Each `FunctionDeclaration` must have a unique name. Please use a different name."
)
self._index[declaration.name] = declaration
def __getitem__(
self, name: str | protos.FunctionCall
) -> FunctionDeclaration | protos.FunctionDeclaration:
if not isinstance(name, str):
name = name.name
return self._index[name]
def __call__(self, fc: protos.FunctionCall) -> protos.Part:
declaration = self[fc]
if not callable(declaration):
return None
response = declaration(fc)
if response is None:
return None
return protos.Part(function_response=response)
def to_proto(self):
return [tool.to_proto() for tool in self._tools]
ToolsType = Union[Iterable[ToolType], ToolType] Added 6 test cases for each connected async and sync functions simultaneouslyimport google.generativeai as genai
import asyncio
import time
from typing import List, Dict, Any, Callable, Union, Awaitable
import nest_asyncio
import random
from datetime import datetime
nest_asyncio.apply()
# Async functions for operations that would typically be I/O bound
async def get_weather(city: str) -> Dict[str, Any]:
"""Simulate getting weather data"""
await asyncio.sleep(1) # Simulate API call
weather_conditions = ["Sunny", "Cloudy", "Rainy", "Partly Cloudy"]
return {
"city": city,
"temperature": random.randint(0, 35),
"condition": random.choice(weather_conditions),
"humidity": random.randint(30, 90)
}
async def fetch_data(query: str) -> Dict[str, Any]:
"""Simulate fetching data"""
await asyncio.sleep(1)
return {
"query": query,
"timestamp": datetime.now().isoformat(),
"result": f"Sample data for {query}"
}
# Regular synchronous functions for simple operations
def calculate_distance(city1: str, city2: str) -> float:
"""Calculate distance between cities"""
return random.uniform(100, 1000)
def fetch_user_data(user_id: str) -> Dict[str, Any]:
"""Get user data"""
return {
"user_id": user_id,
"name": "Sample User",
"last_active": datetime.now().isoformat()
}
def get_city_info(city: str) -> Dict[str, Any]:
"""Get city information"""
return {
"city": city,
"population": random.randint(100000, 10000000),
"country": "Sample Country"
}
def process_image(image_path: str) -> Dict[str, Any]:
"""Process image"""
return {
"image_path": image_path,
"dimensions": "1920x1080",
"format": "jpg",
"analysis": "Sample image analysis"
}
def analyze_text(text: str) -> Dict[str, Any]:
"""Analyze text"""
return {
"text": text,
"sentiment": random.choice(["positive", "negative", "neutral"]),
"word_count": len(text.split())
}
class AIAssistant:
def __init__(self, model_name: str = "gemini-1.5-flash", api_key: str = None):
self.model_name = model_name
if api_key:
genai.configure(api_key=api_key)
self.model = genai.GenerativeModel(model_name)
self.tools: Dict[str, Callable] = {}
self._register_default_tools()
def register_tool(self, name: str, func: Callable):
"""Register a new tool"""
self.tools[name] = func
def _register_default_tools(self):
"""Register built-in tools"""
self.register_tool("get_weather", get_weather)
self.register_tool("fetch_data", fetch_data)
self.register_tool("calculate_distance", calculate_distance)
self.register_tool("fetch_user_data", fetch_user_data)
self.register_tool("get_city_info", get_city_info)
self.register_tool("process_image", process_image)
self.register_tool("analyze_text", analyze_text)
async def _execute_tool(self, tool_name: str, *args, **kwargs) -> Any:
"""Execute a tool and handle both sync and async functions"""
if tool_name not in self.tools:
raise ValueError(f"Tool {tool_name} not found")
tool = self.tools[tool_name]
if asyncio.iscoroutinefunction(tool):
return await tool(*args, **kwargs)
return tool(*args, **kwargs)
def _parse_required_tools(self, response: str) -> Dict[str, List[Any]]:
"""Parse model response to determine which tools to execute"""
required_tools = {}
if "weather" in response.lower():
required_tools["get_weather"] = ["New York"]
if "distance" in response.lower():
required_tools["calculate_distance"] = ["Tokyo", "Osaka"]
if "process" in response.lower() and "image" in response.lower():
required_tools["process_image"] = ["example.jpg"]
if "user" in response.lower():
required_tools["fetch_user_data"] = ["sample_user"]
return required_tools
async def process_request(self, prompt: str) -> str:
"""Process user request and execute appropriate tools"""
try:
response = self.model.generate_content(
prompt,
generation_config={
"temperature": 0.7,
"top_p": 0.8,
"top_k": 40,
"max_output_tokens": 1024
}
)
required_tools = self._parse_required_tools(response.text)
# Execute tools and gather results
results = {}
for tool_name, args in required_tools.items():
results[tool_name] = await self._execute_tool(tool_name, *args)
final_response = self.model.generate_content(
f"{prompt}\nTool Results: {results}",
generation_config={"temperature": 0.7}
)
return final_response.text
except Exception as e:
return f"Error processing request: {str(e)}"
async def main():
assistant = AIAssistant(
model_name="gemini-1.5-flash",
api_key="x"
)
prompts = [
"What's the weather in New York?",
"Calculate the distance between Tokyo and Osaka",
"Process this weather data image and analyze the trends",
"What is the user's data?",
]
for prompt in prompts:
print(f"\nPrompt: {prompt}")
response = await assistant.process_request(prompt)
print(f"Response: {response}")
if __name__ == "__main__":
asyncio.run(main())``` Result-
|
Hey! i was wondering, what do you think of this? |
Description of the change
This change implements proper async function handling in the
GenerativeModel
class by modifying theCallableFunctionDeclaration
andFunctionLibrary
classes. The implementation adds support for detecting and properly awaiting async functions when they are passed as tools, resolving runtime errors related to unhandled coroutines.Key modifications include:
inspect.iscoroutinefunction()
CallableFunctionDeclaration.__call__
FunctionLibrary
Motivation
The current implementation fails to properly handle async functions when passed as tools to the GenerativeModel, resulting in runtime errors such as "coroutine was never awaited" and incorrect protobuf message conversion. This change is required to enable developers to use async functions with the GenerativeModel's tools parameter, allowing integration with asynchronous APIs and services.
Type of change
Bug fix, Feature Request
Checklist
git pull --rebase upstream main
).