Перейти к содержанию

Tools

На этой странице Прежде чем писать инструмент, спросите себя: не должен ли это бытьскилл вместо этого? Только встроенные основные инструменты Эта страница предназначена для добавления встроенного инструмента Hermes непосредственно в репозиторий. Если вам нужен личный, локальный для проекта или иной пользовательский инструмент без изменения ядра Hermes, используйте вместо этого путь плагинов: * Плагины * Создание плагина Hermes

По умолчанию используйте плагины для большинства пользовательских инструментов. Следуйте этой странице только тогда, когда вы явно хотите добавить новый встроенный инструмент в tools/ и toolsets.py. Делайте это Скиллом, когда возможность может быть выражена как инструкции + shell-команды + существующие инструменты (поиск arXiv, git-процессы, управление Docker, обработка PDF). Делайте это Инструментом, когда требуется сквозная интеграция с API-ключами, пользовательская логика обработки, работа с бинарными данными или стриминг (автоматизация браузера, TTS, анализ изображений).

Обзор

Добавление инструмента затрагивает 2 файла: 1. tools/your_tool.py — обработчик, схема, функция проверки, вызов registry.register() 2. toolsets.py — добавление имени инструмента в _HERMES_CORE_TOOLS (или конкретный набор инструментов)

Любой файл tools/*.py с вызовом registry.register() на верхнем уровне автоматически обнаруживается при запуске — ручной список импорта не требуется.

Шаг 1: Создание файла встроенного инструмента

Каждый файл инструмента следует одной и той же структуре: [code] # tools/weather_tool.py
"""Инструмент погоды — получение текущей погоды для местоположения."""

import json  
import os  
import logging

logger = logging.getLogger(__name__)


# --- Проверка доступности ---

def check_weather_requirements() -> bool:  
    """Вернуть True, если зависимости инструмента доступны."""  
    return bool(os.getenv("WEATHER_API_KEY"))


# --- Обработчик ---

def weather_tool(location: str, units: str = "metric") -> str:  
    """Получить погоду для местоположения. Возвращает JSON-строку."""  
    api_key = os.getenv("WEATHER_API_KEY")  
    if not api_key:  
        return json.dumps({"error": "WEATHER_API_KEY не настроен"})  
    try:  
        # ... вызов API погоды ...  
        return json.dumps({"location": location, "temp": 22, "units": units})  
    except Exception as e:  
        return json.dumps({"error": str(e)})


# --- Схема ---

WEATHER_SCHEMA = {  
    "name": "weather",  
    "description": "Получить текущую погоду для местоположения.",  
    "parameters": {  
        "type": "object",  
        "properties": {  
            "location": {  
                "type": "string",  
                "description": "Название города или координаты (например, 'Москва' или '55.7,37.6')"  
            },  
            "units": {  
                "type": "string",  
                "enum": ["metric", "imperial"],  
                "description": "Единицы измерения температуры (по умолчанию: metric)",  
                "default": "metric"  
            }  
        },  
        "required": ["location"]  
    }  
}


# --- Регистрация ---

from tools.registry import registry

registry.register(  
    name="weather",  
    toolset="weather",  
    schema=WEATHER_SCHEMA,  
    handler=lambda args, **kw: weather_tool(  
        location=args.get("location", ""),  
        units=args.get("units", "metric")),  
    check_fn=check_weather_requirements,  
    requires_env=["WEATHER_API_KEY"],  
)

[/code]

Ключевые правила

Важно * Обработчики ОБЯЗАНЫ возвращать JSON-строку (через json.dumps()), никогда не сырые словари * Ошибки ОБЯЗАНЫ возвращаться как {"error": "message"}, никогда не выбрасываться как исключения * check_fn вызывается при построении определений инструментов — если она возвращает False, инструмент молча исключается * handler получает (args: dict, **kwargs), где args — аргументы вызова инструмента от LLM

Шаг 2: Добавление встроенного инструмента в набор

В toolsets.py добавьте имя инструмента: [code] # Если он должен быть доступен на всех платформах (CLI + мессенджеры):
_HERMES_CORE_TOOLS = [
...
"weather", # <-- добавить сюда
]

# Или создайте новый отдельный набор инструментов:  
"weather": {  
    "description": "Инструменты поиска погоды",  
    "tools": ["weather"],  
    "includes": []  
},

[/code]

~~Шаг 3: Добавление импорта обнаружения~~ (Больше не требуется)

Модули инструментов с вызовом registry.register() на верхнем уровне автоматически обнаруживаются функцией discover_builtin_tools() в tools/registry.py. Никакого ручного списка импорта для поддержки — просто создайте файл в tools/, и он будет подхвачен при запуске.

Асинхронные обработчики

Если вашему обработчику нужен асинхронный код, пометьте его параметром is_async=True: [code] async def weather_tool_async(location: str) -> str:
async with aiohttp.ClientSession() as session:
...
return json.dumps(result)

registry.register(  
    name="weather",  
    toolset="weather",  
    schema=WEATHER_SCHEMA,  
    handler=lambda args, **kw: weather_tool_async(args.get("location", "")),  
    check_fn=check_weather_requirements,  
    is_async=True,  # registry вызывает _run_async() автоматически  
)

[/code] Реестр обрабатывает асинхронное bridging прозрачно — вам никогда не нужно вызывать asyncio.run() самостоятельно.

Обработчики, которым нужен task_id

Инструменты, управляющие состоянием сессии, получают task_id через **kwargs: [code] def _handle_weather(args, **kw):
task_id = kw.get("task_id")
return weather_tool(args.get("location", ""), task_id=task_id)

registry.register(  
    name="weather",  
    ...  
    handler=_handle_weather,  
)

[/code]

Инструменты, перехватываемые циклом агента

Некоторые инструменты (todo, memory, session_search, delegate_task) требуют доступа к состоянию агента сессии. Они перехватываются run_agent.py до того, как достигают реестра. Реестр всё ещё хранит их схемы, но dispatch() возвращает резервную ошибку, если перехват был обойдён.

Опционально: интеграция с мастером настройки

Если ваш инструмент требует API-ключ, добавьте его в hermes_cli/config.py: [code] OPTIONAL_ENV_VARS = {
...
"WEATHER_API_KEY": {
"description": "API-ключ погоды для поиска погоды",
"prompt": "API-ключ погоды",
"url": "https://weatherapi.com/",
"tools": ["weather"],
"password": True,
},
}

[/code]

Контрольный список