https://github.com/openbb-finance/openbb-ai
https://github.com/openbb-finance/openbb-ai
Last synced: 5 months ago
JSON representation
- Host: GitHub
- URL: https://github.com/openbb-finance/openbb-ai
- Owner: OpenBB-finance
- License: mit
- Created: 2025-05-22T13:35:44.000Z (5 months ago)
- Default Branch: main
- Last Pushed: 2025-06-09T11:07:08.000Z (5 months ago)
- Last Synced: 2025-06-09T12:23:41.499Z (5 months ago)
- Language: Python
- Size: 76.2 KB
- Stars: 0
- Watchers: 1
- Forks: 1
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# OpenBB Custom Agent SDK
This package provides a set of pydantic models, tools and helpers to build
custom agents that are compatible with OpenBB Workspace.
For some example agents that demonstrate the full usage of the SDK, see the
[example agents repository](https://github.com/OpenBB-finance/agents-for-openbb).
## Features
- [Streaming Conversations](#message_chunk)
- [Reasoning steps / status updates](#reasoning_step)
- [Retrieve widget data from OpenBB Workspace](#get_widget_data)
- [Citations](#cite-and-citations)
- [Display tables](#table)
- [Create charts](#chart)
- [Widget priorities](#widget-priority)
To understand more about how everything works, see the
[Details](#details) section of this README.
## Usage
### Initial Setup
To use the OpenBB Custom Agent SDK, you need to install the package:
```bash
pip install openbb-ai
```
Your agent must consist of two endpoints:
1. A `query` endpoint. This is the main endpoint that will be called by the OpenBB Workspace. It returns responses using [Server-Sent Events](https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events/Using_server-sent_events) (SSEs).
2. An `agents.json` endpoint. This is the endpoint that will be called by the OpenBB Workspace to retrieve the agent's definition, and is what allows it to be added to the OpenBB Workspace.
All helper functions return Server-Sent Event (SSE) messages that should be streamed back to the OpenBB Workspace from your agent's execution loop. For example, using FastAPI with `EventSourceResponse` from `sse_starlette`:
```python
from fastapi import FastAPI
from sse_starlette import EventSourceResponse
from openbb_ai import (
reasoning_step,
message_chunk,
get_widget_data,
cite,
citations,
table,
chart,
)
app = FastAPI()
@app.get("/agents.json")
async def agents_json():
return JSONResponse(
content={
"": {
"name": "My Agent",
"description": "This is my agent",
"image": f"{AGENT_BASE_URL}/my-agent/logo.png",
"endpoints": {"query": f"{AGENT_BASE_URL}/query"}, # must match the query endpoint below
"features": {
"streaming": True, # must be True
"widget-dashboard-select": True, # Enable access to priority widgets
"widget-dashboard-search": True, # Enable access to non-priority widgets on current dashboard
},
}
}
)
@app.get("/query")
async def stream(request: QueryRequest):
async def execution_loop():
async def event_generator():
# Your agent's logic lives here
yield reasoning_step("Starting agent", event_type="INFO")
yield message_chunk("Hello, world!")
async for event in event_generator():
yield event.model_dump()
return EventSourceResponse(execution_loop())
```
### `QueryRequest`
`QueryRequest` is the most important Pydantic model and entrypoint for all
requests to the agent. It should be used as the request body for FastAPI
endpoints (if you're using FastAPI).
Refer to the `QueryRequest` model definition (`openbb_ai.models.QueryRequest`) for full details on its fields and validation.
**Agent backends are stateless**: full conversation history (messages), widget
definitions, context, URLs, and any other state will be included in each
`QueryRequest`. Each request to the agent can / should be handled independently
with all necessary data provided upfront.
Key fields:
- `messages`: List of messages to submit to the agent. Supports both chat (`LlmClientMessage`) and function call result (`LlmClientFunctionCallResultMessage`) messages.
- `widgets`: Optional `WidgetCollection` organizing widgets into `primary`, `secondary`, and `extra` groups.
- `context`: Optional additional context items (`RawContext`) to supplement processing. Yielded `table` and `chart` artifacts are automatically added to this list by OpenBB Workspace.
- `urls`: Optional list of URLs (up to 4) to retrieve and include as context.
- ... and more.
### `message_chunk`
Create a message chunk SSE to stream back chunks of text to OpenBB Workspace,
typically from the agent's streamed response.
```python
from openbb_ai.helpers import message_chunk
yield message_chunk("Hello, world!").model_dump()
```
### `reasoning_step`
OpenBB Workspace allows you to return "reasoning steps" (sometimes referred to
as "thought steps" or even "status updates") from your custom agent to the
front-end, at any point in the agent's execution. This is often useful for
providing updates and extra information to the user, particularly for
long-running queries, or for complicated workflows.
To send a reasoning step SSE to OpenBB Workspace, use the `reasoning_step`
helper function:
```python
from openbb_ai.helpers import reasoning_step
yield reasoning_step(
message="Processing data",
event_type="INFO",
details={"step": 1},
).model_dump()
```
### `get_widget_data`
Create a function call SSE that retrieves data from widgets on the OpenBB
Workspace.
```python
from openbb_ai.helpers import get_widget_data
from openbb_ai.models import WidgetRequest
widget_requests = [WidgetRequest(widget=..., input_arguments={...})]
yield get_widget_data(widget_requests).model_dump()
```
For more technical details on how this works, see the
[Function calling to OpenBB Workspace (to retrieve widget data)](#function-calling-to-openbb-workspace-to-retrieve-widget-data)
section of this README.
### `cite` and `citations`
Create citations for widgets to display on OpenBB Workspace. Use `cite` to
construct a `Citation` for a widget and `citations` to stream a collection of
citations as an SSE to the client.
```python
from openbb_ai.helpers import cite, citations
citation = cite(
widget=widget,
input_arguments={"param1": "value1", "param2": 123},
extra_details={"note": "Optional extra details"},
)
yield citations([citation]).model_dump()
```
### `table`
Create a table message artifact SSE to display a table as streamed in-line
agent output in OpenBB Workspace.
```python
from openbb_ai.helpers import table
yield table(
data=[
{"x": 1, "y": 2, "z": 3},
{"x": 2, "y": 3, "z": 4},
{"x": 3, "y": 4, "z": 5},
{"x": 4, "y": 5, "z": 6},
],
name="My Table",
description="This is a table of the data",
).model_dump()
```
### `chart`
Create a chart message artifact SSE to display various types of charts
(line, bar, scatter, pie, donut) as streamed in-line agent output in OpenBB
Workspace.
```python
from openbb_ai.helpers import chart
yield chart(
type="line",
data=[
{"x": 1, "y": 2},
{"x": 2, "y": 3},
{"x": 3, "y": 4},
{"x": 4, "y": 5},
],
x_key="x",
y_keys=["y"],
name="My Chart",
description="This is a chart of the data",
).model_dump()
yield chart(
type="pie",
data=[
{"amount": 1, "category": "A"},
{"amount": 2, "category": "B"},
{"amount": 3, "category": "C"},
{"amount": 4, "category": "D"},
],
angle_key="amount",
callout_label_key="category",
name="My Chart",
description="This is a chart of the data",
).model_dump()
```
### Widget Priority
Custom agents receive three widget types via the `QueryRequest.widgets` field:
- **Primary widgets**: Explicitly added by the user to the context
- **Secondary widgets**: Present on the active dashboard but not explicitly added
- **Extra widgets**: Any widgets added to OpenBB Workspace (visible or not)
Currently, only primary and secondary widgets are accessible to custom agents, with extra widget support coming soon.
The dashboard below shows a Management Team widget (primary/priority) and a Historical Stock Price widget (secondary):

If we inspect the `request.widgets` attribute of the `QueryRequest` object, we
can see the following was sent through to the custom agent:
```python
>>> request.widgets
WidgetCollection(
primary=[
Widget(
uuid=UUID('68ab6973-ed1a-45aa-ab20-efd3e016dd48'),
origin='OpenBB API',
widget_id='management_team',
name='Management Team',
description='Details about the management team of a company, including name, title, and compensation.',
params=[
WidgetParam(
name='symbol',
type='ticker',
description='The symbol of the asset, e.g. AAPL,GOOGL,MSFT',
default_value=None,
current_value='AAPL',
options=[]
)
],
metadata={
'source': 'Financial Modelling Prep',
'lastUpdated': 1746177646279
}
)
],
secondary=[
Widget(
uuid=UUID('bfa0aaaf-0b63-49b9-bb48-b13ef9db514b'),
origin='OpenBB API',
widget_id='eod_price',
name='Historical Stock Price',
description='Historical stock price data, including open, high, low, close, volume, etc.',
params=[
WidgetParam(
name='symbol',
type='ticker',
description='The symbol of the asset, e.g. AAPL,GOOGL,MSFT',
default_value=None,
current_value='AAPL',
options=[]
),
WidgetParam(
name='start_date',
type='date',
description='The start date of the historical data',
default_value='2023-05-02',
current_value='2023-05-02',
options=[]
)
],
metadata={
'source': 'Financial Modelling Prep',
'lastUpdated': 1746177655947
}
)
],
extra=[]
)
```
You can also see the parameter information of each widget in the `params` field
of the `Widget` object.
## Details
This section contains more specific technical details about how the various
components work together.
### Architecture
```
┌─────────────────────┐ ┌───────────────────────────────────────────┐
│ │ │ │
│ │ │ Agent │
│ │ │ (Backend) │
│ │ 1. HTTP POST │ │
│ OpenBB Workspace │ ───────────> │ ┌─────────────┐ ┌─────────────────┐ │
│ (Frontend) │ /query │ │ │ │ │ │
│ │ │ │ LLM │───>│ Function │ │
│ ┌───────────────┐ │ │ │ Processing │ │ Call │ │
│ │ Widget Data │ │ <─────────── │ │ │<───│ Processing │ │
│ │ Retrieval │ │ 2. Function │ │ │ │ │ │
│ └───────────────┘ │ Call SSE │ └─────────────┘ └─────────────────┘ │
│ ^ │ │ │
│ │ │ 3. HTTP POST │ │
│ └───────────│ ───────────> │ │
│ Execute & │ /query │ │
│ Return Results │ │ │
│ │ <─────────── │ │
│ │ 4. SSE │ │
│ │ (text chunks, │ │
│ │reasoning steps)│ │
└─────────────────────┘ └───────────────────────────────────────────┘
```
The architecture consists of two main components:
1. **OpenBB Workspace (Frontend)**: The user interface where queries are entered
2. **Agent (Backend)**: Programmed by you, handles the processing of queries, executing internal function calls, and returns answers
The frontend communicates with the backend via REST requests to the `query`
endpoint as defined in the `agent.json` schema.
### Function calling to OpenBB Workspace (to retrieve widget data)
When retrieving data from widgets on the OpenBB Workspace, your custom agent must
execute a **remote** function call, which gets interpreted by the OpenBB
Workspace. This is in contrast to **local** function calling, which is executed locally by the agent within its own runtime / environment.
Unlike local function calling, where the function is executed entirely on the
custom agent backend, a remote function call to the OpenBB Workspace is
partially executed on the OpenBB Workspace, and the results are sent back to the
custom agent backend.
Below is a timing diagram of how a remote function call to the OpenBB Workspace works:
```
OpenBB Workspace Custom Agent
│ │
│ 1. POST /query │
│ { │
│ messages: [...], │
│ widgets: {...} │
│ } │
│───────────────────────────────>│
│ │
│ 2. Function Call SSE │
│<───────────────────────────────│
│ (Connection then closed) │
│ │
│ 3. POST /query │
│ { │
│ messages: [ │
│ ...(original messages), │
│ function_call, │
│ function_call_result │
│ ], │
│ widgets: {...} │
│ } │
│───────────────────────────────>│
│ │
│ 4. SSEs (text chunks, │
│ reasoning steps, etc.) │
│<───────────────────────────────│
│ │
```
This is what happens "under-the-hood" when you yield from the `get_widget_data`
helper function, and close the connection: OpenBB Workspace executes the
function call (retrieving the widget data), and then sends a follow-up request
to the `query` endpoint of the agent, containing the function call and its
result.