Skip to content

strands.experimental.bidi.agent.agent

Bidirectional Agent for real-time streaming conversations.

Provides real-time audio and text interaction through persistent streaming connections. Unlike traditional request-response patterns, this agent maintains long-running conversations where users can interrupt, provide additional input, and receive continuous responses including audio output.

Key capabilities:

  • Persistent conversation connections with concurrent processing
  • Real-time audio input/output streaming
  • Automatic interruption detection and tool execution
  • Event-driven communication with model providers
class BidiAgent()

Defined in: src/strands/experimental/bidi/agent/agent.py:55

Agent for bidirectional streaming conversations.

Enables real-time audio and text interaction with AI models through persistent connections. Supports concurrent tool execution and interruption handling.

def __init__(model: BidiModel | str | None = None,
tools: list[str | AgentTool | ToolProvider] | None = None,
system_prompt: str | None = None,
messages: Messages | None = None,
record_direct_tool_call: bool = True,
load_tools_from_directory: bool = False,
agent_id: str | None = None,
name: str | None = None,
description: str | None = None,
hooks: list[HookProvider] | None = None,
state: AgentState | dict | None = None,
session_manager: "SessionManager | None" = None,
tool_executor: ToolExecutor | None = None,
**kwargs: Any)

Defined in: src/strands/experimental/bidi/agent/agent.py:62

Initialize bidirectional agent.

Arguments:

  • model - BidiModel instance, string model_id, or None for default detection.
  • tools - Optional list of tools with flexible format support.
  • system_prompt - Optional system prompt for conversations.
  • messages - Optional conversation history to initialize with.
  • record_direct_tool_call - Whether to record direct tool calls in message history.
  • load_tools_from_directory - Whether to load and automatically reload tools in the ./tools/ directory.
  • agent_id - Optional ID for the agent, useful for connection management and multi-agent scenarios.
  • name - Name of the Agent.
  • description - Description of what the Agent does.
  • hooks - Optional list of hook providers to register for lifecycle events.
  • state - Stateful information for the agent. Can be either an AgentState object, or a json serializable dict.
  • session_manager - Manager for handling agent sessions including conversation history and state. If provided, enables session-based persistence and state management.
  • tool_executor - Definition of tool execution strategy (e.g., sequential, concurrent, etc.).
  • **kwargs - Additional configuration for future extensibility.

Raises:

  • ValueError - If model configuration is invalid or state is invalid type.
  • TypeError - If model type is unsupported.
@property
def tool() -> _ToolCaller

Defined in: src/strands/experimental/bidi/agent/agent.py:175

Call tool as a function.

Returns:

ToolCaller for method-style tool execution.

Example:

agent = BidiAgent(model=model, tools=[calculator])
agent.tool.calculator(expression="2+2")
@property
def tool_names() -> list[str]

Defined in: src/strands/experimental/bidi/agent/agent.py:190

Get a list of all registered tool names.

Returns:

Names of all tools available to this agent.

async def start(invocation_state: dict[str, Any] | None = None) -> None

Defined in: src/strands/experimental/bidi/agent/agent.py:199

Start a persistent bidirectional conversation connection.

Initializes the streaming connection and starts background tasks for processing model events, tool execution, and connection management.

Arguments:

  • invocation_state - Optional context to pass to tools during execution. This allows passing custom data (user_id, session_id, database connections, etc.) that tools can access via their invocation_state parameter.

Raises:

RuntimeError: If agent already started.

Example:

await agent.start(invocation_state=\{
"user_id": "user_123",
"session_id": "session_456",
"database": db_connection,
})
async def send(input_data: BidiAgentInput | dict[str, Any]) -> None

Defined in: src/strands/experimental/bidi/agent/agent.py:230

Send input to the model (text, audio, image, or event dict).

Unified method for sending text, audio, and image input to the model during an active conversation session. Accepts TypedEvent instances or plain dicts (e.g., from WebSocket clients) which are automatically reconstructed.

Arguments:

  • input_data - Can be:

    • str: Text message from user
    • BidiInputEvent: TypedEvent
    • dict: Event dictionary (will be reconstructed to TypedEvent)

Raises:

  • RuntimeError - If start has not been called.
  • ValueError - If invalid input type.

Example:

await agent.send(“Hello”) await agent.send(BidiAudioInputEvent(audio=“base64…”, format=“pcm”, …)) await agent.send({“type”: “bidirectional_text_input”, “text”: “Hello”, “role”: “user”})

async def receive() -> AsyncGenerator[BidiOutputEvent, None]

Defined in: src/strands/experimental/bidi/agent/agent.py:281

Receive events from the model including audio, text, and tool calls.

Yields:

Model output events processed by background tasks including audio output, text responses, tool calls, and connection updates.

Raises:

  • RuntimeError - If start has not been called.
async def stop() -> None

Defined in: src/strands/experimental/bidi/agent/agent.py:297

End the conversation connection and cleanup all resources.

Terminates the streaming connection, cancels background tasks, and closes the connection to the model provider.

async def __aenter__(
invocation_state: dict[str, Any] | None = None) -> "BidiAgent"

Defined in: src/strands/experimental/bidi/agent/agent.py:306

Async context manager entry point.

Automatically starts the bidirectional connection when entering the context.

Arguments:

  • invocation_state - Optional context to pass to tools during execution. This allows passing custom data (user_id, session_id, database connections, etc.) that tools can access via their invocation_state parameter.

Returns:

Self for use in the context.

async def __aexit__(*_: Any) -> None

Defined in: src/strands/experimental/bidi/agent/agent.py:323

Async context manager exit point.

Automatically ends the connection and cleans up resources including when exiting the context, regardless of whether an exception occurred.

async def run(inputs: list[BidiInput],
outputs: list[BidiOutput],
invocation_state: dict[str, Any] | None = None) -> None

Defined in: src/strands/experimental/bidi/agent/agent.py:332

Run the agent using provided IO channels for bidirectional communication.

Arguments:

  • inputs - Input callables to read data from a source
  • outputs - Output callables to receive events from the agent
  • invocation_state - Optional context to pass to tools during execution. This allows passing custom data (user_id, session_id, database connections, etc.) that tools can access via their invocation_state parameter.

Example:

# Using model defaults:
model = BidiNovaSonicModel()
audio_io = BidiAudioIO()
text_io = BidiTextIO()
agent = BidiAgent(model=model, tools=[calculator])
await agent.run(
inputs=[audio_io.input()],
outputs=[audio_io.output(), text_io.output()],
invocation_state=\{"user_id": "user_123"}
)
# Using custom audio config:
model = BidiNovaSonicModel(
provider_config=\{"audio": \{"input_rate": 48000, "output_rate": 24000}}
)
audio_io = BidiAudioIO()
agent = BidiAgent(model=model, tools=[calculator])
await agent.run(
inputs=[audio_io.input()],
outputs=[audio_io.output()],
)