Async Iterators for Streaming¶
Async iterators provide asynchronous streaming of agent events through the stream_async
method. This approach is ideal for asynchronous frameworks like FastAPI, aiohttp, or Django Channels where you need fine-grained control over async execution flow.
For a complete list of available events including text generation, tool usage, lifecycle, and reasoning events, see the streaming overview.
Note: For synchronous event handling, consider callback handlers instead.
Note, Strands also offers an
invoke_async
method for non-iterative async invocations.
Basic Usage¶
import asyncio
from strands import Agent
from strands_tools import calculator
# Initialize our agent without a callback handler
agent = Agent(
tools=[calculator],
callback_handler=None
)
# Async function that iterators over streamed agent events
async def process_streaming_response():
agent_stream = agent.stream_async("Calculate 2+2")
async for event in agent_stream:
print(event)
# Run the agent
asyncio.run(process_streaming_response())
FastAPI Example¶
Here's how to integrate stream_async
with FastAPI to create a streaming endpoint:
from fastapi import FastAPI, HTTPException
from fastapi.responses import StreamingResponse
from pydantic import BaseModel
from strands import Agent
from strands_tools import calculator, http_request
app = FastAPI()
class PromptRequest(BaseModel):
prompt: str
@app.post("/stream")
async def stream_response(request: PromptRequest):
async def generate():
agent = Agent(
tools=[calculator, http_request],
callback_handler=None
)
try:
async for event in agent.stream_async(request.prompt):
if "data" in event:
# Only stream text chunks to the client
yield event["data"]
except Exception as e:
yield f"Error: {str(e)}"
return StreamingResponse(
generate(),
media_type="text/plain"
)
Example - Event Loop Lifecycle Tracking¶
This async stream processor illustrates the event loop lifecycle events and how they relate to each other. It's useful for understanding the flow of execution in the Strands agent:
from strands import Agent
from strands_tools import calculator
# Create agent with event loop tracker
agent = Agent(
tools=[calculator],
callback_handler=None
)
# This will show the full event lifecycle in the console
async for event in agent.stream_async("What is the capital of France and what is 42+7?"):
# Track event loop lifecycle
if event.get("init_event_loop", False):
print("🔄 Event loop initialized")
elif event.get("start_event_loop", False):
print("▶️ Event loop cycle starting")
elif "message" in event:
print(f"📬 New message created: {event['message']['role']}")
elif event.get("complete", False):
print("✅ Cycle completed")
elif event.get("force_stop", False):
print(f"🛑 Event loop force-stopped: {event.get('force_stop_reason', 'unknown reason')}")
# Track tool usage
if "current_tool_use" in event and event["current_tool_use"].get("name"):
tool_name = event["current_tool_use"]["name"]
print(f"🔧 Using tool: {tool_name}")
# Show only a snippet of text to keep output clean
if "data" in event:
# Only show first 20 chars of each chunk for demo purposes
data_snippet = event["data"][:20] + ("..." if len(event["data"]) > 20 else "")
print(f"📟 Text: {data_snippet}")
The output will show the sequence of events:
- First the event loop initializes (
init_event_loop
) - Then the cycle begins (
start_event_loop
) - New cycles may start multiple times during execution (
start
) - Text generation and tool usage events occur during the cycle
- Finally, the cycle completes (
complete
) or may be force-stopped