Sample code for 30+ languages & platforms
Unicode C++

Streaming AI with Automatic JavaScript AI Tool Function Calling

See more AI Examples

Demonstrates how to get AI responses in streaming mode, including possible tool function calling using Chilkat with embedded Chilkat.Js JavaScript. Automatic JavaScript tool calls are characterized by:
  • Tool function implementations are in JavaScript.
  • The JavaScript also provides a tool registry and permissions.
  • The JavaScript runs embedded within your application using Chilkat.Js.
  • Tool calls are handled entirely within Chilkat. Your application does not need to manually check for function calls in the AI response. Internal to the Ask function, Chilkat will handle function calls by calling your registered JavaScript tools and will send results back to the AI model until the final response is received.

Chilkat Unicode C++ Downloads

Unicode C++
#include <CkStringBuilderW.h>
#include <CkAiW.h>

void ChilkatSample(void)
    {
    bool success = false;

    // ----------------------------------------------------------------------------------
    // The Javascript file loaded here is shown at the bottom of this page.
    // -----------------------------------------------------------------------------------

    CkStringBuilderW sbJs;
    success = sbJs.LoadFile(L"qa_data/js_tools/horoscope_tools.js",L"utf-8");
    if (success == false) {
        wprintf(L"%s\n",sbJs.lastErrorText());
        return;
    }

    CkAiW ai;

    // Register the tools that will be made available to the AI.
    bool evalOnly = false;
    bool allowAllKeyword = true;
    ai.RegisterJsTools(sbJs,evalOnly,allowAllKeyword);

    // 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;

    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"js_function_call",true)) {
                    // Call the JavaScript function.
                    // We don't need to find the exact function and make the call.  
                    // We can simply pass the sbDelta (which contains information about the function to be called and the args).
                    // Chilkat will find the registered JavaScript function, call it, and add the results to the conversation.
                    // We indicate that function calls were made, which results in a followup Ask (in the outer loop).
                    success = ai.StreamingJsToolCall(sbDelta);
                    if (success == false) {
                        // The JS function call failed.
                        streamingDone = true;
                    }
                    else {
                        madeFunctionCalls = true;
                    }

                }
                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());
    }