Unicode C++
Unicode C++
Streaming AI with Manual AI Tool Function Calling
See more AI Examples
Demonstrates how to get AI responses in streaming mode, including manual tool function calls.Chilkat Unicode C++ Downloads
#include <CkJsonObjectW.h>
#include <CkAiW.h>
#include <CkStringBuilderW.h>
void ChilkatSample(void)
{
bool success = false;
// Create the following JSON to define tool functions available for the AI to use.
// Note: You'll use the following JSON format regardless of the AI provider, whether
// it be ChatGPT, Gemini, Claude, Grok, etc. Chilkat automatically converts to the required
// format needed for a given AI provider.
// In this example, the application is providing a single function the AI may choose to call.
// {
// "tools": [
// {
// "name": "get_horoscope",
// "description": "Get today's horoscope for an astrological sign.",
// "parameters": {
// "properties": {
// "sign": {
// "type": "string",
// "description": "An astrological sign like Taurus or Aquarius"
// }
// }
// }
// }
// ]
// }
CkJsonObjectW jsonTools;
int toolIdx = 0;
jsonTools.put_I(toolIdx);
jsonTools.UpdateString(L"tools[i].name",L"get_horoscope");
jsonTools.UpdateString(L"tools[i].description",L"Get today's horoscope for an astrological sign.");
jsonTools.UpdateString(L"tools[i].parameters.properties.sign.type",L"string");
jsonTools.UpdateString(L"tools[i].parameters.properties.sign.description",L"An astrological sign like Taurus or Aquarius");
// More tools can be added as desired..
jsonTools.put_EmitCompact(false);
wprintf(L"%s\n",jsonTools.emit());
CkAiW ai;
// Register the tools that will be made available to the AI.
ai.RegisterManualTools(jsonTools);
// The provider can be "openai", "google", "claude", "grok", "mistral", "custom", etc.
ai.put_Provider(L"openai");
// Use your provider's API key.
ai.put_ApiKey(L"MY_API_KEY");
// Choose a model.
ai.put_Model(L"gpt-5-mini");
// Tool function calling must always occur within a conversation.
const wchar_t *conversation_name = L"convo_astrology";
const wchar_t *sysMessage = L"You are a helpful astrologer";
const wchar_t *devMessage = L"Respond only with markdown.";
ai.NewConvo(conversation_name,sysMessage,devMessage);
// Provide inputs
ai.InputAddText(L"What is my horoscope? I am an Aquarius.");
// Get the response in streaming mode.
ai.put_Streaming(true);
// In streaming mode, if we receive an AI event that is a request for tool use,
// we'll need to make the call to the JavaScript and then continue with a followup Ask,
// until the final response is received.
CkStringBuilderW sbEventName;
CkStringBuilderW sbDelta;
CkStringBuilderW sbFullResponse;
// When PollAi returns with an event, it's highly unlikely the
// call to NextAiEvent does not immediately return. Setting a max
// timeout is just a precaution..
int maxWaitMs = 5000;
CkJsonObjectW jsonFn;
bool finished = false;
int numAsks = 0;
// Set a max # of followup Asks to prevent any unexpected infinite looping.
while (!finished && (numAsks < 10)) {
// Send the request to the AI model.
success = ai.Ask(L"text");
if (success == false) {
wprintf(L"%s\n",ai.lastErrorText());
return;
}
bool madeFunctionCalls = false;
bool streamingDone = false;
while (!streamingDone) {
int result = ai.PollAi(false);
if (result < 0) {
wprintf(L"%s\n",ai.lastErrorText());
wprintf(L"Failed.\n");
return;
}
if (result > 0) {
// We have an event..
success = ai.NextAiEvent(maxWaitMs,sbEventName,sbDelta);
if (success == false) {
wprintf(L"%s\n",ai.lastErrorText());
return;
}
// Is this an event where the AI is requesting a function call?
if (sbEventName.ContentsEqual(L"function_call",true)) {
jsonFn.LoadSb(sbDelta);
// Note: Chilkat will convert responses from all AI providers to this format:
// {
// "function_call": [
// {
// "name": "get_horoscope",
// "call_id": "call_RYmeysYQFocFc7Z2ofkv61dW",
// "arguments": "{\"sign\":\"Aquarius\"}",
// "args": {
// "sign": "Aquarius"
// }
// }
// ]
// }
int numFnCalls = jsonFn.SizeOfArray(L"function_call");
int fn_idx = 0;
while ((fn_idx < numFnCalls)) {
jsonFn.put_I(fn_idx);
CkStringBuilderW sbFnName;
jsonFn.StringOfSb(L"function_call[i].name",sbFnName);
const wchar_t *callId = jsonFn.stringOf(L"function_call[i].call_id");
if (sbFnName.ContentsEqual(L"get_horoscope",true) == true) {
// The get_horoscope function (as defined above) has one argument named "sign".
const wchar_t *zodiac_sign = jsonFn.stringOf(L"function_call[i].args.sign");
wprintf(L"zodiac_sign = %s\n",zodiac_sign);
// Insert application code here to call your app's get_horoscope function, passing the zodiac_sign to it..
// For this example, we'll pretend the app's get_horoscope function returned the following:
const wchar_t *applicationFnCallResult = L"Aquarius: Next Tuesday you will befriend a baby otter.";
// Provide the tool call result as an input for the followup Ask.
ai.InputAddFnResult(callId,applicationFnCallResult);
madeFunctionCalls = true;
}
// Your application would add code to check for and handle each possible function call.
fn_idx = fn_idx + 1;
}
}
else {
if (!sbEventName.ContentsEqual(L"empty",true)) {
sbFullResponse.AppendSb(sbDelta);
if (sbEventName.ContentsEqual(L"null_terminator",true)) {
streamingDone = true;
}
}
}
}
else {
// No event arrived, so wait a short time rather than spin in a loop..
ai.SleepMs(100);
}
}
if (!madeFunctionCalls) {
finished = true;
}
numAsks = numAsks + 1;
}
wprintf(L"Full Response:\n");
wprintf(L"%s\n",sbFullResponse.getAsString());
}