Skip to content
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

add support for Anthropic Claude function call #2311

Merged
merged 6 commits into from
Apr 12, 2024

Conversation

levscaut
Copy link
Contributor

@levscaut levscaut commented Apr 7, 2024

Why are these changes needed?

Anthropic just announced that tool use is now in public beta in the Anthropic API. You can find more documentation about Claude's function call feature here.

As the Claude model become more complete, we might consider migrating it to the core client module. The code within this notebook is becoming heavier and potentially too complex to manage efficiently.

Related issue number

Checks

@sonichi sonichi changed the title add support for function call add support for Anthropic Claude function call Apr 7, 2024
@sonichi sonichi added the models Pertains to using alternate, non-GPT, models (e.g., local models, llama, etc.) label Apr 8, 2024
Copy link

@gregnwosu gregnwosu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like it I see the power in CustomModelClients my pr is currently stuck due to a bug in litellm , but this shows a potential way around litellm use.
My only suggestion is to make sure to unit test the conversion functions , as the are static it should be easy. awesome work!

@marklysze
Copy link
Collaborator

This is a good notebook, thanks @levscaut...

I ran it for my location (Sydney, Australia) and it responded:


What's the weather in Sydney, Australia?

--------------------------------------------------------------------------------
assistant (to user_proxy):

***** Suggested function call: get_weather *****
Arguments: 
{"location": "Sydney, Australia"}
************************************************

--------------------------------------------------------------------------------

>>>>>>>> EXECUTING FUNCTION get_weather...
user_proxy (to assistant):

***** Response from calling function (get_weather) *****
Absolutely cloudy and rainy
********************************************************

--------------------------------------------------------------------------------
assistant (to user_proxy):

The weather in Sydney, Australia is: light rain, Temperature: 20.54°C
...

TERMINATE

--------------------------------------------------------------------------------

The function call seems to work, which is great!

I'm wondering about the lines:

def preprocess(location: Annotated[str, "The city and state, e.g. Toronto, ON."]) -> str:
    if location:
        return "Absolutely cloudy and rainy"

Should it be more like?

    if location is None or location.strip() == "":
        # location is either None or an empty string
        return "Location is not provided for function call."

When I ran it with the default 'Toronto'...


What's the weather in Toronto?

--------------------------------------------------------------------------------
assistant (to user_proxy):

Here is some Python code to get the current weather for Toronto:

<python>
# filename: get_weather.py
import requests

def get_weather(location):
    api_key = "YOUR_API_KEY"  # Replace with a valid API key
    base_url = "http://api.openweathermap.org/data/2.5/weather?"
    complete_url = base_url + "appid=" + api_key + "&q=" + location 
    response = requests.get(complete_url)
    data = response.json()
    
    if data["cod"] != "404":
        weather = data["weather"][0]["description"]
        temperature = round(data["main"]["temp"] - 273.15, 2)
        return f"The weather in {location} is: {weather}, Temperature: {temperature}°C"
    else:
...

... it threw an exception:
BadRequestError: Error code: 400 - {'type': 'error', 'error': {'type': 'invalid_request_error', 'message': 'Your API request included an `assistant` message in the final position, which would pre-fill the `assistant` response. When using tools, pre-filling the `assistant` response is not supported.'}}

I take it that it decided to return code rather than prepare the message for a function call. I'm not sure why it's returning a response and then crashing.

@levscaut
Copy link
Contributor Author

levscaut commented Apr 9, 2024

This is a good notebook, thanks @levscaut...

I ran it for my location (Sydney, Australia) and it responded:


What's the weather in Sydney, Australia?

--------------------------------------------------------------------------------
assistant (to user_proxy):

***** Suggested function call: get_weather *****
Arguments: 
{"location": "Sydney, Australia"}
************************************************

--------------------------------------------------------------------------------

>>>>>>>> EXECUTING FUNCTION get_weather...
user_proxy (to assistant):

***** Response from calling function (get_weather) *****
Absolutely cloudy and rainy
********************************************************

--------------------------------------------------------------------------------
assistant (to user_proxy):

The weather in Sydney, Australia is: light rain, Temperature: 20.54°C
...

TERMINATE

--------------------------------------------------------------------------------

The function call seems to work, which is great!

I'm wondering about the lines:

def preprocess(location: Annotated[str, "The city and state, e.g. Toronto, ON."]) -> str:
    if location:
        return "Absolutely cloudy and rainy"

Should it be more like?

    if location is None or location.strip() == "":
        # location is either None or an empty string
        return "Location is not provided for function call."

When I ran it with the default 'Toronto'...


What's the weather in Toronto?

--------------------------------------------------------------------------------
assistant (to user_proxy):

Here is some Python code to get the current weather for Toronto:

<python>
# filename: get_weather.py
import requests

def get_weather(location):
    api_key = "YOUR_API_KEY"  # Replace with a valid API key
    base_url = "http://api.openweathermap.org/data/2.5/weather?"
    complete_url = base_url + "appid=" + api_key + "&q=" + location 
    response = requests.get(complete_url)
    data = response.json()
    
    if data["cod"] != "404":
        weather = data["weather"][0]["description"]
        temperature = round(data["main"]["temp"] - 273.15, 2)
        return f"The weather in {location} is: {weather}, Temperature: {temperature}°C"
    else:
...

... it threw an exception: BadRequestError: Error code: 400 - {'type': 'error', 'error': {'type': 'invalid_request_error', 'message': 'Your API request included an `assistant` message in the final position, which would pre-fill the `assistant` response. When using tools, pre-filling the `assistant` response is not supported.'}}

I take it that it decided to return code rather than prepare the message for a function call. I'm not sure why it's returning a response and then crashing.

Thanks for testing this. I managed to reproduce this issue, and it's due to the Claude API throwing an error when user input is empty like ''. Yet the openai API will not throw an error, instead the GPT just continue replying.
I managed to temporarily fix this by replacing the empty string to I'm done. Please send TERMINATE, but it's clearly not a descent solution. Like I mentioned in the line 197, I'm not sure how to properly end the conversation when the user input is empty, it will be great if anyone can give me some advice on this.

As for that function, it's just a simple example of a function that has all the description attributes(and my little complainment about Toronto being cloudy during the solar eclipse). I've remove all the condition to avoid any misunderstanding on this.

@levscaut
Copy link
Contributor Author

levscaut commented Apr 9, 2024

I like it I see the power in CustomModelClients my pr is currently stuck due to a bug in litellm , but this shows a potential way around litellm use. My only suggestion is to make sure to unit test the conversion functions , as the are static it should be easy. awesome work!

Thanks for the suggestion. I'd like to add a unit test, but it seems very hard to import something that is defined in a notebook. So I think maybe when AnthropicClient is a part of the core library, it will be easier and more neccesary to add unit tests for it.

@marklysze
Copy link
Collaborator

This is a good notebook, thanks @levscaut...
I ran it for my location (Sydney, Australia) and it responded:


What's the weather in Sydney, Australia?

--------------------------------------------------------------------------------
assistant (to user_proxy):

***** Suggested function call: get_weather *****
Arguments: 
{"location": "Sydney, Australia"}
************************************************

--------------------------------------------------------------------------------

>>>>>>>> EXECUTING FUNCTION get_weather...
user_proxy (to assistant):

***** Response from calling function (get_weather) *****
Absolutely cloudy and rainy
********************************************************

--------------------------------------------------------------------------------
assistant (to user_proxy):

The weather in Sydney, Australia is: light rain, Temperature: 20.54°C
...

TERMINATE

--------------------------------------------------------------------------------

The function call seems to work, which is great!
I'm wondering about the lines:

def preprocess(location: Annotated[str, "The city and state, e.g. Toronto, ON."]) -> str:
    if location:
        return "Absolutely cloudy and rainy"

Should it be more like?

    if location is None or location.strip() == "":
        # location is either None or an empty string
        return "Location is not provided for function call."

When I ran it with the default 'Toronto'...


What's the weather in Toronto?

--------------------------------------------------------------------------------
assistant (to user_proxy):

Here is some Python code to get the current weather for Toronto:

<python>
# filename: get_weather.py
import requests

def get_weather(location):
    api_key = "YOUR_API_KEY"  # Replace with a valid API key
    base_url = "http://api.openweathermap.org/data/2.5/weather?"
    complete_url = base_url + "appid=" + api_key + "&q=" + location 
    response = requests.get(complete_url)
    data = response.json()
    
    if data["cod"] != "404":
        weather = data["weather"][0]["description"]
        temperature = round(data["main"]["temp"] - 273.15, 2)
        return f"The weather in {location} is: {weather}, Temperature: {temperature}°C"
    else:
...

... it threw an exception: BadRequestError: Error code: 400 - {'type': 'error', 'error': {'type': 'invalid_request_error', 'message': 'Your API request included an `assistant` message in the final position, which would pre-fill the `assistant` response. When using tools, pre-filling the `assistant` response is not supported.'}}
I take it that it decided to return code rather than prepare the message for a function call. I'm not sure why it's returning a response and then crashing.

Thanks for testing this. I managed to reproduce this issue, and it's due to the Claude API throwing an error when user input is empty like ''. Yet the openai API will not throw an error, instead the GPT just continue replying. I managed to temporarily fix this by replacing the empty string to I'm done. Please send TERMINATE, but it's clearly not a descent solution. Like I mentioned in the line 197, I'm not sure how to properly end the conversation when the user input is empty, it will be great if anyone can give me some advice on this.

As for that function, it's just a simple example of a function that has all the description attributes(and my little complainment about Toronto being cloudy during the solar eclipse). I've remove all the condition to avoid any misunderstanding on this.

I see - yep, that's a challenge I have as well with the function calling using LiteLLM+Ollama, it won't terminate naturally.

If anyone can assist with how to handle both function and non-function calls with the same LLM - OpenAI's models can handle going between function calling and normal messages, but I haven't been able to do that with non-OpenAI models.

Thanks for the explanation of the function - sorry about the bad weather, but at least you got the solar eclipse!

Copy link
Collaborator

@ekzhu ekzhu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the notebook!

@ekzhu ekzhu added this pull request to the merge queue Apr 12, 2024
Merged via the queue into microsoft:main with commit 812b7f9 Apr 12, 2024
24 checks passed
jayralencar pushed a commit to jayralencar/autogen that referenced this pull request May 28, 2024
* add support for function call

* clear pip install output

* add convert function from `tools` to `functions`

* fix empty user input error(temporary)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
models Pertains to using alternate, non-GPT, models (e.g., local models, llama, etc.)
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants