Skip to main content

ai12z WebSocket API Documentation

Table of Contents

  1. Introduction
  2. Prerequisites
  3. Connecting to the WebSocket API
  4. Sending Queries
  5. Handling Responses
  6. Example Client Implementation
  7. Advanced Topics
  8. Reference
  9. Troubleshooting
  10. Glossary
  11. About ai12z
  12. API Version

Introduction

The ai12z WebSocket API enables real-time communication between clients and the ai12z server, facilitating dynamic interactions such as submitting queries and receiving responses asynchronously. This documentation guides you through setting up and using the WebSocket API, complete with an example client implementation.

tip

Before creating something custom, be sure to check out the out of box Web Components and React Controls

Prerequisites

Obtaining an API Key

To use the ai12z WebSocket API, you need a valid apiKey. You can obtain an API key by registering on our Developer Portal and creating a new application.

Required Libraries

Ensure you include the following libraries in your project:

  • Socket.IO: For WebSocket communication.
    <script src="https://cdn.socket.io/4.0.1/socket.io.min.js"></script>
  • Showdown: For converting Markdown to HTML.
    <script src="https://cdn.jsdelivr.net/npm/showdown/dist/showdown.min.js"></script>

Connecting to the WebSocket API

URL Format

The WebSocket URL is constructed based on the environment you are working in. For example:

  • Production: wss://api.ai12z.net

Establishing a Connection

Use the io.connect method from the Socket.IO library to establish a WebSocket connection:

const endpoint = "wss://api.ai12z.net"
const socket = io.connect(endpoint)

Sending Queries

Sending Text Queries

To send a text query to the ai12z server, emit the evaluate_query event with the required data:

const data = {
apiKey: apiKey,
query: query,
conversationId: conversationId, // Optional, for follow-up queries
event: "evaluate_query",
base64Images: [], // Optional, if not sending images
}

socket.emit("evaluate_query", data)

Sending Queries with Images

To send a query along with images, include the base64Images array in your data payload. You can send multiple images.

Note: There is a maximum payload size of 16 MB per emit. If your payload exceeds this limit, resize your images to a width of around 1024 pixels and compress them before converting to Base64.

const data = {
apiKey: apiKey,
query: query,
conversationId: conversationId, // Optional
event: "evaluate_query",
base64Images: [
"....",
"data:image/png;base64,...",
],
includeTags: [], // Optional
excludeTags: [], // Optional
requestMetadata: {}, // Optional
}

socket.emit("evaluate_query", data)

Optional Fields

  • includeTags (Optional): An array of tags to include in the query context.
  • excludeTags (Optional): An array of tags to exclude from the query context.
  • requestMetadata (Optional): An object containing additional metadata for the request.

Handling Responses

Partial Responses

The server sends partial responses (tokens) via the response event. Accumulate these tokens in a buffer (e.g., a string variable) and convert the accumulated Markdown to HTML for display.

let markdownBuffer = ""

socket.on("response", function (event) {
markdownBuffer += event.data // Accumulate markdown data
updateResponseContainer()
})

Final Responses

The end_response event indicates the end of a response. It provides the complete answer and additional data.

socket.on("end_response", function (data) {
if (data.error) {
console.error("Error from server:", data.error)
// Handle error accordingly
return
}

conversationId = data.conversationId || conversationId
handleEndResponse(data)
})

Response Data Structure

The data object received in the end_response event includes the following fields:

FieldTypeDescription
answerStringThe complete answer from the AI, including hyperlinks, images, and videos.
formModelObjectIf applicable, the form model data for client-side rendering.
controlDataObjectData returned by the agent that bypasses the LLM.
controlTypeStringThe type of control data (e.g., form, carousel, custom).
titleStringFor AnswerAI, the most relevant title from the vector database.
linkStringFor AnswerAI, the most relevant link from the vector database.
descriptionStringFor AnswerAI, the most relevant description from the vector database.
relevanceScoreNumberFor AnswerAI, the relevance score from the vector database.
assetTypeStringThe type of asset (e.g., web, pdf, docx).
didAnswerBooleanIndicates if the AI provided an answer (true) or not (false).
contextObjectFor AnswerAI, the contextual data from the vector database.
insightIdStringUsed for tracking user feedback on the content (like/dislike).
errorStringContains error information if an error occurred; otherwise, null.
conversationIdStringThe conversation ID for maintaining context in follow-up queries.

Error Handling

Check the error field in the end_response data object to handle any errors returned by the server.

if (data.error) {
console.error("Error from server:", data.error)
// Display error message to the user
}

Example Client Implementation

Below is an example client that connects to the ai12z WebSocket API, sends a query, and handles the response.

HTML Structure

<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>ai12z WebSocket Client</title>
<!-- Include Socket.IO -->
<script src="https://cdn.socket.io/4.0.1/socket.io.min.js"></script>
<!-- Include Showdown for Markdown to HTML conversion -->
<script src="https://cdn.jsdelivr.net/npm/showdown/dist/showdown.min.js"></script>
</head>
<body>
<input type="text" id="queryInput" placeholder="Type your question here" />
<button onclick="askAI()">Ask AI</button>
<div id="responseContainer"></div>

<!-- Your JavaScript code will go here -->
<script src="client.js"></script>
</body>
</html>

JavaScript Code

Create a file named client.js and include the following code:

// Replace 'your-api-key' with your actual API key
const apiKey = "your-api-key"
let conversationId = ""
const endpoint = "wss://api.ai12z.net/" // Ensure the correct WebSocket path
let socket
let markdownBuffer = ""

// Initialize Markdown converter
const converter = new showdown.Converter()

function connectWebSocket() {
socket = io(endpoint, {
transports: ["websocket"], // Force WebSocket transport
secure: true, // Ensure secure connection
rejectUnauthorized: false, // If you're using self-signed certificates for SSL, this might help
})

socket.on("connect_error", (error) => {
console.error("WebSocket connection error:", error)
})

// Handle partial responses
socket.on("response", function (event) {
markdownBuffer += event.data // Accumulate markdown data
updateResponseContainer()
})

// Handle final response
socket.on("end_response", function (data) {
if (data.error) {
console.error("Error from server:", data.error)
return
}

conversationId = data.conversationId || conversationId
handleEndResponse(data)
})
}

// Send query to ai12z server
function askAI() {
const query = document.getElementById("queryInput").value
const data = {
apiKey: apiKey,
query: query,
conversationId: conversationId,
event: "evaluate_query",
base64Images: [], // Include images if needed
}

socket.emit("evaluate_query", data)
markdownBuffer = "" // Clear the markdown buffer for new response
updateResponseContainer()
}

// Update the response container with the accumulated Markdown converted to HTML
function updateResponseContainer() {
// Convert the accumulated markdownBuffer to HTML
const html = converter.makeHtml(markdownBuffer)

// Find the container in the DOM to display the response
const responseContainer = document.getElementById("responseContainer")

// Set the HTML content of the container to the converted HTML
responseContainer.innerHTML = html
}

// Handle the final response
function handleEndResponse(data) {
markdownBuffer = data.answer // Update markdown buffer with the complete response
updateResponseContainer()

if (data.formModel) {
console.log("Form Model:", data.formModel)
// Handle form rendering here
}

// Handle other control types if necessary
if (data.controlType) {
switch (data.controlType) {
case "carousel":
// Handle carousel rendering
break
case "custom":
// Handle custom control rendering
break
// Add more cases as needed
}
}
}

// Connect to WebSocket on page load
window.onload = connectWebSocket

Explanation

  1. HTML Elements:

    • Input Field: An input field (queryInput) for the user to type their query.
    • Ask AI Button: A button that triggers the askAI() function.
    • Response Container: A div (responseContainer) where the AI's response will be displayed.
  2. JavaScript Code:

    • Variables:
      • apiKey: Your API key.
      • conversationId: Maintains conversation context.
      • endpoint: The WebSocket endpoint URL.
      • socket: The WebSocket connection instance.
      • markdownBuffer: Accumulates partial responses.
      • converter: An instance of Showdown's Markdown converter.
    • Functions:
      • connectWebSocket(): Establishes the WebSocket connection and sets up event listeners.
      • askAI(): Sends the user's query to the ai12z server.
      • updateResponseContainer(): Converts the accumulated Markdown to HTML and updates the response container.
      • handleEndResponse(data): Processes the final response, updates the conversation ID, and handles any additional data like forms or custom controls.
  3. Event Listeners:

    • socket.on("response", callback): Handles partial responses.
    • socket.on("end_response", callback): Handles the final response and error checking.
  4. Handling Forms and Control Data:

    • If data.formModel is present, it indicates that the AI has provided a form to render on the client side.
    • data.controlType can be used to handle different types of controls like carousels or custom components.

Advanced Topics

Image Processing Before Sending

To ensure your images do not exceed the payload size limit (16 MB), resize and compress them before conversion to Base64.

function resizeAndCompressImage(file, maxWidth, callback) {
const reader = new FileReader()
reader.onload = function (event) {
const img = new Image()
img.onload = function () {
const canvas = document.createElement("canvas")
const scaleSize = maxWidth / img.width
canvas.width = maxWidth
canvas.height = img.height * scaleSize
const ctx = canvas.getContext("2d")
ctx.drawImage(img, 0, 0, canvas.width, canvas.height)
const compressedDataUrl = canvas.toDataURL("image/jpeg", 0.7) // Adjust quality as needed
callback(compressedDataUrl)
}
img.src = event.target.result
}
reader.readAsDataURL(file)
}

Maintaining Conversation Context

The conversationId is used to maintain the context of a conversation across multiple queries. If you provide a conversationId, the AI will consider previous interactions in its response. If omitted or set to an empty string, a new conversation context is started.

Reference

Data Object Definitions

Request Data Object (evaluate_query event):

FieldTypeRequiredDescription
apiKeyStringYesYour API key.
queryStringYesThe user's query or question.
conversationIdStringNoThe conversation ID for maintaining context.
eventStringYesShould be set to "evaluate_query".
base64ImagesArrayNoAn array of Base64-encoded images.
includeTagsArrayNoTags to include in the query context.
excludeTagsArrayNoTags to exclude from the query context.
requestMetadataObjectNoAdditional metadata for the request.

Event Listeners

  • response Event: Receives partial responses (tokens) from the server.

    socket.on("response", function (event) {
    // Handle partial response
    })
  • end_response Event: Indicates the end of a response and provides the complete answer.

    socket.on("end_response", function (data) {
    // Handle final response
    })

Things to consider

Consider using the ai12z WebComponents and React control, it handles so much of what you will build

  • Client Responsibility:
    • This code does not manage a chatBot.
    • Buttons have javaScript sendQuery, that your custom chatBot would need to handle
    • Form script needs to be created

Troubleshooting

  • Connection Errors: If you cannot establish a connection, check your internet connectivity and ensure that the endpoint URL is correct.

  • Authentication Failures: If you receive an authentication error, verify that your API key is valid and has not expired.

  • Payload Size Exceeded: If you encounter payload size errors, reduce the size of your images or split your data into smaller chunks.

  • Unhandled Errors: Always check the error field in the end_response event to handle any server-side errors.

Glossary

  • LLM (Large Language Model): A type of AI model that can understand and generate human-like text.

  • Vector Database: A database optimized for storing and querying high-dimensional vectors, often used in machine learning applications.

  • AnswerAI: A feature of ai12z that provides answers based on vector database searches.

  • Bubble: A message or response unit displayed in the user interface.

  • Control Data: Data returned by the agent that bypasses the Large Language Model.

  • Form Model: A data structure representing a form to be rendered on the client side.

About ai12z

The ai12z platform provides advanced AI capabilities, including natural language understanding and image processing. You can use it to build chatbots, virtual assistants, and other AI-driven applications.

API Version

This documentation refers to ai12z WebSocket API version 1.0. Ensure that your client library is compatible with this version.


By following this guide, you should be able to integrate the ai12z WebSocket API into your application, handle real-time communication, and process AI responses effectively. If you have any questions or need further assistance, please refer to our Developer Support page.