Skip to content

Commit

Permalink
User proxy documentation and fixes (microsoft#4401)
Browse files Browse the repository at this point in the history
* Fix handoff bug in user proxy agent

* Update documentation

---------
  • Loading branch information
ekzhu authored Nov 27, 2024
1 parent 52790a8 commit 7e589a1
Show file tree
Hide file tree
Showing 2 changed files with 51 additions and 6 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,38 @@


class UserProxyAgent(BaseChatAgent):
"""An agent that can represent a human user in a chat."""
"""An agent that can represent a human user through an input function.
This agent can be used to represent a human user in a chat system by providing a custom input function.
Args:
name (str): The name of the agent.
description (str, optional): A description of the agent.
input_func (Optional[Callable[[str], str]], Callable[[str, Optional[CancellationToken]], Awaitable[str]]): A function that takes a prompt and returns a user input string.
.. note::
Using :class:`UserProxyAgent` puts a running team in a temporary blocked
state until the user responds. So it is important to time out the user input
function and cancel using the :class:`~autogen_core.base.CancellationToken` if the user does not respond.
The input function should also handle exceptions and return a default response if needed.
For typical use cases that involve
slow human responses, it is recommended to use termination conditions
such as :class:`~autogen_agentchat.task.HandoffTermination` or :class:`~autogen_agentchat.task.SourceMatchTermination`
to stop the running team and return the control to the application.
You can run the team again with the user input. This way, the state of the team
can be saved and restored when the user responds.
See `Pause for User Input <https://microsoft.github.io/autogen/dev/user-guide/agentchat-user-guide/tutorial/teams.html#pause-for-user-input>`_ for more information.
"""

def __init__(
self,
name: str,
description: str = "a human user",
*,
description: str = "A human user",
input_func: Optional[InputFuncType] = None,
) -> None:
"""Initialize the UserProxyAgent."""
Expand All @@ -34,10 +60,12 @@ def produced_message_types(self) -> List[type[ChatMessage]]:
return [TextMessage, HandoffMessage]

def _get_latest_handoff(self, messages: Sequence[ChatMessage]) -> Optional[HandoffMessage]:
"""Find the most recent HandoffMessage in the message sequence."""
for message in reversed(messages):
if isinstance(message, HandoffMessage):
return message
"""Find the HandoffMessage in the message sequence that addresses this agent."""
if len(messages) > 0 and isinstance(messages[-1], HandoffMessage):
if messages[-1].target == self.name:
return messages[-1]
else:
raise RuntimeError(f"Handoff message target does not match agent name: {messages[-1].source}")
return None

async def _get_input(self, prompt: str, cancellation_token: Optional[CancellationToken]) -> str:
Expand Down
17 changes: 17 additions & 0 deletions python/packages/autogen-agentchat/tests/test_userproxy_agent.py
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,23 @@ def custom_input(prompt: str) -> str:
assert response.chat_message.source == "test_user"
assert response.chat_message.target == "assistant"

# The latest message if is a handoff message, it must be addressed to this agent.
messages = [
TextMessage(content="Initial message", source="assistant"),
HandoffMessage(content="Handing off to user for confirmation", source="assistant", target="other_agent"),
]
with pytest.raises(RuntimeError):
await agent.on_messages(messages, CancellationToken())

# No handoff message if the latest message is not a handoff message addressed to this agent.
messages = [
TextMessage(content="Initial message", source="assistant"),
HandoffMessage(content="Handing off to other agent", source="assistant", target="other_agent"),
TextMessage(content="Another message", source="other_agent"),
]
response = await agent.on_messages(messages, CancellationToken())
assert isinstance(response.chat_message, TextMessage)


@pytest.mark.asyncio
async def test_cancellation() -> None:
Expand Down

0 comments on commit 7e589a1

Please sign in to comment.