Skip to content

Sessions

A Session is a live, multi-turn conversation. Open one with agent.start_session(), send messages, and the context carries across turns.

from caw import Agent

agent = Agent(provider="claude_code", model="opus", reasoning="high")
agent.set_system_prompt("You are a security reviewer.")

with agent.start_session() as session:
    turn1 = session.send("Review src/auth.py for vulnerabilities")
    print(turn1.result)

    turn2 = session.send("Now check src/api.py")
    print(turn2.result)
# session.end() runs on exit and returns the full Trajectory

Using the session as a context manager is the easy path — __exit__ calls session.end(), which finalizes the trajectory, persists it, and stops any tool servers. If you don't use with, call session.end() yourself.

One-shot vs. session

For a single message, agent.completion(message) is a convenience wrapper that opens a session, sends once, and ends it:

traj = agent.completion("Explain this code")
print(traj.result)

Inspecting progress mid-session

session.trajectory is available during the session, not just after:

with agent.start_session() as session:
    session.send("Remember the number 42.")
    session.send("What number did I just tell you?")

    traj = session.trajectory
    print(f"Turns: {traj.num_turns}")
    print(f"Total tool calls: {traj.total_tool_calls}")
    print(f"Total tokens: {traj.usage.total_tokens}")

Async sends

send_async() runs the blocking send in a thread and processes overlapping calls in FIFO order, so you can do async work while a turn is in flight:

import asyncio

task = asyncio.create_task(session.send_async(prompt))
while not task.done():
    # ... do other async work ...
    await asyncio.sleep(0.5)
turn = await task

Interactive mode

agent.interactive(prompt) hands control to the user's terminal — stdin/stdout/stderr are inherited so the user talks to the agent directly, while caw captures a copy of the output. All three providers support it (Claude Code, Codex, and opencode), each launching its own full-screen TUI with your initial prompt.

Pass select_provider=True to choose which backend to launch at runtime: caw shows an arrow-key menu of the installed providers (↑/↓ to move, Enter to choose, q/Esc to cancel) and launches the one you pick, ignoring the agent's configured provider. Cancelling the menu returns an InteractiveResult with exit code 130 without launching anything. caw.installed_providers() exposes the same list (name + provider) for your own menus.

See examples/interactive.py:

"""Interactive mode — launch the agent and let the user take over.

Pass ``select_provider=True`` to pick which installed provider to launch from
an arrow-key menu (↑/↓ to move, Enter to choose, q/Esc to cancel) instead of
using the agent's configured provider.
"""

import sys

from caw import Agent


def main():
    # `select_provider` is taken from the first CLI arg: `python interactive.py pick`.
    pick = len(sys.argv) > 1 and sys.argv[1] in ("pick", "select", "--select-provider")

    agent = Agent()

    prompt = (
        "List the directories in the current directory, then wait for me to tell "
        "you which one to count the Python files in."
    )
    result = agent.interactive(prompt, capture_bytes=4096, select_provider=pick)

    print(f"\nExit code: {result.exit_code}")
    if result.session_id:
        print(f"Session ID: {result.session_id}")
    print(f"Captured {len(result.output)} chars of terminal output")


if __name__ == "__main__":
    main()

Auto-wait on usage limits

By default, when a provider reports a usage limit mid-session, send() sleeps until the limit resets and then resumes automatically — transparently to you. Disable it per agent with Agent(..., auto_wait=False) or globally with CAW_AUTOWAIT=0.

Persistence and resuming

Pass data_dir= to persist a session to disk, and grab a resume_handle to continue it later (even in another process). Those are covered in Resuming sessions and Persistence.