By default, GPT-4 does parallel tool calls. In order to get it to call functions repeatedly based on previous function call results, you can use the runTools
API which came out of beta.
OpenAI calls it Automated Function Calls.
From their documentation: openai.chat.completions.runTools()
returns a Runner for automating function calls with chat completions. The runner automatically calls the JavaScript functions you provide and sends their results back to the API, looping as long as the model requests function calls.
They have a complete example in the repo, showcasing GPT-4 autonomously calling functions based on previous function call results.
#!/usr/bin/env -S npm run tsn -T
import OpenAI from "openai";
// gets API Key from environment variable OPENAI_API_KEY
const openai = new OpenAI();
const functions = [
{
name: "list",
description:
"list queries books by genre, and returns a list of names of books",
parameters: {
type: "object",
properties: {
genre: {
type: "string",
enum: ["mystery", "nonfiction", "memoir", "romance", "historical"],
},
},
},
function: list,
parse: JSON.parse,
},
{
name: "search",
description:
"search queries books by their name and returns a list of book names and their ids",
parameters: {
type: "object",
properties: {
name: { type: "string" },
},
},
function: search,
parse: JSON.parse,
},
{
name: "get",
description:
"get returns a book's detailed information based on the id of the book. Note that this does not accept names, and only IDs, which you can get by using search.",
parameters: {
type: "object",
properties: {
id: { type: "string" },
},
},
function: get,
parse: JSON.parse,
},
];
async function main() {
const runner = await openai.beta.chat.completions
.runFunctions({
model: "gpt-3.5-turbo",
messages: [
{
role: "system",
content:
"Please use our book database, which you can access using functions to answer the following questions.",
},
{
role: "user",
content:
"I really enjoyed reading To Kill a Mockingbird, could you recommend me a book that is similar and tell me why?",
},
],
functions,
})
.on("message", (msg) => console.log(msg))
.on("content", (diff) => process.stdout.write(diff));
const result = await runner.finalChatCompletion();
console.log(result);
console.log();
console.log(runner.messages);
}
const db = [
{
id: "a1",
name: "To Kill a Mockingbird",
genre: "historical",
description: `Compassionate, dramatic, and deeply moving, "To Kill A Mockingbird" takes readers to the roots of human behavior - to innocence and experience, kindness and cruelty, love and hatred, humor and pathos. Now with over 18 million copies in print and translated into forty languages, this regional story by a young Alabama woman claims universal appeal. Harper Lee always considered her book to be a simple love story. Today it is regarded as a masterpiece of American literature.`,
},
{
id: "a2",
name: "All the Light We Cannot See",
genre: "historical",
description: `In a mining town in Germany, Werner Pfennig, an orphan, grows up with his younger sister, enchanted by a crude radio they find that brings them news and stories from places they have never seen or imagined. Werner becomes an expert at building and fixing these crucial new instruments and is enlisted to use his talent to track down the resistance. Deftly interweaving the lives of Marie-Laure and Werner, Doerr illuminates the ways, against all odds, people try to be good to one another.`,
},
{
id: "a3",
name: "Where the Crawdads Sing",
genre: "historical",
description: `For years, rumors of the “Marsh Girl” haunted Barkley Cove, a quiet fishing village. Kya Clark is barefoot and wild; unfit for polite society. So in late 1969, when the popular Chase Andrews is found dead, locals immediately suspect her.
But Kya is not what they say. A born naturalist with just one day of school, she takes life's lessons from the land, learning the real ways of the world from the dishonest signals of fireflies. But while she has the skills to live in solitude forever, the time comes when she yearns to be touched and loved. Drawn to two young men from town, who are each intrigued by her wild beauty, Kya opens herself to a new and startling world—until the unthinkable happens.`,
},
];
async function list({ genre }: { genre: string }) {
return db
.filter((item) => item.genre === genre)
.map((item) => ({ name: item.name, id: item.id }));
}
async function search({ name }: { name: string }) {
return db
.filter((item) => item.name.includes(name))
.map((item) => ({ name: item.name, id: item.id }));
}
async function get({ id }: { id: string }) {
return db.find((item) => item.id === id)!;
}
main();
This higher level of abstraction allowed me to remove quite a bit of boilerplate code. Functionality wise it's close to what the Assistant API offers. I've tried the Assistant API in January 2024 and wasn't impressed. It didn't quite feel production ready yet.
Using runTools
we get a bit of the Assistant API's autonomity while using the battle tested completion API.
Erben Systems GmbH
Watterstrasse 81, c/o Sarbach Treuhand AG, 8105 Regensdorf, Switzerland
CHE-174.268.027 MwSt