Skip to content

Commit

Permalink
feat: allow to load tools from external modules
Browse files Browse the repository at this point in the history
  • Loading branch information
jrmi committed Dec 21, 2024
1 parent 7abeb1e commit 7cfed4f
Show file tree
Hide file tree
Showing 22 changed files with 424 additions and 153 deletions.
5 changes: 5 additions & 0 deletions docs/config.rst
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,11 @@ Here is an example:
#MODEL = "local/<model-name>"
#OPENAI_BASE_URL = "http://localhost:11434/v1"
# Uncomment to change tool configuration
#TOOL_FORMAT = "markdown" # Select the tool formal. One of `markdown`, `xml`, `tool`
#TOOL_ALLOW_LIST = "save,append,patch,python" # Comma separated list of allowed tools
#TOOL_MODULES = "gptme.tools,custom.tools" # List of python comma separated python module path
The ``prompt`` section contains options for the prompt.

The ``env`` section contains environment variables that gptme will fall back to if they are not set in the shell environment. This is useful for setting the default model and API keys for :doc:`providers`.
Expand Down
74 changes: 74 additions & 0 deletions docs/custom_tool.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
Creating a Custom Tool for gptme
=================================

Introduction
------------
In gptme, a custom tool allows you to extend the functionality of the assistant by
defining new tools that can be executed.
This guide will walk you through the process of creating and registering a custom tool.

Creating a Custom Tool
-----------------------
To create a custom tool, you need to define a new instance of the `ToolSpec` class.
This class requires several parameters:

- **name**: The name of the tool.
- **desc**: A description of what the tool does.
- **instructions**: Instructions on how to use the tool.
- **examples**: Example usage of the tool.
- **execute**: A function that defines the tool's behavior when executed.
- **block_types**: The block types to detects.
- **parameters**: A list of parameters that the tool accepts.

Here is a basic example of defining a custom tool:

.. code-block:: python
import random
from gptme.tools.base import ToolSpec, Parameter, ToolUse
from gptme.message import Message
def execute(code, args, kwargs, confirm):
if code is None and kwargs is not None:
code = kwargs.get('side_count')
yield Message('system', f"Result: {random.randint(1,code)}")
def examples(tool_format):
return f"""
> User: Throw a dice and give me the result.
> Assistant:
{ToolUse("dice", [], "6").to_output(tool_format)}
> System: 3
> assistant: The result is 3
""".strip()
tool = ToolSpec(
name="dice",
desc="A dice simulator.",
instructions="This tool generate a random integer value like a dice.",
examples=examples,
execute=execute,
block_types=["dice"],
parameters=[
Parameter(
name="side_count",
type="integer",
description="The number of faces of the dice to throw.",
required=True,
),
],
)
Registering the Tool
---------------------
To ensure your tool is available for use, you can specify the module in the `TOOL_MODULES` env variable or
setting in your :doc:`project configuration file <config>`, which will automatically load your custom tools.

.. code-block:: toml
TOOL_MODULES = "gptme.tools,path.to.your.custom_tool_module"
Don't remove the `gptme.tools` package unless you know exactly what you are doing.

Remember your module must be in the Python path so either you can install it or tweak
the `PYTHONPATH` env variable.
1 change: 1 addition & 0 deletions docs/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ See the `README <https://github.com/ErikBjare/gptme/blob/master/README.md>`_ fil
evals
bot
finetuning
custom_tool
arewetiny
timeline
alternatives
Expand Down
28 changes: 19 additions & 9 deletions gptme/chat.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,13 @@
import re
import sys
import termios
from typing import cast
import urllib.parse
from collections.abc import Generator
from pathlib import Path

from .commands import action_descriptions, execute_cmd
from .config import get_config
from .constants import PROMPT_USER
from .init import init
from .llm import reply
Expand All @@ -19,11 +21,10 @@
from .tools import (
ToolFormat,
ToolUse,
execute_msg,
has_tool,
loaded_tools,
get_tools,
)
from .tools.base import ConfirmFunc
from .tools.base import ConfirmFunc, set_tool_format
from .tools.browser import read_url
from .util import console, path_with_tilde, print_bell
from .util.ask_execute import ask_execute
Expand All @@ -46,7 +47,7 @@ def chat(
show_hidden: bool = False,
workspace: Path | None = None,
tool_allowlist: list[str] | None = None,
tool_format: ToolFormat = "markdown",
tool_format: ToolFormat | None = None,
) -> None:
"""
Run the chat loop.
Expand All @@ -71,6 +72,15 @@ def chat(
console.log(f"Using logdir {path_with_tilde(logdir)}")
manager = LogManager.load(logdir, initial_msgs=initial_msgs, create=True)

config = get_config()
tool_format_with_default: ToolFormat = tool_format or cast(
ToolFormat, config.get_env("TOOL_FORMAT", "markdown")
)

# By defining the tool_format at the last moment we ensure we can use the
# configuration for subagent
set_tool_format(tool_format_with_default)

# change to workspace directory
# use if exists, create if @log, or use given path
log_workspace = logdir / "workspace"
Expand Down Expand Up @@ -128,7 +138,7 @@ def confirm_func(msg) -> bool:
manager.log,
stream,
confirm_func,
tool_format,
tool_format_with_default,
workspace,
)
)
Expand Down Expand Up @@ -177,7 +187,7 @@ def confirm_func(msg) -> bool:
# ask for input if no prompt, generate reply, and run tools
clear_interruptible() # Ensure we're not interruptible during user input
for msg in step(
manager.log, stream, confirm_func, tool_format, workspace
manager.log, stream, confirm_func, tool_format_with_default, workspace
): # pragma: no cover
manager.append(msg)
# run any user-commands, if msg is from user
Expand All @@ -189,7 +199,7 @@ def step(
log: Log | list[Message],
stream: bool,
confirm: ConfirmFunc,
tool_format: ToolFormat = "markdown",
tool_format: ToolFormat,
workspace: Path | None = None,
) -> Generator[Message, None, None]:
"""Runs a single pass of the chat."""
Expand Down Expand Up @@ -222,7 +232,7 @@ def step(

tools = None
if tool_format == "tool":
tools = [t for t in loaded_tools if t.is_runnable()]
tools = [t for t in get_tools() if t.is_runnable()]

# generate response
msg_response = reply(msgs, get_model().model, stream, tools)
Expand All @@ -232,7 +242,7 @@ def step(
# log response and run tools
if msg_response:
yield msg_response.replace(quiet=True)
yield from execute_msg(msg_response, confirm)
yield from msg_response.execute(confirm)
except KeyboardInterrupt:
clear_interruptible()
yield Message("system", "Interrupted")
Expand Down
14 changes: 4 additions & 10 deletions gptme/cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,9 @@
import click
from pick import pick

from gptme.config import get_config

from .chat import chat
from .config import get_config
from .commands import _gen_help
from .constants import MULTIPROMPT_SEPARATOR
from .dirs import get_logs_dir
Expand All @@ -22,12 +22,7 @@
from .logmanager import ConversationMeta, get_user_conversations
from .message import Message
from .prompts import get_prompt
from .tools import (
ToolFormat,
ToolSpec,
init_tools,
set_tool_format,
)
from .tools import ToolFormat, init_tools, get_available_tools
from .util import epoch_to_age
from .util.generate_name import generate_name
from .util.interrupt import handle_keyboard_interrupt, set_interruptible
Expand All @@ -39,7 +34,7 @@
script_path = Path(os.path.realpath(__file__))
commands_help = "\n".join(_gen_help(incl_langtags=False))
available_tool_names = ", ".join(
sorted([tool.name for tool in ToolSpec.get_tools().values() if tool.available])
sorted([tool.name for tool in get_available_tools() if tool.available])
)


Expand Down Expand Up @@ -187,8 +182,7 @@ def main(

config = get_config()

tool_format = tool_format or config.get_env("TOOL_FORMAT") or "markdown"
set_tool_format(tool_format)
tool_format = tool_format or config.get_env("TOOL_FORMAT", "markdown")

# early init tools to generate system prompt
init_tools(frozenset(tool_allowlist) if tool_allowlist else None)
Expand Down
10 changes: 5 additions & 5 deletions gptme/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@
print_msg,
toml_to_msgs,
)
from .tools import ToolUse, execute_msg, loaded_tools
from .tools import ToolUse, get_tools
from .tools.base import ConfirmFunc, get_tool_format
from .util.export import export_chat_to_html
from .util.useredit import edit_text_with_editor
Expand Down Expand Up @@ -125,20 +125,20 @@ def handle_cmd(
print("Replaying conversation...")
for msg in manager.log:
if msg.role == "assistant":
for reply_msg in execute_msg(msg, confirm):
for reply_msg in msg.execute(confirm):
print_msg(reply_msg, oneline=False)
case "impersonate":
content = full_args if full_args else input("[impersonate] Assistant: ")
msg = Message("assistant", content)
yield msg
yield from execute_msg(msg, confirm=lambda _: True)
yield from msg.execute(confirm=lambda _: True)
case "tokens":
manager.undo(1, quiet=True)
log_costs(manager.log.messages)
case "tools":
manager.undo(1, quiet=True)
print("Available tools:")
for tool in loaded_tools:
for tool in get_tools():
print(
f"""
# {tool.name}
Expand Down Expand Up @@ -220,7 +220,7 @@ def _gen_help(incl_langtags: bool = True) -> Generator[str, None, None]:
yield " /python print('hello')"
yield ""
yield "Supported langtags:"
for tool in loaded_tools:
for tool in get_tools():
if tool.block_types:
yield f" - {tool.block_types[0]}" + (
f" (alias: {', '.join(tool.block_types[1:])})"
Expand Down
4 changes: 2 additions & 2 deletions gptme/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,11 @@ class Config:
env: dict

def get_env(self, key: str, default: str | None = None) -> str | None:
"""Gets an enviromnent variable, checks the config file if it's not set in the environment."""
"""Gets an environment variable, checks the config file if it's not set in the environment."""
return os.environ.get(key) or self.env.get(key) or default

def get_env_required(self, key: str) -> str:
"""Gets an enviromnent variable, checks the config file if it's not set in the environment."""
"""Gets an environment variable, checks the config file if it's not set in the environment."""
if val := os.environ.get(key) or self.env.get(key):
return val
raise KeyError( # pragma: no cover
Expand Down
16 changes: 15 additions & 1 deletion gptme/message.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,8 @@
from dataclasses import dataclass, field
from datetime import datetime
from pathlib import Path
from typing import Literal
from typing import Literal, TYPE_CHECKING
from collections.abc import Generator

import tomlkit
from rich.syntax import Syntax
Expand All @@ -21,6 +22,9 @@

logger = logging.getLogger(__name__)

if TYPE_CHECKING:
from .tools.base import ConfirmFunc


@dataclass(frozen=True, eq=False)
class Message:
Expand Down Expand Up @@ -67,6 +71,16 @@ def replace(self, **kwargs) -> Self:
"""Replace attributes of the message."""
return dataclasses.replace(self, **kwargs)

def execute(self, confirm: "ConfirmFunc") -> "Generator[Message, None, None]":
"""Uses any tools called in a message and returns the response."""
from .tools.base import ToolUse

assert self.role == "assistant", "Only assistant messages can be executed"

for tooluse in ToolUse.iter_from_content(self.content):
if tooluse.is_runnable:
yield from tooluse.execute(confirm)

def to_dict(self, keys=None) -> dict:
"""Return a dict representation of the message, serializable to JSON."""

Expand Down
6 changes: 3 additions & 3 deletions gptme/prompts.py
Original file line number Diff line number Diff line change
Expand Up @@ -199,14 +199,14 @@ def prompt_tools(
examples: bool = True, tool_format: ToolFormat = "markdown"
) -> Generator[Message, None, None]:
"""Generate the tools overview prompt."""
from .tools import loaded_tools # fmt: skip
from .tools import get_tools # fmt: skip

assert loaded_tools, "No tools loaded"
assert get_tools(), "No tools loaded"

use_tool = tool_format == "tool"

prompt = "# Tools aliases" if use_tool else "# Tools Overview"
for tool in loaded_tools:
for tool in get_tools():
if not use_tool or not tool.is_runnable():
prompt += tool.get_tool_prompt(examples, tool_format)

Expand Down
5 changes: 2 additions & 3 deletions gptme/server/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,6 @@
from ..llm.models import get_model
from ..logmanager import LogManager, get_user_conversations, prepare_messages
from ..message import Message
from ..tools import execute_msg
from ..tools.base import ToolUse

logger = logging.getLogger(__name__)
Expand Down Expand Up @@ -124,7 +123,7 @@ def api_conversation_generate(logfile: str):
manager.append(msg)

# Execute any tools
reply_msgs = list(execute_msg(msg, confirm_func))
reply_msgs = list(msg.execute(confirm_func))
for reply_msg in reply_msgs:
manager.append(reply_msg)

Expand Down Expand Up @@ -191,7 +190,7 @@ def generate() -> Generator[str, None, None]:
yield f"data: {flask.json.dumps({'role': 'assistant', 'content': output, 'stored': True})}\n\n"

# Execute any tools and stream their output
for reply_msg in execute_msg(msg, confirm_func):
for reply_msg in msg.execute(confirm_func):
logger.debug(
f"Tool output: {reply_msg.role} - {reply_msg.content[:100]}..."
)
Expand Down
Loading

0 comments on commit 7cfed4f

Please sign in to comment.