
When you deploy an AI agent into production, the first question isn’t “does it work?” — it’s “how do I know when it stops working?”
LLM-powered agents are probabilistic systems. They make decisions, call tools, and chain together operations in ways that are difficult to predict. Without observability, debugging a multi-step agent failure is like reading a novel with half the pages torn out.
In this post, I’ll walk you through building a multimodal async LangGraph agent with full observability powered by Google BigQuery. By the end, you’ll have:
- A working multimodal agent (text + images) with every event logged to BigQuery in real time
- SQL queries to analyze latency, token usage, tool errors, multimodal content, and cost
- BigQuery AI.GENERATE() for automated root cause analysis
- Production-ready configuration patterns
Let’s build it.
Why BigQuery Agent Analytics?
You might wonder: why not LangSmith, Langfuse, or another observability platform?
Those are great tools. But BigQuery gives you something unique:
- Serverless scale — millions of events, no infrastructure to manage
- SQL analytics — query agent behavior with tools your data team already knows
- Real-time streaming — events land via the Storage Write API within seconds
- AI on your logs — use AI.GENERATE() to analyze failures with Gemini directly in BigQuery
- Native multimodal support — capture and analyze multimodallogs natively with standard SQL
- Unified platform — agent logs live alongside your business data
The AsyncBigQueryCallbackHandler uses the same agent_events_v2 schema as the ADK BigQuery Agent Analytics plugin, giving you a standardized event format from day one.
Prerequisites
- A Google Cloud project with BigQuery API enabled
- A BigQuery dataset (the handler auto-creates the table)
- Authentication: gcloud auth application-default login
- Correct IAM permission setting
pip install "langchain-google-community[bigquery]" langchain-google-genai langgraph
Step 1: Define tools
We’ll build a multimodal travel assistant that can analyze images of landmarks. Clear docstrings help the LLM choose the right tool:
from langchain_core.tools import tool
@tool
def lookup_landmark(name: str) -> str:
"""Look up information about a landmark or location.
Args:
name: The name of the landmark or location.
"""
landmarks = {
"eiffel tower": (
"The Eiffel Tower is a wrought-iron lattice tower in Paris, France. "
"Built 1887-1889. Height: 330m. Visited by ~7 million people annually."
),
"golden gate bridge": (
"The Golden Gate Bridge is a suspension bridge in San Francisco, CA. "
"Opened 1937. Span: 1,280m. Color: International Orange."
),
"colosseum": (
"The Colosseum is an ancient amphitheatre in Rome, Italy. "
"Built 72-80 AD. Capacity: 50,000-80,000 spectators."
),
}
name_lower = name.lower().strip()
for key, info in landmarks.items():
if key in name_lower or name_lower in key:
return info
return f"No information found for '{name}'."
@tool
def get_travel_tips(destination: str) -> str:
"""Get travel tips for a destination.
Args:
destination: The travel destination city or country.
"""
tips = {
"paris": (
"Paris Travel Tips:\n"
"- Best time to visit: April-June or September-October\n"
"- Get a Paris Museum Pass for skip-the-line access\n"
"- The Metro is the fastest way to get around\n"
"- Tip: Visit the Eiffel Tower at sunset for the best views"
),
"rome": (
"Rome Travel Tips:\n"
"- Book Colosseum tickets in advance to skip lines\n"
"- Dress modestly for Vatican/church visits\n"
"- Best gelato is found away from tourist areas"
),
}
dest_lower = destination.lower().strip()
for key, info in tips.items():
if key in dest_lower or dest_lower in key:
return info
return f"No specific tips for '{destination}'."
Step 2: Configure the async callback handler
The AsyncBigQueryCallbackHandler intercepts every event in the LangGraph execution lifecycle and streams it to BigQuery:
from langchain_google_community.callbacks.bigquery_callback import (
AsyncBigQueryCallbackHandler,
BigQueryLoggerConfig,
)
config = BigQueryLoggerConfig(
batch_size=1, # Write immediately (dev); use 50+ in production
batch_flush_interval=0.5,
log_multi_modal_content=True, # Log image/audio/video content parts
)
handler = AsyncBigQueryCallbackHandler(
project_id="your-project-id",
dataset_id="agent_analytics",
table_id="agent_events_v2",
config=config,
graph_name="multimodal_travel_agent", # Enables LangGraph-specific event tracking
)
Key points:
- graph_name activates LangGraph integration — without it, you miss NODE_STARTING, NODE_COMPLETED, and graph-level tracking.
- log_multi_modal_content enables the content_parts field, which logs each part of a multimodal message with its MIME type (text/plain, image/jpeg, etc.) and storage mode.
- The handler is asyncio-native and safe for concurrent requests — each invocation gets its own trace via ContextVar.
- The table is auto-created with date partitioning and clustering by event_type, agent, and user_id.
Step 3: Build the async agent
Wire everything into an async LangGraph agent with a ReAct pattern. Gemini 2.5 Flash handles both text and image inputs natively:
from typing import Annotated, TypedDict
from langchain_core.messages import AIMessage, BaseMessage, HumanMessage
from langchain_google_genai import ChatGoogleGenerativeAI
from langgraph.graph import END, START, StateGraph
from langgraph.graph.message import add_messages
from langgraph.prebuilt import ToolNode
class AgentState(TypedDict):
messages: Annotated[list[BaseMessage], add_messages]
tools = [lookup_landmark, get_travel_tips]
llm = ChatGoogleGenerativeAI(
model="gemini-2.5-flash", project="your-project-id"
).bind_tools(tools)
async def agent_node(state: AgentState) -> dict:
response = await llm.ainvoke(state["messages"])
return {"messages": [response]}
def should_continue(state: AgentState) -> str:
last = state["messages"][-1]
return "tools" if isinstance(last, AIMessage) and last.tool_calls else END
workflow = StateGraph(AgentState)
workflow.add_node("agent", agent_node)
workflow.add_node("tools", ToolNode(tools))
workflow.add_edge(START, "agent")
workflow.add_conditional_edges("agent", should_continue, {"tools": "tools", END: END})
workflow.add_edge("tools", "agent")
agent = workflow.compile()
Step 4: Run with graph context tracking
Wrap the invocation in graph_context to capture the full execution lifecycle. For multimodal input, pass images via HumanMessage with image_url content parts:
import asyncio
metadata = {
"session_id": "multimodal-session-001",
"user_id": "demo-user",
"agent": "multimodal_travel_agent",
}
# Send an image alongside a text query
multimodal_message = HumanMessage(
content=[
{"type": "text", "text": "What landmark is this? Look it up and give me travel tips."},
{"type": "image_url", "image_url": {"url": "..."}},
]
)
async with handler.graph_context("multimodal_travel_agent", metadata=metadata):
result = await agent.ainvoke(
{"messages": [multimodal_message]},
config={"callbacks": [handler], "metadata": metadata},
)
print(result["messages"][-1].content)
await handler.shutdown() # Always shut down to flush remaining events
The graph_context emits GRAPH_START on entry and GRAPH_END (or GRAPH_ERROR) on exit, with total latency measured. The image is automatically logged in content_parts with its MIME type. You can also run queries concurrently — each gets its own trace:
tasks = [
run_query(text_message, "session-001"),
run_query(image_message, "session-002"),
]
responses = await asyncio.gather(*tasks)
What gets logged?
After running the agent, BigQuery contains a full execution trace. Here’s real output from our multimodal travel agent — it received a PNG image of the Eiffel Tower, identified the landmark, called lookup_landmark and get_travel_tips, then synthesized a response:
Real data from a live multimodal run against Gemini 2.5 Flash on February 14, 2026. Token estimates use the CEIL(LENGTH(TO_JSON_STRING(content)) / 4) approximation.
The trace tells a clear story: the first LLM call (image analysis + tool selection) took 2.5s with 2 content parts — a text query and an image/jpeg. Both tools executed in parallel in 4ms. The second LLM call (synthesis) received 5 content parts (original text + image + two tool results + AI tool call response) and completed in 1.7s. Total graph execution: 5.8s.
The Content Parts column reveals what makes multimodal logging powerful. Each LLM_REQUEST breaks down into individual parts with MIME types:
Without GCS configured, images are stored as [BASE64 IMAGE] placeholders. With GCS offloading enabled (gcs_bucket_name), images are uploaded to Cloud Storage and the uri field contains a gs:// reference — keeping BigQuery rows lightweight while preserving full media access.
Notice how token usage grows through the graph: the first LLM_REQUEST sends 611 tokens (user query + image), but the second sends 1,315 tokens (including tool results). This is typical of ReAct agents — context accumulates with each cycle. Monitoring this growth helps you catch runaway token consumption before it hits your budget.
Every event includes trace_id for correlation, span_id/parent_span_id for OpenTelemetry-compatible tracing, latency_ms, content (for token estimation), content_parts (for multimodal detail), and attributes (node name, tool name, graph name).
Analyze your agent with SQL
With structured events in BigQuery, you can answer questions that would be impossible with traditional logging.
Reconstruct a full execution trace
SELECT timestamp, event_type,
JSON_VALUE(attributes, '$.tool_name') AS tool,
JSON_VALUE(latency_ms, '$.total_ms') AS latency_ms, status
FROM `your-project.agent_analytics.agent_events_v2`
WHERE trace_id = 'your-trace-id'
ORDER BY timestamp;
Token usage and cost estimation
Estimate per-agent costs directly from logged content:
WITH token_estimates AS (
SELECT agent, event_type,
CEIL(LENGTH(TO_JSON_STRING(content)) / 4) AS estimated_tokens
FROM `your-project.agent_analytics.agent_events_v2`
WHERE DATE(timestamp) = CURRENT_DATE()
AND event_type IN ('LLM_REQUEST', 'LLM_RESPONSE')
)
SELECT agent,
SUM(IF(event_type='LLM_REQUEST', estimated_tokens, 0)) AS input_tokens,
SUM(IF(event_type='LLM_RESPONSE', estimated_tokens, 0)) AS output_tokens,
-- Gemini 2.5 Flash: $0.30/1M input, $2.50/1M output
ROUND(SUM(IF(event_type='LLM_REQUEST', estimated_tokens, 0)) * 0.0000003 +
SUM(IF(event_type='LLM_RESPONSE', estimated_tokens, 0)) * 0.0000025, 4) AS cost_usd
FROM token_estimates
GROUP BY agent ORDER BY cost_usd DESC;
Tool usage and error analysis
Find which tools fail most — high error rates may indicate tool description problems or upstream API issues:
SELECT
JSON_VALUE(attributes, '$.tool_name') AS tool_name,
COUNT(*) AS total_calls,
COUNTIF(event_type = 'TOOL_COMPLETED') AS successes,
COUNTIF(event_type = 'TOOL_ERROR') AS failures,
ROUND(COUNTIF(event_type = 'TOOL_ERROR') * 100.0 / NULLIF(COUNT(*), 0), 2) AS error_rate_pct,
ROUND(AVG(IF(event_type = 'TOOL_COMPLETED',
CAST(JSON_VALUE(latency_ms, '$.total_ms') AS FLOAT64), NULL)), 0) AS avg_latency_ms
FROM `your-project.agent_analytics.agent_events_v2`
WHERE event_type IN ('TOOL_COMPLETED', 'TOOL_ERROR')
AND DATE(timestamp) >= DATE_SUB(CURRENT_DATE(), INTERVAL 7 DAY)
GROUP BY tool_name ORDER BY total_calls DESC;
Multimodal content analysis
Find all events containing images and inspect their content parts:
SELECT
timestamp, event_type,
ARRAY_LENGTH(content_parts) AS num_parts,
(SELECT cp.mime_type FROM UNNEST(content_parts) cp
WHERE cp.mime_type LIKE 'image/%' LIMIT 1) AS image_type,
(SELECT cp.storage_mode FROM UNNEST(content_parts) cp
WHERE cp.mime_type LIKE 'image/%' LIMIT 1) AS storage_mode
FROM `your-project.agent_analytics.agent_events_v2`
WHERE EXISTS (
SELECT 1 FROM UNNEST(content_parts) cp WHERE cp.mime_type LIKE 'image/%'
)
ORDER BY timestamp DESC LIMIT 10;
Latency percentiles
SELECT event_type, agent,
ROUND(AVG(CAST(JSON_VALUE(latency_ms, '$.total_ms') AS FLOAT64)), 0) AS avg_ms,
ROUND(APPROX_QUANTILES(CAST(JSON_VALUE(latency_ms, '$.total_ms') AS FLOAT64), 100)[OFFSET(95)], 0) AS p95_ms
FROM `your-project.agent_analytics.agent_events_v2`
WHERE event_type IN ('LLM_RESPONSE', 'TOOL_COMPLETED', 'GRAPH_END')
AND JSON_VALUE(latency_ms, '$.total_ms') IS NOT NULL AND DATE(timestamp) = CURRENT_DATE()
GROUP BY event_type, agent ORDER BY avg_ms DESC;
BigQuery AI for root cause analysis
Here’s where BigQuery goes beyond traditional analytics. Use AI.GENERATE() to have Gemini analyze failures directly on your logs:
SELECT session_id,
AI.GENERATE(
('Analyze this agent execution log. Identify the root cause of failure and suggest fixes:\n\n',
full_trace),
connection_id => 'your-project.us.bqml_connection',
endpoint => 'gemini-2.5-flash'
).result AS root_cause_analysis
FROM (
SELECT session_id,
STRING_AGG(
CONCAT(CAST(timestamp AS STRING), ' | ', event_type, ' | ',
COALESCE(JSON_VALUE(attributes, '$.tool_name'), ''), ' | ',
COALESCE(status, ''), ' | ', SUBSTR(TO_JSON_STRING(content), 1, 500)),
'\n' ORDER BY timestamp
) AS full_trace
FROM `your-project.agent_analytics.agent_events_v2`
WHERE DATE(timestamp) = CURRENT_DATE()
AND session_id IN (
SELECT DISTINCT session_id FROM `your-project.agent_analytics.agent_events_v2`
WHERE status = 'ERROR' AND DATE(timestamp) = CURRENT_DATE())
GROUP BY session_id
) LIMIT 5;
This turns raw logs into actionable insights: “The agent failed because lookup_landmark was called with 'Tower' — the tool only recognizes full landmark names like 'Eiffel Tower'. Update the tool description to clarify expected input format."
You can also use BigQuery Conversational Analytics to ask questions in natural language — “Show me error rates by agent this week” — no SQL required.
Production configuration
Development — see everything immediately:
config = BigQueryLoggerConfig(batch_size=1, batch_flush_interval=0.5)
Production — optimize for throughput and cost:
config = BigQueryLoggerConfig(
batch_size=50,
batch_flush_interval=5.0,
queue_max_size=50000,
shutdown_timeout=30.0,
event_allowlist=[ # Only capture what matters
"LLM_RESPONSE", "LLM_ERROR",
"TOOL_COMPLETED", "TOOL_ERROR",
"GRAPH_END", "GRAPH_ERROR",
],
)
Multimodal agents — offload large media to GCS for production:
config = BigQueryLoggerConfig(
log_multi_modal_content=True,
gcs_bucket_name="my-agent-logs", # Offload images/audio/video to GCS
connection_id="us.my-bq-connection",
max_content_length=500 * 1024,
)
What’s next?
Once events are flowing to BigQuery:
- Analytics notebook — six-phase analysis framework (latency, errors, sessions, users, time-series) that runs in BigQuery Studio, Colab, or Vertex AI Workbench
- FastAPI monitoring dashboard — real-time event streaming with 25+ REST endpoints
For the complete API reference, see the official documentation. Full example suite: bigquery_callback examples.
Building Observable AI Agents: Real-Time Analytics for LangGraph with BigQuery Agent Analytics was originally published in Google Cloud – Community on Medium, where people are continuing the conversation by highlighting and responding to this story.
Source Credit: https://medium.com/google-cloud/building-observable-ai-agents-real-time-analytics-for-langgraph-with-bigquery-agent-analytics-9a1ac20837ec?source=rss—-e52cf94d98af—4
