Tool/function calling

Introduction

One of the most interesting aspects of modern chat models is their ability to make use of external tools that are defined by the caller.

When making a chat request to the chat model, the caller advertises one or more tools (defined by their function name, description, and a list of expected arguments), and the chat model can choose to respond with one or more “tool calls”. These tool calls are requests from the chat model to the caller to execute the function with the given arguments; the caller is expected to execute the functions and “return” the results by submitting another chat request with the conversation so far, plus the results. The chat model can then use those results in formulating its response, or, it may decide to make additional tool calls.

Note that the chat model does not directly execute any external tools! It only makes requests for the caller to execute them. It’s easy to think that tool calling might work like this:

Diagram showing the wrong mental model of tool calls: a user initiates a request that flows to the assistant, which then runs the code, and returns the result back to the user.”
Diagram showing the wrong mental model of tool calls: a user initiates a request that flows to the assistant, which then runs the code, and returns the result back to the user.”

But in fact it works like this:

Diagram showing the correct mental model for tool calls: a user sends a request that needs a tool call, the assistant requests that the user runs that tool, returns the result to the assistant, which uses it to generate the final answer.
Diagram showing the correct mental model for tool calls: a user sends a request that needs a tool call, the assistant requests that the user runs that tool, returns the result to the assistant, which uses it to generate the final answer.

The value that the chat model brings is not in helping with execution, but with knowing when it makes sense to call a tool, what values to pass as arguments, and how to use the results in formulating its response.

library(ellmer)

Motivating example

Let’s take a look at an example where we really need an external tool. Chat models generally do not know the current time, which makes questions like these impossible.

chat <- chat_openai(model = "gpt-4o")
chat$chat("How long ago did Neil Armstrong touch down on the moon?")
#> Neil Armstrong touched down on the moon on July 20, 1969. As of 2023, that was 
#> 54 years ago.

Since the model doesn’t know what day it is, the result is incorrect.

Defining a tool function

The first thing we’ll do is define an R function that returns the current time.

#' Gets the current time in the given time zone.
#'
#' @param tz The time zone to get the current time in.
#' @return The current time in the given time zone.
get_current_time <- function(tz = "UTC") {
  format(Sys.time(), tz = tz, usetz = TRUE)
}

Note that we’ve gone through the trouble of creating roxygen2 comments. This isn’t necessary, but as we’ll see shortly, can make it a bit easier to generate a tool defintion.

To turn a function into a tool, we provide some additional metadata that the model will use:

get_current_time <- tool(
  get_current_time,
  name = "get_current_time",
  description = "Returns the current time.",
  arguments = list(
    tz = type_string(
      "Time zone to display the current time in. Defaults to `\"UTC\"`.",
      required = FALSE
    )
  )
)

This is a fair amount of code to write, even for such a simple function. Fortunately, you don’t have to write this by hand! I generated the above tool() call by calling create_tool_def(get_current_time), which uses an LLM to generate the tool() call for you. create_tool_def() is not perfect, so you must review the generated code before using it, but it is a big time-saver.

Note that a tool is just a special type of function so we can still call it:

get_current_time()
#> [1] "2025-06-25 16:53:23 UTC"

Registering and using tools

Now we need to give our chat object access to our tool. We do this with $register_tool():

chat$register_tool(get_current_time)

That’s all we need to do! Let’s retry our query:

chat$chat("How long ago did Neil Armstrong touch down on the moon?")
#> Neil Armstrong touched down on the moon on July 20, 1969. As of June 2025, that
#> was almost 56 years ago.

That’s correct! Without any further guidance, the chat model decided to call our tool function and successfully used its result in formulating its response.

If we print the chat we can see where the model decided to use the tool:

chat
#> <Chat OpenAI/gpt-4o turns=6 input=355 output=79 cost=$0.00>
#> ── user ────────────────────────────────────────────────────────────────────────
#> How long ago did Neil Armstrong touch down on the moon?
#> ── assistant [input=19 output=30 cost=$0.00] ───────────────────────────────────
#> Neil Armstrong touched down on the moon on July 20, 1969. As of 2023, that was 54 years ago.
#> ── user ────────────────────────────────────────────────────────────────────────
#> How long ago did Neil Armstrong touch down on the moon?
#> ── assistant [input=150 output=16 cost=$0.00] ──────────────────────────────────
#> [tool request (fc_03a8c37ecdd381e601690baca5278c81909e74f03d120da606)]: get_current_time(tz = "UTC")
#> ── user ────────────────────────────────────────────────────────────────────────
#> [tool result  (fc_03a8c37ecdd381e601690baca5278c81909e74f03d120da606)]: 2025-06-25 16:53:23 UTC
#> ── assistant [input=186 output=33 cost=$0.00] ──────────────────────────────────
#> Neil Armstrong touched down on the moon on July 20, 1969. As of June 2025, that was almost 56 years ago.

(Full disclosure: I originally tried this example with the default model of gpt-4o-mini and it got the tool calling right but the date math wrong, hence the explicit model="gpt-4o".)

This tool example was extremely simple, but you can imagine doing much more interesting things from tool functions: calling APIs, reading from or writing to a database, kicking off a complex simulation, or even calling a complementary GenAI model (like an image generator). Or if you are using ellmer in a Shiny app, you could use tools to set reactive values, setting off a chain of reactive updates.

Tool inputs and outputs

Remember that tool arguments come from the LLM, and tool results are returned to the LLM. This implies that you should keep both as simple as possible.

Inputs to a tool call, must be defined by type_boolean(), type_integer(), type_number(), type_string(), type_enum(), type_array(), or type_object(). We recommend keeping them as simple as possible, focusing on basic scalar types as much as you can.

The output of the tool call will be interpreted by the LLM, just as if you had typed that information into the data. That means you’ll generally want to produce text or other atomic vectors. For more complex data, ellmer will automatically serialize the result to JSON, which LLMs generally seem to be good at understanding. If you must have more direct control of the structure of the JSON that’s returned, you can return a JSON-serializable value wrapped in I(), which ellmer will leave alone until the entire request is JSON-serialized.

To show off these ideas, here’s a slightly more complicated example simulating a weather API that returns data for multiple cities at once. The get_weather() function returns a data frame that ellmer will automatically convert into JSON in row-major format, which our experiments suggest is good for LLMs.

get_weather <- tool(
  function(cities) {
    raining <- c(London = "heavy", Houston = "none", Chicago = "overcast")
    temperature <- c(London = "cool", Houston = "hot", Chicago = "warm")
    wind <- c(London = "strong", Houston = "weak", Chicago = "strong")

    data.frame(
      city = cities,
      raining = unname(raining[cities]),
      temperature = unname(temperature[cities]),
      wind = unname(wind[cities])
    )
  },
  name = "get_weather",
  description = "
    Report on weather conditions in multiple cities. For efficiency, request
    all weather updates using a single tool call
  ",
  arguments = list(
    cities = type_array(type_string(), "City names")
  )
)

Now we register and use it:

chat <- chat_openai()
#> Using model = "gpt-4.1".
chat$register_tool(get_weather)
chat$chat("Give me a weather update for London and Chicago")
#> Here is the weather update:
#> 
#> - London: Heavy rain, cool temperature, and strong winds.
#> - Chicago: Overcast skies, warm temperature, and strong winds.

We can print the chat to confirm that the model only performed a single tool call:

chat
#> <Chat OpenAI/gpt-4.1 turns=4 input=197 output=53 cost=$0.00>
#> ── user ────────────────────────────────────────────────────────────────────────
#> Give me a weather update for London and Chicago
#> ── assistant [input=67 output=18 cost=$0.00] ───────────────────────────────────
#> [tool request (fc_0695837b713996880169151e123e0881939637e2fd9d66180e)]: get_weather(cities = c("London", "Chicago"))
#> ── user ────────────────────────────────────────────────────────────────────────
#> [tool result  (fc_0695837b713996880169151e123e0881939637e2fd9d66180e)]: [{"city":"London","raining":"heavy","temperature":"cool","wind":"strong"},{"city":"Chicago","raining":"overcast","temperature":"warm","wind":"strong"}]
#> ── assistant [input=130 output=35 cost=$0.00] ──────────────────────────────────
#> Here is the weather update:
#> 
#> - London: Heavy rain, cool temperature, and strong winds.
#> - Chicago: Overcast skies, warm temperature, and strong winds.

Image and PDF tool output

ellmer allow tools to return image or PDF content that can be returned with the tool result, if the LLM or API supports vision capabilities.

Simply return a content_image_file(), content_pdf_file(), or similar content type from the tool function. For example, here’s a simple tool to screenshot a website:

screenshot_website <- tool(
  function(url) {
    tmpf <- withr::local_tempfile(fileext = ".png")
    webshot2::webshot(url, file = tmpf)
    content_image_file(tmpf)
  },
  name = "screenshot_website",
  description = "Take a screenshot of a website.",
  arguments = list(
    url = type_string("The URL of the website")
  )
)

You could use this tool to allow the LLM to “see” websites, like the tidyverse website:

chat <- chat_openai()
#> Using model = "gpt-4.1".
chat$register_tool(screenshot_website)
chat$chat("Describe the design aesthetic of https://tidyverse.org")
#> https://tidyverse.org screenshot completed
#> The design aesthetic of the Tidyverse website (https://tidyverse.org) is
#> clean, modern, and minimalistic, with several distinct features:
#>
#> - **Color Palette**: The overall site uses a lot of white space with navy
#>   and dark backgrounds for some elements, accentuated by the colorful
#>   hexagonal logos for various R packages.
#> - **Typography**: Simple, sans-serif fonts contribute to readability and
#>   a contemporary look.
#> - **Hexagonal Icons**: Prominent display of tidyverse package logos in
#>   hexagonal shapes, emphasizing the modular, package-oriented
#>   nature of the Tidyverse.
#> - **Layout**: A balanced, spacious two-column layout. The left side
#>   features graphic elements; the right side provides concise, text-based
#>   information.
#>
#> Overall, the design communicates clarity, ease of use, and a focus on
#> modern data science tools.