Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adds final answer tool for all agents #31703

Merged
merged 5 commits into from
Jul 3, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 10 additions & 3 deletions docs/source/en/agents.md
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ We implement two versions of ReactJsonAgent:

![Framework of a React Agent](https://huggingface.co/datasets/huggingface/documentation-images/resolve/main/blog/open-source-llms-as-agents/ReAct.png)

For example, here is how a ReAct agent would work its way through the following question.
For example, here is how a ReAct Code agent would work its way through the following question.

```py3
>>> agent.run(
Expand Down Expand Up @@ -188,7 +188,7 @@ You can still authorize additional imports by passing the authorized modules as
>>> from transformers import ReactCodeAgent

>>> agent = ReactCodeAgent(tools=[], additional_authorized_imports=['requests', 'bs4'])
>>>agent.run("Could you get me the title of the page at url 'https://huggingface.co/blog'?")
>>> agent.run("Could you get me the title of the page at url 'https://huggingface.co/blog'?")

(...)
'Hugging Face – Blog'
Expand Down Expand Up @@ -256,6 +256,13 @@ agent = ReactJsonAgent(tools=[PythonInterpreterTool()], system_prompt="{your_cus
> Please make sure to define the `<<tool_descriptions>>` string somewhere in the `template` so the agent is aware
of the available tools.


### Inspecting an agent run

Here are a few useful attributes to inspect what happened after a run:
- `agent.logs` stores the fine-grained logs of the agent. At every step of the agent's run, everything gets stored in a dictionary that then is appended to `agent.logs`.
- Running `agent.write_inner_memory_from_logs()` creates an inner memory of the agent's logs for the LLM to view, as a list of chat messages. This method goes over each step of the log and only stores what it's interested in as a message: for instance, it will save the system prompt and task in separate messages, then for each step it will store the LLM output as a message, and the tool call output as another message. Use this if you want a higher-level view of what has happened - but not every log will be transcripted by this method.

## Tools

A tool is an atomic function to be used by an agent.
Expand Down Expand Up @@ -379,7 +386,7 @@ And the output:
`"The most downloaded model for the 'text-to-video' task is ByteDance/AnimateDiff-Lightning."`


### Manage agent toolbox
### Manage your agent's toolbox

If you have already initialized an agent, it is inconvenient to reinitialize it from scratch with a tool you want to use. With Transformers, you can manage an agent's toolbox by adding or replacing a tool.

Expand Down
3 changes: 1 addition & 2 deletions src/transformers/agents/agents.py
Original file line number Diff line number Diff line change
Expand Up @@ -337,6 +337,7 @@ def __init__(
self._toolbox.add_base_tools(add_python_interpreter=(self.__class__ == ReactJsonAgent))
else:
self._toolbox = Toolbox(tools, add_base_tools=add_base_tools)
self._toolbox.add_tool(FinalAnswerTool())

self.system_prompt = format_prompt_with_tools(
self._toolbox, self.system_prompt_template, self.tool_description_template
Expand Down Expand Up @@ -631,8 +632,6 @@ def __init__(
tool_description_template=tool_description_template,
**kwargs,
)
if "final_answer" not in self._toolbox.tools:
self._toolbox.add_tool(FinalAnswerTool())

def provide_final_answer(self, task) -> str:
"""
Expand Down
2 changes: 1 addition & 1 deletion src/transformers/agents/default_tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -180,7 +180,7 @@ def forward(self, code):

class FinalAnswerTool(Tool):
name = "final_answer"
description = "Provides a final answer to the given problem"
description = "Provides a final answer to the given problem."
inputs = {"answer": {"type": "text", "description": "The final answer to the problem"}}
output_type = "any"

Expand Down
18 changes: 13 additions & 5 deletions src/transformers/agents/prompts.py
Original file line number Diff line number Diff line change
Expand Up @@ -52,8 +52,9 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
To help you, I will give you access to a set of tools that you can use. Each tool is a Python function and has a description explaining the task it performs, the inputs it expects and the outputs it returns.
You should first explain which tool you will use to perform the task and for what reason, then write the code in Python.
Each instruction in Python should be a simple assignment. You can print intermediate results if it makes sense to do so.
In the end, use tool 'final_answer' to return your answer, its argument will be what gets returned.
You can use imports in your code, but only from the following list of modules: <<authorized_imports>>
Be sure to provide a 'Code:' token, else the system will be stuck in a loop.
Be sure to provide a 'Code:' token, else the run will fail.

Tools:
<<tool_descriptions>>
Expand All @@ -68,7 +69,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
translated_question = translator(question=question, src_lang="French", tgt_lang="English")
print(f"The translated question is {translated_question}.")
answer = image_qa(image=image, question=translated_question)
print(f"The answer is {answer}")
final_answer(f"The answer is {answer}")
```<end_action>

---
Expand All @@ -80,6 +81,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
answer = document_qa(document, question="What is the oldest person?")
print(f"The answer is {answer}.")
image = image_generator(answer)
final_answer(image)
```<end_action>

---
Expand All @@ -89,6 +91,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
Code:
```py
image = image_generator(prompt=caption)
final_answer(image)
```<end_action>

---
Expand All @@ -100,6 +103,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
summarized_text = summarizer(text)
print(f"Summary: {summarized_text}")
audio_summary = text_reader(summarized_text)
final_answer(audio_summary)
```<end_action>

---
Expand All @@ -111,6 +115,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
answer = text_qa(text=text, question=question)
print(f"The answer is {answer}.")
image = image_generator(answer)
final_answer(image)
```<end_action>

---
Expand All @@ -120,10 +125,11 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
Code:
```py
caption = image_captioner(image)
final_answer(caption)
```<end_action>

---
Above example were using tools that might not exist for you. You only have access to those Tools:
Above example were using tools that might not exist for you. You only have acces to those Tools:
<<tool_names>>

Remember to make sure that variables you use are all defined.
Expand Down Expand Up @@ -250,7 +256,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
}<end_action>


Above example were using notional tools that might not exist for you. You only have access to those tools:
Above example were using notional tools that might not exist for you. You only have acces to those tools:
<<tool_descriptions>>

Here are the rules you should always follow to solve your task:
Expand Down Expand Up @@ -357,7 +363,9 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
4. Take care to not chain too many sequential tool calls in the same code block, especially when the output format is unpredictable. For instance, a call to search has an unpredictable return format, so do not have another tool call that depends on its output in the same block: rather output results with print() to use them in the next block.
5. Call a tool only when needed, and never re-do a tool call that you previously did with the exact same parameters.
6. Don't name any new variable with the same name as a tool: for instance don't name a variable 'final_answer'.
7. You can use imports in your code, but only from the following list of modules: <<authorized_imports>>
7. Never create any notional variables in our code, as having these in your logs might derail you from the true variables.
8. You can use imports in your code, but only from the following list of modules: <<authorized_imports>>
9. Don't give up! You're in charge of solving the task, not providing directions to solve it.

Now Begin! If you solve the task correctly, you will receive a reward of $1,000,000.
"""
1 change: 0 additions & 1 deletion src/transformers/agents/tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -650,7 +650,6 @@ def fn(*args, **kwargs):
"text-to-speech": "TextToSpeechTool",
"translation": "TranslationTool",
"python_interpreter": "PythonInterpreterTool",
"final_answer": "FinalAnswerTool",
}


Expand Down
Loading