This repository contains the code for the demo paper:
AgentQuest: A Modular Benchmark Framework to Measure Progress and Improve LLM Agents
- Structure of This Repo
- Prerequisites
- How to Install and Test a Benchmark?
- How to Write a Custom Driver?
- How to Use a Driver?
- How to Use an LLM Agent?
This repo is organized as follows:
agentquest
folder. It contains the main libraries and drivers needed to run and evaluate the benchmarksbenchmarks
folder. It contains the documentation, the python requirements and the installation script (setup.sh
) of each benchmarkexamples
folder. It contains one instance of each benchmark callable from command line or jupyter notebook.agents
folder. It contains examples on how to implement LLM agents through different frameworks testing one benchmark.
Install our agentquest
package through
$ pip3 install -e .
or
$ python3 -m venv .venv
$ source .venv/bin/activate
$ pip install -e .
By default we provide the following benchmarks:
Reasoning Type | Categories | |
---|---|---|
MasterMind | Deductive | 5 |
AlfWorld | Embodied | 6 |
The first step is to install the requirements of each benchmark through the Makefile
.
You can both install all the benchmarks or a specific one.
To install all the benchmarks in a row, just open a terminal and run
$ make installall
Otherwise, to install a specific benchmark you can run
$ make install-BENCHMARK_NAME
where BENCHMARK_NAME
is one among mastermind
or alfworld
.
In the example
folder we provide two ways to test each benchmark.
- In the
scripts
folder you can find python scripts to test one instance of the implemented benchmarks from command line.
To test one benchmark, open a terminal and run:
$ python3 examples/scripts/BENCHMARK_NAME
- In the
notebooks
folder you can find jupyter notebooks providing an overview on how to use the benchmarks through python codes.
Note The AlfWorld benchmark requires to set a symlink to alfworld_data
directory. Run the following command from the root of this repository.
ln -s $PWD/benchmarks/alfworld/alfworld_data /tmp/alfworld_data
A driver is a wrapper allowing a user (human, LLM agent, generic script, etc.) to interact with an environment (e.g., AlfWorld, Mastermind). This allows to write a custom plug-and-play interface for an existing benchmark or for a new proposed one. The interaction between the environment occurs through two entities:
- Observations reporting the current environment state and the information about the game conclusion
- Actions, or an external inputs provided by the user changing the environment status.
Here we show how to design a custom driver for any benchmark.
Create a folder for your custom driver in the benchmarks
folder and create an empty driver file.
$ make init-CUSTOM_BENCHMARK
Replace CUSTOM_BENCHMARK
according to your new benchmark. This will create:
- the
agentquest/drivers/CUSTOM_BENCHMARK
folder for the driver to customize. The folder contains the scriptCUSTOM_BENCHMARK_driver.py
and the__init__.py
file - the
benchmarks/CUSTOM_BENCHMARK
folder containing- an empty
README.md
file for the benchmark documentation - an empty
requirements.txt
file for the python requirements of the benchmark - an empty
setup.sh
file as installation script.
- an empty
- the
agentquest/data/CUSTOM_BENCHMARK
folder for the instances of the benchmark.
After having designed the driver and the benchmark, remember to provide documentation, requirements and setup instructions through the files in benchmarks/CUSTOM_BENCHMARK
.
Note to remove an initialized benchmark, just run
$ make clean-CUSTOM_BENCHMARK
Observations inform the users about the current environment status. We provide a template with two mandatory attributes:
output
. A string reporting the environment statusdone
. A boolean variable indicating if the game is ended (True) or still running (False)
class Observation():
def __init__(self, output:str, done:bool):
self.output = output # Current environment status
self.done = done # True if the game ended, False otherwise.
Actions allows the user to interact with the environment changing its status. We provide a template with one mandatory attribute:
action_value
. A string that, once processed by the driver, triggers the environment change.
To customize the interactions, you can define optional attributes. In the following we provide an overview of the Action
class
class Action():
def __init__(self, action_value:str, **kwargs):
self.action_value = action_value # Action to interact with the environment
# Set additional custom attributes
for key, value in kwargs.items():
setattr(self, key, value)
Finally, we provide an overview of the main driver template.
from agentquest.utils import Observation
class CustomDriver():
def __init__(self, environment):
self.environment = environment
# Write your driver initialization here.
# ...
The reset method reset the driver and the environment returning the first observation from the environment.
def reset(self):
# Reset your environment status here.
# ...
# Instantiate your initial observation from the environment
obs = Observation(
output='This is the initial environment status',
done=False
)
return obs
The step method processes an action to modify the environment and returns the observation from the environment (i.e., the environment status after the actions).
def step(self, action):
action = action.action_value # Retrieve the action value
# Process your action here
# ...
# Run one game round. Typically you change/update the game status
# according to the provided output
# Instantiate your current observation from the environment
obs = Observation(
output='This is the current game status',
done=False
)
return obs
As a very simple example we explain the driver for the Mastermind game.
Mastermind is a code-breaking game where one player creates a secret code of a fixed lenght with digits from 0 to 9, and the other player tries to guess the code. The code-maker provides feedbackdigit after each guess, indicating the number of correct colors and their correct positions, allowing the code-breaker to deduce and refine their guesses.
from agentquest.utils import Observation
from collections import Counter
class MasterMindDriver():
def __init__(self, truth:str):
self.truth = truth # The game solution
self.split_truth = [x for x in truth] # Split the solution in digits
Define custom methods used by the environment:
get_correct_positions
. Custom function retrievig the number of digits in the same position appearing in two lists.count_common_elements
. Custom function retrieving the number of common digits between two lists independently from their positions.
def get_correct_positions(self, list1:list, list2:list):
number_and_position = 0
# Compare the position of the digits in the lists
for i, j in zip(list1, list2):
if i == j: number_and_position += 1
return number_and_position
def count_common_elements(self, list1:list, list2:list):
# Get common elements
common_elements = Counter(list1) & Counter(list2)
# Get the number of common elements
count = sum(common_elements.values())
return count
Define the mandatory reset
method initializing the environment.
def reset(self):
# Get the length of the solution
code_len = len(self.split_truth)
# Instantiate the starting environment status
obs = Observation(
output = f'Start guessing the {code_len} digits number.',
done = False
)
return obs
Define the mandatory step
method implementing one Mastermind round.
def step(self, action):
# Retrieve the action value and trim exceeding digits
action = action.action_value[:len(self.split_truth)]
# If the guess is correct, end the game
if action == self.truth:
obs = Observation(output='You Won!', done=True)
return obs
# Get the guessed number in digits
split_guess = [x for x in str(action)]
# Get the number of correct digits in the correct position
number_and_position = self.get_correct_positions(self.split_truth, split_guess)
# Get the number correct digits independently from their position
number_only = self.count_common_elements(self.split_truth, split_guess)
number_only -= number_and_position
# Instantiate the current observation
obs = Observation(
output = (
f'Your guess has {number_only} correct numbers in the wrong '
f'position and {number_and_position} correct numbers in the '
f'correct position. Keep guessing.'
),
done = False
)
return obs
We provide a simple example from the same Mastermind environment just decribed. With the following snippet you will be able to run a game of Mastermind.
Fistly import the driver and the action template.
from agentquest.drivers.mastermind import MasterMindDriver
from agentquest.utils import Action
Initialize the driver providing the final solution to guess and reset the environment getting the first observation.
driver = MasterMindDriver(truth='7327')
obs = driver.reset()
obs.output, obs.done
>>> ('Start guessing the 4 digits number.', False)
Instantiate the action with the current guess and run one Mastermind round.
guess = Action(action_value='1234')
obs = driver.step(guess)
obs.output, obs.done
>>> ('Start guessing the 4 digits number.', False)
>>> ('Your guess has 2 correct numbers in the wrong position and 0 correct '
... 'numbers in the correct position. Keep guessing.', False)
Repeat...
guess = Action(action_value='7327')
obs = driver.step(guess)
obs.output, obs.done
>>> ('You Won!', True)
We provide a set of examples showcasing how to implement an LLM agent to solve the MasterMind benchmark. Generalizing to other benchmarks is straightforward.
mastermind-openai-chat-model-with-metrics.ipynb
: MasterMind benchmark with OpenAI Simple Chat model and computation of Progress and Repetition Rates metricsmastermind-langchain-chat-model.ipynb
: MasterMind benchmark with LangChain Simple Chat modelmastermind-langchain-react.ipynb
: MasterMind benchmark with ReAct agentmastermind-langchain-openai-assistant.ipynb
: MasterMind benchmark with OpenAI Assistant
Note All the examples require to configure OpenAI/Azure API keys.