Using Tools
Give your agents additional capabilities while maintaining a natural conversation flow.
Ultravox includes built-in tools and you can create custom tools.
Here are some of the things you can do with tools:
Communicate with the Outside world
Order Lookup
Knowledge Base
Create Support Case
Transfer Call
End Call
Any functionality you can encapsulate in a function can be exposed to your agents as a tool. Addtionally, Ultravox automatically calls the underlying function so you don't have to sweat gluing things together.
Unlike using tools with single-generation LLM APIs, Ultravox actually calls your tool. This means you need to do a bit more work upfront in defining tools with the proper authentication and parameters.
While tools are supported across multiple variants of the Ultravox model, using tools with smaller models (i.e. 8B) typically don’t work well. YMMV. See available models for more info.
Built-in Tools
Ultravox currently includes the following built-in tools.
Tool Name | Tool Action |
---|---|
hangUp | Terminates the call. You can also have your own custom tools end the call. |
playDtmfSounds | Plays dual-tone multi-frequency (a.k.a dialpad) tones. See DTMF for details on sending and receiving tones in your voice applications. |
Using a Built-in Tool
You give your AI agent access tools when you create the call or create a new call stage.
Note: Using built-in tools is the same as using any other custom durable tool that you have created except for one difference: you can override built-in tools by using the same name.
For example, if you created a durable tool named “hangUp” and then provide that tool by name (i.e. not by the toolId), then your tool would be used instead of the built-in hangUp tool.
Viewing Tools
To view all tools available to you, including built-in tools, use the List Tools API.
The List Tools API returns all available built-in tools along with any custom tools that you have created.
Custom Tools
Custom tools enable you to communicate with the outside world. Anything that you can do in a function can now be done by your agent via a custom tool.
Creating Your First Custom Tool
Let’s look at creating a tool that sends an email with a summary of the conversation.
There are three steps:
Define the Tool
We need to define the tool and provide it to our agent. Everything we provide here will be seen by the agent so we need to be thoughtful of names, descriptions, parameters, etc.
What’s happening here:
- We are adding
selectedTools
to the request body of theCreate Call
. - There’s a single tool named
sendConversationSummary
. - This tool requires a single dynamic parameter called
conversationSummary
that is passed in the request body. - The tool’s functionality is available via POST at the url
https://foo.bar/sendSummary
.
Implement the Function
Now that we’ve defined the tool, let’s implement the functionality. This is a simplified example using Express.js and imagines a generic email API provider.
This function does the following:
- Accepts the
conversationSummary
via a POST. - Passes the data along to another function (
sendEmail
) that will send it via email.
Instruct the Agent on Tool Use
The last thing we need to do is provide additional instructions to the agent on how to use the tool. This is done by adding to the systemPrompt
. Let’s update what we used in the first step.
We’ve updated the system prompt that is used when the Ultravox call is created to instruct the agent when and how to use the tool.
Server vs. Client Tools
You can implement your tools as either server (the tool’s functionality is exposed via a URL) or client (the tool’s functionality is implemented in your client application) tools. From the perspective of Ultravox Realtime, there is no difference between server or client tools. When the model chooses to call a tool, Ultravox will simply call the tool however it’s defined.
There are a couple things to note:
Tool Authentication
Ultravox has rich support for tools auth. When creating a tool, you must specify what is required for successful auth to the backend service.
Three methods for passing API keys are supported and are used when creating the tool:
Query Parameter
Header
HTTP Authentication
Bearer
) must be provided when the tool is created.You then pass in the key(s) in the authTokens
property of selectedTools
when creating a call.
Tool Parameters
Tool parameters define what gets passed in to your backend service when the tool is called. When creating a tool, parameters are defined as one of three types:
Dynamic
Static
Automatic
Dynamic Parameters
Dynamic parameters will have their values set by the model. Creating a dynamic parameter on a tool looks like this:
Parameter Overrides
You can choose to set static values for dynamic parameters when you start a call. The model won’t see any parameters that you override. When creating a call simply pass in the overrides with each tool:
Parameter overrides don’t make sense for temporary tools. Instead of overriding a dynamic parameter, use a static parameter instead.
Static Parameters
If you have parameters that are known at the time you create the tool, static parameters can be used.
Automatic Parameters
Automatic parameters are used when you want a consistent, predictable value (not generated by the model) but you don’t know the value when the tool is created.
There are currently two use cases:
- Call ID → Used for sending the current Ultravox call ID to the tool. Can be used to provide a unique identifier or for logging purposes. Example: you want to store the call ID in your CRM as part of the current caller’s record.
Use "knownValue": "KNOWN_PARAM_CALL_ID"
.
- Conversation History → Sends the entire conversation history up to that point to the tool. Primarily used for tools that create a new call stage or when you want a tool to do something with the messages. Example: Pre-process conversation history messages when changing call stage and dynamically updating the system prompt for the new stage.
Use "knownValue": "KNOWN_PARAM_CONVERSATION_HISTORY"
.
You can find more on Automatic Parameters in the Base Tool Definition.
Temporary vs. Durable Tools
Custom tools come in two varieties: temporary and durable. There is much more information below but there are a few things to consider right upfront:
Creation
No Functional Difference
Iteration Speed
Reuse & Collaboration
The distinction between temporary and durable only applies to custom tools. Built-in tools are always durable.
Temporary Tools
Temporary tools are created each time you create a new Call and exist exclusively within the context of that call. (Temporary tools aren’t visible in the List Tools response for example.)
Iteration is faster when using temporary tools because you don’t have to create/update/delete tools as you build out your application. You can simply adjust the JSON in the request body and start a new call.
Creating & Using Temporary Tools
Temporary tools are defined and passed in the request body of the Create Call
endpoint. They are available during the current call.
Durable Tools
In addition to temporary tools, Ultravox supports the creation of durable tools. There is no functional difference between durable and temporary tools within the context of a call.
Durable tools are persisted and can be reused across calls or applications. They shine once you have things dialed in, when you want to share tools across multiple applications, or if you have split responsibilities on the team.
Creating Durable Tools
The /tools
endpoint in the Ultravox API is for working with durable tools. You create durable tools either by uploading an OpenAPI spec or via the request body in the Create Tool
endpoint. Your OpenAPI spec must be either json
or yaml
format.
Using Durable Tools
To use a durable tool in a call, set the toolName
or toolId
field instead of using a temporaryTool
.
For example:
Additional Information
Changing Call State
For most tools, the response will include data you want the model to use (e.g. the results of a lookup). However, Ultravox has support for special tool actions that can end the call or change the call stage.
These tool actions require setting a special response type.
Response Type | Tool Action |
---|---|
hang-up | Terminates the call. In addition to having Ultravox end the call after periods of user inactivity, your custom tool can end the call. |
new-stage | Creates a new call stage. See here for more. |
How you set the response type depends on if you are using a server/HTTP or a client tool:
Server/HTTP Tools
Server tools must respond with the X-Ultravox-Response-Type
header set to either hang-up
or new-stage
Client Tools
For client tools, set responseType="hang-up"
or responseType="new-stage"
on your ClientToolResult
object.
Debugging
The Ultravox SDK enables viewing debug messages for any active call. These messages include tool calls.
Tool Definition Schema
The definition
object in the tool creation and update requests follows the BaseToolDefinition schema.
Here’s a breakdown of its main components:
description
(string): A clear, concise description of what the tool does.dynamicParameters
(array, optional): List of parameters that can be set by the AI model when using the tool. Each parameter is an object containining:name
(string): The name of the parameter.location
(string): Where the parameter is used (“PARAMETER_LOCATION_QUERY”, “PARAMETER_LOCATION_PATH”, “PARAMETER_LOCATION_HEADER”, “PARAMETER_LOCATION_BODY”).schema
(object): JSON Schema definition of the parameter. This typically includes things like type, description, enum values, format, other restrictions, etc.required
(boolean): Whether the parameter is required.
staticParameters
(array, optional): List of parameters that are always set to a known, fixed value when the tool is used. These are unconditionally added when the tool is invoked. These parameters are not exposed to or set by the model. Example: you use an API for various things but want to track which requests come from your Ultravox app so you always appendutm=ultravox
to the query parameters.automaticParameters
(array, optional): Additional parameters automatically set by the system. Used when the value is not known when the tool is created but that will be known when the tool is called. Example: you want to use the uniquecall_id
from ultravox as a key in your backend and you have the tool includecall_id
in the request body when your tool’s API is called.requirements
(object, optional): Any specific requirements for using the tool. Currently this is used for security (e.g. API keys or HTTP Auth).http
(object): Details for invoking the tool via HTTP. For server tools.baseUrlPattern
(string): The base URL pattern for the tool, possibly with placeholders for path parameters.httpMethod
(string): The HTTP method for the tool (e.g., “GET”, “POST”).
client
(object): Declares the tool as a client tool. Exactly one ofhttp
orclient
must be set for a tool.
Best Practices for Creating Tools
-
Clear Naming: Choose a descriptive and unique name for your tool that clearly indicates its function.
-
Detailed Description: Provide a comprehensive description of what the tool does, including any important details about its usage or limitations. This and the name will help the model decide when and how to use your tool.
-
Precise Parameters: Define your dynamic parameters carefully, ensuring that the AI model has all the information it needs to use the tool effectively.
-
Error Handling: Consider how your tool will handle errors or unexpected inputs, and document this behavior in the tool’s description.
-
Iterate Faster: Use temporary tools when you are building your application. Persist durable tools in the system when things have stabilized.
-
Version Control: When updating tools, consider creating a new version (e.g., “stock_price_v2”) rather than modifying the existing tool. This allows testing of the new tool before impacting new calls made with the prior version of the tool.
-
Security: Be mindful of security when creating tools, especially when they interact with external APIs. Use appropriate authentication methods and avoid exposing sensitive information.
-
Testing: Thoroughly test your tools before deploying them in production conversations to ensure they function as expected.