Skip to content

Commit

Permalink
Adds final answer tool for all agents
Browse files Browse the repository at this point in the history
  • Loading branch information
aymeric-roucher committed Jun 28, 2024
1 parent e655029 commit ca914cc
Show file tree
Hide file tree
Showing 5 changed files with 25 additions and 12 deletions.
11 changes: 9 additions & 2 deletions docs/source/en/agents.md
Original file line number Diff line number Diff line change
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 @@ -169,6 +169,7 @@ def __init__(self, tools: List[Tool], add_base_tools: bool = False):
self._tools = {tool.name: tool for tool in tools}
if add_base_tools:
self.add_base_tools()
self.add_tool(FinalAnswerTool())
self._load_tools_if_needed()

def add_base_tools(self, add_python_interpreter: bool = False):
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
20 changes: 14 additions & 6 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 All @@ -145,7 +151,7 @@ def download_prompt(prompt_or_repo_id, agent_name, mode="run"):
"action_input": $INPUT
}<end_action>
Make sure to have the $INPUT as a dictionary in the right format for the tool you are using, and do not put variable names as input if you can find the right values.
Make sure to have the $INPUT as a dictionnary in the right format for the tool you are using, and do not put variable names as input if you can find the right values.
You should ALWAYS use the following format:
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

0 comments on commit ca914cc

Please sign in to comment.