first commit
This commit is contained in:
884
autonomous-ai-agents/hermes-agent/SKILL.md
Normal file
884
autonomous-ai-agents/hermes-agent/SKILL.md
Normal file
@@ -0,0 +1,884 @@
|
||||
---
|
||||
name: hermes-agent
|
||||
description: "Configure, extend, or contribute to Hermes Agent."
|
||||
version: 2.0.0
|
||||
author: Hermes Agent + Teknium
|
||||
license: MIT
|
||||
metadata:
|
||||
hermes:
|
||||
tags: [hermes, setup, configuration, multi-agent, spawning, cli, gateway, development]
|
||||
homepage: https://github.com/NousResearch/hermes-agent
|
||||
related_skills: [claude-code, codex, opencode]
|
||||
---
|
||||
|
||||
# Hermes Agent
|
||||
|
||||
Hermes Agent is an open-source AI agent framework by Nous Research that runs in your terminal, messaging platforms, and IDEs. It belongs to the same category as Claude Code (Anthropic), Codex (OpenAI), and OpenClaw — autonomous coding and task-execution agents that use tool calling to interact with your system. Hermes works with any LLM provider (OpenRouter, Anthropic, OpenAI, DeepSeek, local models, and 15+ others) and runs on Linux, macOS, and WSL.
|
||||
|
||||
What makes Hermes different:
|
||||
|
||||
- **Self-improving through skills** — Hermes learns from experience by saving reusable procedures as skills. When it solves a complex problem, discovers a workflow, or gets corrected, it can persist that knowledge as a skill document that loads into future sessions. Skills accumulate over time, making the agent better at your specific tasks and environment.
|
||||
- **Persistent memory across sessions** — remembers who you are, your preferences, environment details, and lessons learned. Pluggable memory backends (built-in, Honcho, Mem0, and more) let you choose how memory works.
|
||||
- **Multi-platform gateway** — the same agent runs on Telegram, Discord, Slack, WhatsApp, Signal, Matrix, Email, and 10+ other platforms with full tool access, not just chat.
|
||||
- **Provider-agnostic** — swap models and providers mid-workflow without changing anything else. Credential pools rotate across multiple API keys automatically.
|
||||
- **Profiles** — run multiple independent Hermes instances with isolated configs, sessions, skills, and memory.
|
||||
- **Extensible** — plugins, MCP servers, custom tools, webhook triggers, cron scheduling, and the full Python ecosystem.
|
||||
|
||||
People use Hermes for software development, research, system administration, data analysis, content creation, home automation, and anything else that benefits from an AI agent with persistent context and full system access.
|
||||
|
||||
**This skill helps you work with Hermes Agent effectively** — setting it up, configuring features, spawning additional agent instances, troubleshooting issues, finding the right commands and settings, and understanding how the system works when you need to extend or contribute to it.
|
||||
|
||||
**Docs:** https://hermes-agent.nousresearch.com/docs/
|
||||
|
||||
## Quick Start
|
||||
|
||||
```bash
|
||||
# Install
|
||||
curl -fsSL https://raw.githubusercontent.com/NousResearch/hermes-agent/main/scripts/install.sh | bash
|
||||
|
||||
# Interactive chat (default)
|
||||
hermes
|
||||
|
||||
# Single query
|
||||
hermes chat -q "What is the capital of France?"
|
||||
|
||||
# Setup wizard
|
||||
hermes setup
|
||||
|
||||
# Change model/provider
|
||||
hermes model
|
||||
|
||||
# Check health
|
||||
hermes doctor
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## References
|
||||
|
||||
- **references/web-search-from-server.md** — 服务器环境下进行网络搜索的完整指南:Playwright --no-sandbox 配置、各搜索引擎反爬情况、mmx search 限制、MiMo 搜索能力说明
|
||||
|
||||
### Global Flags
|
||||
|
||||
```
|
||||
hermes [flags] [command]
|
||||
|
||||
--version, -V Show version
|
||||
--resume, -r SESSION Resume session by ID or title
|
||||
--continue, -c [NAME] Resume by name, or most recent session
|
||||
--worktree, -w Isolated git worktree mode (parallel agents)
|
||||
--skills, -s SKILL Preload skills (comma-separate or repeat)
|
||||
--profile, -p NAME Use a named profile
|
||||
--yolo Skip dangerous command approval
|
||||
--pass-session-id Include session ID in system prompt
|
||||
```
|
||||
|
||||
No subcommand defaults to `chat`.
|
||||
|
||||
### Chat
|
||||
|
||||
```
|
||||
hermes chat [flags]
|
||||
-q, --query TEXT Single query, non-interactive
|
||||
-m, --model MODEL Model (e.g. anthropic/claude-sonnet-4)
|
||||
-t, --toolsets LIST Comma-separated toolsets
|
||||
--provider PROVIDER Force provider (openrouter, anthropic, nous, etc.)
|
||||
-v, --verbose Verbose output
|
||||
-Q, --quiet Suppress banner, spinner, tool previews
|
||||
--checkpoints Enable filesystem checkpoints (/rollback)
|
||||
--source TAG Session source tag (default: cli)
|
||||
```
|
||||
|
||||
### Configuration
|
||||
|
||||
```
|
||||
hermes setup [section] Interactive wizard (model|terminal|gateway|tools|agent)
|
||||
hermes model Interactive model/provider picker
|
||||
hermes config View current config
|
||||
hermes config edit Open config.yaml in $EDITOR
|
||||
hermes config set KEY VAL Set a config value
|
||||
hermes config path Print config.yaml path
|
||||
hermes config env-path Print .env path
|
||||
hermes config check Check for missing/outdated config
|
||||
hermes config migrate Update config with new options
|
||||
hermes login [--provider P] OAuth login (nous, openai-codex)
|
||||
hermes logout Clear stored auth
|
||||
hermes doctor [--fix] Check dependencies and config
|
||||
hermes status [--all] Show component status
|
||||
```
|
||||
|
||||
### Tools & Skills
|
||||
|
||||
```
|
||||
hermes tools Interactive tool enable/disable (curses UI)
|
||||
hermes tools list Show all tools and status
|
||||
hermes tools enable NAME Enable a toolset
|
||||
hermes tools disable NAME Disable a toolset
|
||||
|
||||
hermes skills list List installed skills
|
||||
hermes skills search QUERY Search the skills hub
|
||||
hermes skills install ID Install a skill (ID can be a hub identifier OR a direct https://…/SKILL.md URL; pass --name to override when frontmatter has no name)
|
||||
hermes skills inspect ID Preview without installing
|
||||
hermes skills config Enable/disable skills per platform
|
||||
hermes skills check Check for updates
|
||||
hermes skills update Update outdated skills
|
||||
hermes skills uninstall N Remove a hub skill
|
||||
hermes skills publish PATH Publish to registry
|
||||
hermes skills browse Browse all available skills
|
||||
hermes skills tap add REPO Add a GitHub repo as skill source
|
||||
```
|
||||
|
||||
### MCP Servers
|
||||
|
||||
```
|
||||
hermes mcp serve Run Hermes as an MCP server
|
||||
hermes mcp add NAME Add an MCP server (--url or --command)
|
||||
hermes mcp remove NAME Remove an MCP server
|
||||
hermes mcp list List configured servers
|
||||
hermes mcp test NAME Test connection
|
||||
hermes mcp configure NAME Toggle tool selection
|
||||
```
|
||||
|
||||
### Gateway (Messaging Platforms)
|
||||
|
||||
```
|
||||
hermes gateway run Start gateway foreground
|
||||
hermes gateway install Install as background service
|
||||
hermes gateway start/stop Control the service
|
||||
hermes gateway restart Restart the service
|
||||
hermes gateway status Check status
|
||||
hermes gateway setup Configure platforms
|
||||
```
|
||||
|
||||
Supported platforms: Telegram, Discord, Slack, WhatsApp, Signal, Email, SMS, Matrix, Mattermost, Home Assistant, DingTalk, Feishu, WeCom, BlueBubbles (iMessage), Weixin (WeChat), API Server, Webhooks. Open WebUI connects via the API Server adapter.
|
||||
|
||||
Platform docs: https://hermes-agent.nousresearch.com/docs/user-guide/messaging/
|
||||
|
||||
### Sessions
|
||||
|
||||
```
|
||||
hermes sessions list List recent sessions
|
||||
hermes sessions browse Interactive picker
|
||||
hermes sessions export OUT Export to JSONL
|
||||
hermes sessions rename ID T Rename a session
|
||||
hermes sessions delete ID Delete a session
|
||||
hermes sessions prune Clean up old sessions (--older-than N days)
|
||||
hermes sessions stats Session store statistics
|
||||
```
|
||||
|
||||
### Cron Jobs
|
||||
|
||||
```
|
||||
hermes cron list List jobs (--all for disabled)
|
||||
hermes cron create SCHED Create: '30m', 'every 2h', '0 9 * * *'
|
||||
hermes cron edit ID Edit schedule, prompt, delivery
|
||||
hermes cron pause/resume ID Control job state
|
||||
hermes cron run ID Trigger on next tick
|
||||
hermes cron remove ID Delete a job
|
||||
hermes cron status Scheduler status
|
||||
```
|
||||
|
||||
### Webhooks
|
||||
|
||||
```
|
||||
hermes webhook subscribe N Create route at /webhooks/<name>
|
||||
hermes webhook list List subscriptions
|
||||
hermes webhook remove NAME Remove a subscription
|
||||
hermes webhook test NAME Send a test POST
|
||||
```
|
||||
|
||||
### Profiles
|
||||
|
||||
```
|
||||
hermes profile list List all profiles
|
||||
hermes profile create NAME Create (--clone, --clone-all, --clone-from)
|
||||
hermes profile use NAME Set sticky default
|
||||
hermes profile delete NAME Delete a profile
|
||||
hermes profile show NAME Show details
|
||||
hermes profile alias NAME Manage wrapper scripts
|
||||
hermes profile rename A B Rename a profile
|
||||
hermes profile export NAME Export to tar.gz
|
||||
hermes profile import FILE Import from archive
|
||||
```
|
||||
|
||||
### Credential Pools
|
||||
|
||||
```
|
||||
hermes auth add Interactive credential wizard
|
||||
hermes auth list [PROVIDER] List pooled credentials
|
||||
hermes auth remove P INDEX Remove by provider + index
|
||||
hermes auth reset PROVIDER Clear exhaustion status
|
||||
```
|
||||
|
||||
### Dashboard (Web UI)
|
||||
```
|
||||
hermes dashboard Start web UI (default port 9119)
|
||||
hermes dashboard --tui Include embedded Chat tab (browser-based hermes --tui)
|
||||
hermes dashboard --status List running dashboard processes
|
||||
hermes dashboard --stop Kill all dashboard processes
|
||||
hermes dashboard --port N Custom port
|
||||
hermes dashboard --insecure Bind to 0.0.0.0 (exposes API keys!)
|
||||
```
|
||||
|
||||
Dashboard provides: Sessions browser, Config editor, API key management, Skills catalog, Cron job manager, and optionally a Chat tab (`--tui`).
|
||||
|
||||
**No built-in password auth.** For remote access, use SSH port-forwarding or a reverse proxy with basic auth. See `references/dashboard-remote-access.md` for full details and password-protection options.
|
||||
|
||||
### Kanban (Task Board)
|
||||
```
|
||||
hermes kanban init Create kanban.db if missing (idempotent)
|
||||
hermes kanban create TITLE Create a new task (--assignee, --parent, --body, --json)
|
||||
hermes kanban list (ls) List tasks (--status, --assignee, --json)
|
||||
hermes kanban show ID Show task with comments + events
|
||||
hermes kanban assign ID P Assign or reassign task to profile P
|
||||
hermes kanban link PARENT CHILD Add parent→child dependency
|
||||
hermes kanban unlink PARENT CHILD Remove dependency
|
||||
hermes kanban claim Atomically claim a ready task (for workers)
|
||||
hermes kanban complete ID Mark task done (--summary, --metadata)
|
||||
hermes kanban block ID REASON Mark task blocked
|
||||
hermes kanban unblock ID Return blocked task to ready
|
||||
hermes kanban archive ID Archive completed/blocked tasks
|
||||
hermes kanban comment ID TEXT Append a comment
|
||||
hermes kanban tail ID Follow a task's event stream
|
||||
hermes kanban stats Per-status + per-assignee counts
|
||||
hermes kanban dispatch One dispatcher pass (reclaim stale, promote, spawn)
|
||||
hermes kanban watch Live-stream task_events (Ctrl+C to exit)
|
||||
hermes kanban log ID Print worker log for a task
|
||||
hermes kanban runs ID Show attempt history (profile, outcome, elapsed)
|
||||
hermes kanban context ID Print full worker context (title + body + parents + comments)
|
||||
hermes kanban gc Garbage-collect archived workspaces and old events
|
||||
```
|
||||
|
||||
Kanban config in `config.yaml`:
|
||||
```yaml
|
||||
kanban:
|
||||
dispatch_in_gateway: true # auto-dispatch via gateway
|
||||
dispatch_interval_seconds: 60 # how often dispatcher checks
|
||||
```
|
||||
|
||||
Gateway must be running for auto-dispatch: `hermes gateway status`.
|
||||
|
||||
Full skill coverage: `kanban-orchestrator` (decomposition playbook), `kanban-worker` (pitfalls/examples).
|
||||
|
||||
### Other
|
||||
```
|
||||
hermes insights [--days N] Usage analytics
|
||||
hermes update Update to latest version
|
||||
hermes pairing list/approve/revoke DM authorization
|
||||
hermes plugins list/install/remove Plugin management
|
||||
hermes honcho setup/status Honcho memory integration (requires honcho plugin)
|
||||
hermes memory setup/status/off Memory provider config
|
||||
hermes completion bash|zsh Shell completions
|
||||
hermes acp ACP server (IDE integration)
|
||||
hermes claw migrate Migrate from OpenClaw
|
||||
hermes uninstall Uninstall Hermes
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Slash Commands (In-Session)
|
||||
|
||||
Type these during an interactive chat session.
|
||||
|
||||
### Session Control
|
||||
```
|
||||
/new (/reset) Fresh session
|
||||
/clear Clear screen + new session (CLI)
|
||||
/retry Resend last message
|
||||
/undo Remove last exchange
|
||||
/title [name] Name the session
|
||||
/compress Manually compress context
|
||||
/stop Kill background processes
|
||||
/rollback [N] Restore filesystem checkpoint
|
||||
/background <prompt> Run prompt in background
|
||||
/queue <prompt> Queue for next turn
|
||||
/resume [name] Resume a named session
|
||||
```
|
||||
|
||||
### Configuration
|
||||
```
|
||||
/config Show config (CLI)
|
||||
/model [name] Show or change model
|
||||
/personality [name] Set personality
|
||||
/reasoning [level] Set reasoning (none|minimal|low|medium|high|xhigh|show|hide)
|
||||
/verbose Cycle: off → new → all → verbose
|
||||
/voice [on|off|tts] Voice mode
|
||||
/yolo Toggle approval bypass
|
||||
/skin [name] Change theme (CLI)
|
||||
/statusbar Toggle status bar (CLI)
|
||||
```
|
||||
|
||||
### Tools & Skills
|
||||
```
|
||||
/tools Manage tools (CLI)
|
||||
/toolsets List toolsets (CLI)
|
||||
/skills Search/install skills (CLI)
|
||||
/skill <name> Load a skill into session
|
||||
/cron Manage cron jobs (CLI)
|
||||
/reload-mcp Reload MCP servers
|
||||
/plugins List plugins (CLI)
|
||||
```
|
||||
|
||||
### Gateway
|
||||
```
|
||||
/approve Approve a pending command (gateway)
|
||||
/deny Deny a pending command (gateway)
|
||||
/restart Restart gateway (gateway)
|
||||
/sethome Set current chat as home channel (gateway)
|
||||
/update Update Hermes to latest (gateway)
|
||||
/platforms (/gateway) Show platform connection status (gateway)
|
||||
```
|
||||
|
||||
### Utility
|
||||
```
|
||||
/branch (/fork) Branch the current session
|
||||
/fast Toggle priority/fast processing
|
||||
/browser Open CDP browser connection
|
||||
/history Show conversation history (CLI)
|
||||
/save Save conversation to file (CLI)
|
||||
/paste Attach clipboard image (CLI)
|
||||
/image Attach local image file (CLI)
|
||||
```
|
||||
|
||||
### Info
|
||||
```
|
||||
/help Show commands
|
||||
/commands [page] Browse all commands (gateway)
|
||||
/usage Token usage
|
||||
/insights [days] Usage analytics
|
||||
/status Session info (gateway)
|
||||
/profile Active profile info
|
||||
```
|
||||
|
||||
### Exit
|
||||
```
|
||||
/quit (/exit, /q) Exit CLI
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Key Paths & Config
|
||||
|
||||
```
|
||||
~/.hermes/config.yaml Main configuration
|
||||
~/.hermes/.env API keys and secrets
|
||||
$HERMES_HOME/skills/ Installed skills
|
||||
~/.hermes/sessions/ Session transcripts
|
||||
~/.hermes/logs/ Gateway and error logs
|
||||
~/.hermes/auth.json OAuth tokens and credential pools
|
||||
~/.hermes/hermes-agent/ Source code (if git-installed)
|
||||
```
|
||||
|
||||
Profiles use `~/.hermes/profiles/<name>/` with the same layout.
|
||||
|
||||
### Config Sections
|
||||
|
||||
Edit with `hermes config edit` or `hermes config set section.key value`.
|
||||
|
||||
| Section | Key options |
|
||||
|---------|-------------|
|
||||
| `model` | `default`, `provider`, `base_url`, `api_key`, `context_length` |
|
||||
| `agent` | `max_turns` (90), `tool_use_enforcement` |
|
||||
| `terminal` | `backend` (local/docker/ssh/modal), `cwd`, `timeout` (180) |
|
||||
| `compression` | `enabled`, `threshold` (0.50), `target_ratio` (0.20) |
|
||||
| `display` | `skin`, `tool_progress`, `show_reasoning`, `show_cost` |
|
||||
| `stt` | `enabled`, `provider` (local/groq/openai/mistral) |
|
||||
| `tts` | `provider` (edge/elevenlabs/openai/minimax/mistral/neutts) |
|
||||
| `memory` | `memory_enabled`, `user_profile_enabled`, `provider` |
|
||||
| `security` | `tirith_enabled`, `website_blocklist` |
|
||||
| `delegation` | `model`, `provider`, `base_url`, `api_key`, `max_iterations` (50), `reasoning_effort` |
|
||||
| `checkpoints` | `enabled`, `max_snapshots` (50) |
|
||||
| `kanban` | `dispatch_in_gateway` (true), `dispatch_interval_seconds` (60) |
|
||||
| `dashboard` | `theme` (default\|midnight\|ember\|mono\|cyberpunk\|rose) |
|
||||
|
||||
Full config reference: https://hermes-agent.nousresearch.com/docs/user-guide/configuration
|
||||
|
||||
### Providers
|
||||
|
||||
20+ providers supported. Set via `hermes model` or `hermes setup`.
|
||||
|
||||
| Provider | Auth | Key env var |
|
||||
|----------|------|-------------|
|
||||
| OpenRouter | API key | `OPENROUTER_API_KEY` |
|
||||
| Anthropic | API key | `ANTHROPIC_API_KEY` |
|
||||
| Nous Portal | OAuth | `hermes auth` |
|
||||
| OpenAI Codex | OAuth | `hermes auth` |
|
||||
| GitHub Copilot | Token | `COPILOT_GITHUB_TOKEN` |
|
||||
| Google Gemini | API key | `GOOGLE_API_KEY` or `GEMINI_API_KEY` |
|
||||
| DeepSeek | API key | `DEEPSEEK_API_KEY` |
|
||||
| xAI / Grok | API key | `XAI_API_KEY` |
|
||||
| Hugging Face | Token | `HF_TOKEN` |
|
||||
| Z.AI / GLM | API key | `GLM_API_KEY` |
|
||||
| MiniMax | API key | `MINIMAX_API_KEY` |
|
||||
| MiniMax CN | API key | `MINIMAX_CN_API_KEY` |
|
||||
| Kimi / Moonshot | API key | `KIMI_API_KEY` |
|
||||
| Alibaba / DashScope | API key | `DASHSCOPE_API_KEY` |
|
||||
| Xiaomi MiMo | API key | `XIAOMI_API_KEY` |
|
||||
| Kilo Code | API key | `KILOCODE_API_KEY` |
|
||||
| AI Gateway (Vercel) | API key | `AI_GATEWAY_API_KEY` |
|
||||
| OpenCode Zen | API key | `OPENCODE_ZEN_API_KEY` |
|
||||
| OpenCode Go | API key | `OPENCODE_GO_API_KEY` |
|
||||
| Qwen OAuth | OAuth | `hermes login --provider qwen-oauth` |
|
||||
| Custom endpoint | Config | `model.base_url` + `model.api_key` in config.yaml — see `references/custom-openai-compatible-providers.md` for step-by-step |
|
||||
| GitHub Copilot ACP | External | `COPILOT_CLI_PATH` or Copilot CLI |
|
||||
|
||||
Full provider docs: https://hermes-agent.nousresearch.com/docs/integrations/providers
|
||||
|
||||
### Toolsets
|
||||
|
||||
Enable/disable via `hermes tools` (interactive) or `hermes tools enable/disable NAME`.
|
||||
|
||||
| Toolset | What it provides |
|
||||
|---------|-----------------|
|
||||
| `web` | Web search and content extraction |
|
||||
| `browser` | Browser automation (Browserbase, Camofox, or local Chromium) |
|
||||
| `terminal` | Shell commands and process management |
|
||||
| `file` | File read/write/search/patch |
|
||||
| `code_execution` | Sandboxed Python execution |
|
||||
| `vision` | Image analysis |
|
||||
| `image_gen` | AI image generation |
|
||||
| `tts` | Text-to-speech |
|
||||
| `skills` | Skill browsing and management |
|
||||
| `memory` | Persistent cross-session memory |
|
||||
| `session_search` | Search past conversations |
|
||||
| `delegation` | Subagent task delegation |
|
||||
| `cronjob` | Scheduled task management |
|
||||
| `clarify` | Ask user clarifying questions |
|
||||
| `messaging` | Cross-platform message sending |
|
||||
| `search` | Web search only (subset of `web`) |
|
||||
| `todo` | In-session task planning and tracking |
|
||||
| `rl` | Reinforcement learning tools (off by default) |
|
||||
| `moa` | Mixture of Agents (off by default) |
|
||||
| `homeassistant` | Smart home control (off by default) |
|
||||
|
||||
**Search tools:** For enhanced search capabilities (Tavily, Brave, SerpAPI, etc.), see `references/search-tools.md` for configuration examples and comparison.
|
||||
|
||||
Tool changes take effect on `/reset` (new session). They do NOT apply mid-conversation to preserve prompt caching.
|
||||
|
||||
---
|
||||
|
||||
## Security & Privacy Toggles
|
||||
|
||||
Common "why is Hermes doing X to my output / tool calls / commands?" toggles — and the exact commands to change them. Most of these need a fresh session (`/reset` in chat, or start a new `hermes` invocation) because they're read once at startup.
|
||||
|
||||
### Secret redaction in tool output
|
||||
|
||||
Secret redaction is **off by default** — tool output (terminal stdout, `read_file`, web content, subagent summaries, etc.) passes through unmodified. If the user wants Hermes to auto-mask strings that look like API keys, tokens, and secrets before they enter the conversation context and logs:
|
||||
|
||||
```bash
|
||||
hermes config set security.redact_secrets true # enable globally
|
||||
```
|
||||
|
||||
**Restart required.** `security.redact_secrets` is snapshotted at import time — toggling it mid-session (e.g. via `export HERMES_REDACT_SECRETS=true` from a tool call) will NOT take effect for the running process. Tell the user to run `hermes config set security.redact_secrets true` in a terminal, then start a new session. This is deliberate — it prevents an LLM from flipping the toggle on itself mid-task.
|
||||
|
||||
Disable again with:
|
||||
```bash
|
||||
hermes config set security.redact_secrets false
|
||||
```
|
||||
|
||||
### PII redaction in gateway messages
|
||||
|
||||
Separate from secret redaction. When enabled, the gateway hashes user IDs and strips phone numbers from the session context before it reaches the model:
|
||||
|
||||
```bash
|
||||
hermes config set privacy.redact_pii true # enable
|
||||
hermes config set privacy.redact_pii false # disable (default)
|
||||
```
|
||||
|
||||
### Command approval prompts
|
||||
|
||||
By default (`approvals.mode: manual`), Hermes prompts the user before running shell commands flagged as destructive (`rm -rf`, `git reset --hard`, etc.). The modes are:
|
||||
|
||||
- `manual` — always prompt (default)
|
||||
- `smart` — use an auxiliary LLM to auto-approve low-risk commands, prompt on high-risk
|
||||
- `off` — skip all approval prompts (equivalent to `--yolo`)
|
||||
|
||||
```bash
|
||||
hermes config set approvals.mode smart # recommended middle ground
|
||||
hermes config set approvals.mode off # bypass everything (not recommended)
|
||||
```
|
||||
|
||||
Per-invocation bypass without changing config:
|
||||
- `hermes --yolo …`
|
||||
- `export HERMES_YOLO_MODE=1`
|
||||
|
||||
Note: YOLO / `approvals.mode: off` does NOT turn off secret redaction. They are independent.
|
||||
|
||||
### Shell hooks allowlist
|
||||
|
||||
Some shell-hook integrations require explicit allowlisting before they fire. Managed via `~/.hermes/shell-hooks-allowlist.json` — prompted interactively the first time a hook wants to run.
|
||||
|
||||
### Disabling the web/browser/image-gen tools
|
||||
|
||||
To keep the model away from network or media tools entirely, open `hermes tools` and toggle per-platform. Takes effect on next session (`/reset`). See the Tools & Skills section above.
|
||||
|
||||
---
|
||||
|
||||
## Voice & Transcription
|
||||
|
||||
### STT (Voice → Text)
|
||||
|
||||
Voice messages from messaging platforms are auto-transcribed.
|
||||
|
||||
Provider priority (auto-detected):
|
||||
1. **Local faster-whisper** — free, no API key: `pip install faster-whisper`
|
||||
2. **Groq Whisper** — free tier: set `GROQ_API_KEY`
|
||||
3. **OpenAI Whisper** — paid: set `VOICE_TOOLS_OPENAI_KEY`
|
||||
4. **Mistral Voxtral** — set `MISTRAL_API_KEY`
|
||||
|
||||
Config:
|
||||
```yaml
|
||||
stt:
|
||||
enabled: true
|
||||
provider: local # local, groq, openai, mistral
|
||||
local:
|
||||
model: base # tiny, base, small, medium, large-v3
|
||||
```
|
||||
|
||||
### TTS (Text → Voice)
|
||||
|
||||
| Provider | Env var | Free? |
|
||||
|----------|---------|-------|
|
||||
| Edge TTS | None | Yes (default) |
|
||||
| ElevenLabs | `ELEVENLABS_API_KEY` | Free tier |
|
||||
| OpenAI | `VOICE_TOOLS_OPENAI_KEY` | Paid |
|
||||
| MiniMax | `MINIMAX_API_KEY` | Paid |
|
||||
| Mistral (Voxtral) | `MISTRAL_API_KEY` | Paid |
|
||||
| NeuTTS (local) | None (`pip install neutts[all]` + `espeak-ng`) | Free |
|
||||
|
||||
Voice commands: `/voice on` (voice-to-voice), `/voice tts` (always voice), `/voice off`.
|
||||
|
||||
---
|
||||
|
||||
## Spawning Additional Hermes Instances
|
||||
|
||||
Run additional Hermes processes as fully independent subprocesses — separate sessions, tools, and environments.
|
||||
|
||||
### When to Use This vs delegate_task
|
||||
|
||||
| | `delegate_task` | Spawning `hermes` process |
|
||||
|-|-----------------|--------------------------|
|
||||
| Isolation | Separate conversation, shared process | Fully independent process |
|
||||
| Duration | Minutes (bounded by parent loop) | Hours/days |
|
||||
| Tool access | Subset of parent's tools | Full tool access |
|
||||
| Interactive | No | Yes (PTY mode) |
|
||||
| Use case | Quick parallel subtasks | Long autonomous missions |
|
||||
|
||||
### One-Shot Mode
|
||||
|
||||
```
|
||||
terminal(command="hermes chat -q 'Research GRPO papers and write summary to ~/research/grpo.md'", timeout=300)
|
||||
|
||||
# Background for long tasks:
|
||||
terminal(command="hermes chat -q 'Set up CI/CD for ~/myapp'", background=true)
|
||||
```
|
||||
|
||||
### Interactive PTY Mode (via tmux)
|
||||
|
||||
Hermes uses prompt_toolkit, which requires a real terminal. Use tmux for interactive spawning:
|
||||
|
||||
```
|
||||
# Start
|
||||
terminal(command="tmux new-session -d -s agent1 -x 120 -y 40 'hermes'", timeout=10)
|
||||
|
||||
# Wait for startup, then send a message
|
||||
terminal(command="sleep 8 && tmux send-keys -t agent1 'Build a FastAPI auth service' Enter", timeout=15)
|
||||
|
||||
# Read output
|
||||
terminal(command="sleep 20 && tmux capture-pane -t agent1 -p", timeout=5)
|
||||
|
||||
# Send follow-up
|
||||
terminal(command="tmux send-keys -t agent1 'Add rate limiting middleware' Enter", timeout=5)
|
||||
|
||||
# Exit
|
||||
terminal(command="tmux send-keys -t agent1 '/exit' Enter && sleep 2 && tmux kill-session -t agent1", timeout=10)
|
||||
```
|
||||
|
||||
### Multi-Agent Coordination
|
||||
|
||||
```
|
||||
# Agent A: backend
|
||||
terminal(command="tmux new-session -d -s backend -x 120 -y 40 'hermes -w'", timeout=10)
|
||||
terminal(command="sleep 8 && tmux send-keys -t backend 'Build REST API for user management' Enter", timeout=15)
|
||||
|
||||
# Agent B: frontend
|
||||
terminal(command="tmux new-session -d -s frontend -x 120 -y 40 'hermes -w'", timeout=10)
|
||||
terminal(command="sleep 8 && tmux send-keys -t frontend 'Build React dashboard for user management' Enter", timeout=15)
|
||||
|
||||
# Check progress, relay context between them
|
||||
terminal(command="tmux capture-pane -t backend -p | tail -30", timeout=5)
|
||||
terminal(command="tmux send-keys -t frontend 'Here is the API schema from the backend agent: ...' Enter", timeout=5)
|
||||
```
|
||||
|
||||
### Session Resume
|
||||
|
||||
```
|
||||
# Resume most recent session
|
||||
terminal(command="tmux new-session -d -s resumed 'hermes --continue'", timeout=10)
|
||||
|
||||
# Resume specific session
|
||||
terminal(command="tmux new-session -d -s resumed 'hermes --resume 20260225_143052_a1b2c3'", timeout=10)
|
||||
```
|
||||
|
||||
### Tips
|
||||
|
||||
- **Prefer `delegate_task` for quick subtasks** — less overhead than spawning a full process
|
||||
- **Use `-w` (worktree mode)** when spawning agents that edit code — prevents git conflicts
|
||||
- **Set timeouts** for one-shot mode — complex tasks can take 5-10 minutes
|
||||
- **Use `hermes chat -q` for fire-and-forget** — no PTY needed
|
||||
- **Use tmux for interactive sessions** — raw PTY mode has `\r` vs `\n` issues with prompt_toolkit
|
||||
- **For scheduled tasks**, use the `cronjob` tool instead of spawning — handles delivery and retry
|
||||
|
||||
**RSS cron job timing — race condition pitfall:** When scheduling a cron job that fetches an RSS feed, the RSS source may update at a specific time each day. If the cron job runs too close to the RSS update time, it will fetch stale (yesterday's) content. Always set the cron job **at least 30 minutes after** the expected RSS update time. Example: 橘鸦AI早报 RSS updates ~09:30 Beijing time → cron should be `0 10 * * *` or later, not `30 9 * * *`. To diagnose: `curl -s <rss_url> | grep '<pubDate>' | head -1` to check the latest update timestamp, then compare with `hermes cron list` → `last_run_at`.
|
||||
|
||||
---
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
### Voice not working
|
||||
1. Check `stt.enabled: true` in config.yaml
|
||||
2. Verify provider: `pip install faster-whisper` or set API key
|
||||
3. In gateway: `/restart`. In CLI: exit and relaunch.
|
||||
|
||||
### Tool not available
|
||||
1. `hermes tools` — check if toolset is enabled for your platform
|
||||
2. Some tools need env vars (check `.env`)
|
||||
3. `/reset` after enabling tools
|
||||
|
||||
### Model/provider issues
|
||||
1. `hermes doctor` — check config and dependencies
|
||||
2. `hermes login` — re-authenticate OAuth providers
|
||||
3. Check `.env` has the right API key
|
||||
4. **Copilot 403**: `gh auth login` tokens do NOT work for Copilot API. You must use the Copilot-specific OAuth device code flow via `hermes model` → GitHub Copilot.
|
||||
|
||||
### Changes not taking effect
|
||||
- **Tools/skills:** `/reset` starts a new session with updated toolset
|
||||
- **Config changes:** In gateway: `/restart`. In CLI: exit and relaunch.
|
||||
- **Code changes:** Restart the CLI or gateway process
|
||||
|
||||
### Stale autostashes after `hermes update`
|
||||
`hermes update` auto-stashes local changes. These accumulate — lock-file-only stashes are safe to drop, source-code stashes need feature-by-feature comparison against current code. See `references/hermes-update-autostash-triage.md` for the full triage workflow.
|
||||
|
||||
### Skills not showing
|
||||
1. `hermes skills list` — verify installed
|
||||
2. `hermes skills config` — check platform enablement
|
||||
3. Load explicitly: `/skill name` or `hermes -s name`
|
||||
|
||||
### Skill library management pitfalls
|
||||
See `references/skill-management-pitfalls.md` for hard-won lessons on deleting skills, writing descriptions, checking dependencies, and avoiding cross-references.
|
||||
|
||||
### Gateway issues
|
||||
Check logs first:
|
||||
```bash
|
||||
grep -i "failed to send\|error" ~/.hermes/logs/gateway.log | tail -20
|
||||
```
|
||||
|
||||
Common gateway problems:
|
||||
- **Gateway dies on SSH logout**: Enable linger: `sudo loginctl enable-linger $USER`
|
||||
- **Gateway dies on WSL2 close**: WSL2 requires `systemd=true` in `/etc/wsl.conf` for systemd services to work. Without it, gateway falls back to `nohup` (dies when session closes).
|
||||
- **Gateway crash loop**: Reset the failed state: `systemctl --user reset-failed hermes-gateway`
|
||||
|
||||
### Platform-specific issues
|
||||
- **Discord bot silent**: Must enable **Message Content Intent** in Bot → Privileged Gateway Intents.
|
||||
- **Slack bot only works in DMs**: Must subscribe to `message.channels` event. Without it, the bot ignores public channels.
|
||||
- **Windows HTTP 400 "No models provided"**: Config file encoding issue (BOM). Ensure `config.yaml` is saved as UTF-8 without BOM.
|
||||
|
||||
### Scraping JS-rendered pages
|
||||
|
||||
When `curl` returns empty or minimal content (just CSS/fonts, no actual data), the page likely requires JavaScript rendering. Use Playwright as a fallback:
|
||||
|
||||
```bash
|
||||
# Install once
|
||||
cd /tmp && npm install playwright && npx playwright install chromium
|
||||
|
||||
# Create a scraper script
|
||||
cat > /tmp/fetch_page.js << 'EOF'
|
||||
const { chromium } = require('playwright');
|
||||
(async () => {
|
||||
const browser = await chromium.launch({ headless: true });
|
||||
const page = await browser.newPage();
|
||||
await page.goto(process.argv[2], { waitUntil: 'networkidle', timeout: 30000 });
|
||||
await page.waitForTimeout(3000);
|
||||
console.log(await page.evaluate(() => document.body.innerText));
|
||||
await browser.close();
|
||||
})();
|
||||
EOF
|
||||
|
||||
# Usage
|
||||
node /tmp/fetch_page.js "https://example.com/page"
|
||||
```
|
||||
|
||||
This pattern applies to sites like xiaoheihe.cn, SPAs, and any page that loads content via JavaScript after initial HTML.
|
||||
|
||||
### Auxiliary models not working
|
||||
If `auxiliary` tasks (vision, compression, session_search) fail silently, the `auto` provider can't find a backend. Either set `OPENROUTER_API_KEY` or `GOOGLE_API_KEY`, or explicitly configure each auxiliary task's provider:
|
||||
```bash
|
||||
hermes config set auxiliary.vision.provider <your_provider>
|
||||
hermes config set auxiliary.vision.model <model_name>
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Pitfalls
|
||||
|
||||
**NEVER modify server security settings without explicit user permission.** This includes:
|
||||
- SSH password authentication (`PasswordAuthentication` in sshd_config)
|
||||
- Firewall rules (ufw, iptables)
|
||||
- Opening ports to public networks
|
||||
- Changing SSH ports
|
||||
- Modifying user passwords
|
||||
|
||||
**Always ask first.** Even if the user asks for help connecting, present options and let them choose. Do not assume they want to weaken security.
|
||||
|
||||
**Dashboard reverse proxy:** See `references/dashboard-reverse-proxy.md` for Nginx + basic auth setup.
|
||||
|
||||
**Passwords and secrets:** When users provide passwords or sensitive credentials, do NOT save them to memory. Use them only for the immediate task and let the user manage their own secrets.
|
||||
|
||||
**Security-sensitive operations checklist:**
|
||||
1. Is this modifying access controls? → Ask first
|
||||
2. Is this exposing services to the internet? → Ask first
|
||||
3. Is this changing authentication methods? → Ask first
|
||||
4. Is this storing credentials? → Ask first
|
||||
|
||||
When in doubt, present the options and let the user decide.
|
||||
|
||||
### Pitfall: send_message MEDIA delivery per platform
|
||||
|
||||
Not all platforms support `MEDIA:<path>` in `send_message` tool. The tool's `_send_to_platform()` function in `tools/send_message_tool.py` has per-platform routing:
|
||||
|
||||
- **Full media support**: telegram, discord, matrix, weixin, signal, yuanbao, feishu
|
||||
- **Text-only (media silently dropped)**: qqbot, slack, whatsapp, email, sms, and others
|
||||
|
||||
When a platform's `send_message` doesn't deliver media, check if the gateway adapter (`gateway/platforms/<name>/adapter.py`) has `send_document`/`send_image` methods. If it does but the tool doesn't route through it, the fix is to add a `_send_<platform>()` variant that uses `get_active_adapter()` (same pattern as `_send_yuanbao()` in send_message_tool.py).
|
||||
|
||||
Known gap (as of May 2026): QQ Bot gateway adapter has full media support but `send_message` tool's `_send_qqbot()` ignores `media_files`. PRD: `ephron-ren-qa/prd-qqbot-media-support.md`.
|
||||
|
||||
### Where to Find Things
|
||||
|
||||
| Looking for... | Location |
|
||||
|----------------|----------|
|
||||
| Config options | `hermes config edit` or [Configuration docs](https://hermes-agent.nousresearch.com/docs/user-guide/configuration) |
|
||||
| Available tools | `hermes tools list` or [Tools reference](https://hermes-agent.nousresearch.com/docs/reference/tools-reference) |
|
||||
| Slash commands | `/help` in session or [Slash commands reference](https://hermes-agent.nousresearch.com/docs/reference/slash-commands) |
|
||||
| Skills catalog | `hermes skills browse` or [Skills catalog](https://hermes-agent.nousresearch.com/docs/reference/skills-catalog) |
|
||||
| Provider setup | `hermes model` or [Providers guide](https://hermes-agent.nousresearch.com/docs/integrations/providers) |
|
||||
| Platform setup | `hermes gateway setup` or [Messaging docs](https://hermes-agent.nousresearch.com/docs/user-guide/messaging/) |
|
||||
| MCP servers | `hermes mcp list` or [MCP guide](https://hermes-agent.nousresearch.com/docs/user-guide/features/mcp) |
|
||||
| Profiles | `hermes profile list` or [Profiles docs](https://hermes-agent.nousresearch.com/docs/user-guide/profiles) |
|
||||
| Cron jobs | `hermes cron list` or [Cron docs](https://hermes-agent.nousresearch.com/docs/user-guide/features/cron) |
|
||||
| Memory | `hermes memory status` or [Memory docs](https://hermes-agent.nousresearch.com/docs/user-guide/features/memory) |
|
||||
| Env variables | `hermes config env-path` or [Env vars reference](https://hermes-agent.nousresearch.com/docs/reference/environment-variables) |
|
||||
| CLI commands | `hermes --help` or [CLI reference](https://hermes-agent.nousresearch.com/docs/reference/cli-commands) |
|
||||
| Gateway logs | `~/.hermes/logs/gateway.log` |
|
||||
| Session files | `~/.hermes/sessions/` or `hermes sessions browse` |
|
||||
| Source code | `~/.hermes/hermes-agent/` |
|
||||
|
||||
---
|
||||
|
||||
## Contributor Quick Reference
|
||||
|
||||
For occasional contributors and PR authors. Full developer docs: https://hermes-agent.nousresearch.com/docs/developer-guide/
|
||||
|
||||
### Project Layout
|
||||
|
||||
```
|
||||
hermes-agent/
|
||||
├── run_agent.py # AIAgent — core conversation loop
|
||||
├── model_tools.py # Tool discovery and dispatch
|
||||
├── toolsets.py # Toolset definitions
|
||||
├── cli.py # Interactive CLI (HermesCLI)
|
||||
├── hermes_state.py # SQLite session store
|
||||
├── agent/ # Prompt builder, context compression, memory, model routing, credential pooling, skill dispatch
|
||||
├── hermes_cli/ # CLI subcommands, config, setup, commands
|
||||
│ ├── commands.py # Slash command registry (CommandDef)
|
||||
│ ├── config.py # DEFAULT_CONFIG, env var definitions
|
||||
│ └── main.py # CLI entry point and argparse
|
||||
├── tools/ # One file per tool
|
||||
│ └── registry.py # Central tool registry
|
||||
├── gateway/ # Messaging gateway
|
||||
│ └── platforms/ # Platform adapters (telegram, discord, etc.)
|
||||
├── cron/ # Job scheduler
|
||||
├── tests/ # ~3000 pytest tests
|
||||
└── website/ # Docusaurus docs site
|
||||
```
|
||||
|
||||
Config: `~/.hermes/config.yaml` (settings), `~/.hermes/.env` (API keys).
|
||||
|
||||
### Adding a Tool (3 files)
|
||||
|
||||
**1. Create `tools/your_tool.py`:**
|
||||
```python
|
||||
import json, os
|
||||
from tools.registry import registry
|
||||
|
||||
def check_requirements() -> bool:
|
||||
return bool(os.getenv("EXAMPLE_API_KEY"))
|
||||
|
||||
def example_tool(param: str, task_id: str = None) -> str:
|
||||
return json.dumps({"success": True, "data": "..."})
|
||||
|
||||
registry.register(
|
||||
name="example_tool",
|
||||
toolset="example",
|
||||
schema={"name": "example_tool", "description": "...", "parameters": {...}},
|
||||
handler=lambda args, **kw: example_tool(
|
||||
param=args.get("param", ""), task_id=kw.get("task_id")),
|
||||
check_fn=check_requirements,
|
||||
requires_env=["EXAMPLE_API_KEY"],
|
||||
)
|
||||
```
|
||||
|
||||
**2. Add to `toolsets.py`** → `_HERMES_CORE_TOOLS` list.
|
||||
|
||||
Auto-discovery: any `tools/*.py` file with a top-level `registry.register()` call is imported automatically — no manual list needed.
|
||||
|
||||
All handlers must return JSON strings. Use `get_hermes_home()` for paths, never hardcode `~/.hermes`.
|
||||
|
||||
### Adding a Slash Command
|
||||
|
||||
1. Add `CommandDef` to `COMMAND_REGISTRY` in `hermes_cli/commands.py`
|
||||
2. Add handler in `cli.py` → `process_command()`
|
||||
3. (Optional) Add gateway handler in `gateway/run.py`
|
||||
|
||||
All consumers (help text, autocomplete, Telegram menu, Slack mapping) derive from the central registry automatically.
|
||||
|
||||
### Agent Loop (High Level)
|
||||
|
||||
```
|
||||
run_conversation():
|
||||
1. Build system prompt
|
||||
2. Loop while iterations < max:
|
||||
a. Call LLM (OpenAI-format messages + tool schemas)
|
||||
b. If tool_calls → dispatch each via handle_function_call() → append results → continue
|
||||
c. If text response → return
|
||||
3. Context compression triggers automatically near token limit
|
||||
```
|
||||
|
||||
### Testing
|
||||
|
||||
```bash
|
||||
python -m pytest tests/ -o 'addopts=' -q # Full suite
|
||||
python -m pytest tests/tools/ -q # Specific area
|
||||
```
|
||||
|
||||
- Tests auto-redirect `HERMES_HOME` to temp dirs — never touch real `~/.hermes/`
|
||||
- Run full suite before pushing any change
|
||||
- Use `-o 'addopts='` to clear any baked-in pytest flags
|
||||
|
||||
### Commit Conventions
|
||||
|
||||
```
|
||||
type: concise subject line
|
||||
|
||||
Optional body.
|
||||
```
|
||||
|
||||
Types: `fix:`, `feat:`, `refactor:`, `docs:`, `chore:`
|
||||
|
||||
### Key Rules
|
||||
|
||||
- **Never break prompt caching** — don't change context, tools, or system prompt mid-conversation
|
||||
- **Message role alternation** — never two assistant or two user messages in a row
|
||||
- Use `get_hermes_home()` from `hermes_constants` for all paths (profile-safe)
|
||||
- Config values go in `config.yaml`, secrets go in `.env`
|
||||
- New tools need a `check_fn` so they only appear when requirements are met
|
||||
@@ -0,0 +1,81 @@
|
||||
# Custom OpenAI-Compatible Providers in Hermes
|
||||
|
||||
When a provider isn't built-in but offers an OpenAI-compatible `/v1` endpoint, add it manually to `config.yaml`.
|
||||
|
||||
## Steps
|
||||
|
||||
1. **Find the base URL and API key env var** — usually `/v1` at the provider's domain.
|
||||
|
||||
2. **List available models:**
|
||||
```bash
|
||||
curl -s "https://<provider>/v1/models" \
|
||||
-H "Authorization: Bearer $API_KEY" \
|
||||
--max-time 15 | python3 -m json.tool
|
||||
```
|
||||
Note model `id`, `input_modalities`, `context_length`, `supported_features`.
|
||||
|
||||
3. **Add provider to config.yaml via Python** (don't hand-edit YAML — indentation errors break everything):
|
||||
```python
|
||||
import yaml, json, os
|
||||
|
||||
config_path = os.path.expanduser("~/.hermes/config.yaml")
|
||||
with open(config_path) as f:
|
||||
config = yaml.safe_load(f)
|
||||
|
||||
# Read API key from .env
|
||||
api_key = ""
|
||||
with open(os.path.expanduser("~/.hermes/.env")) as f:
|
||||
for line in f:
|
||||
if line.startswith("YOUR_KEY_PREFIX="):
|
||||
api_key = line.strip().split("=", 1)[1]
|
||||
break
|
||||
|
||||
config.setdefault("providers", {})["your-provider"] = {
|
||||
"api_key": api_key,
|
||||
"base_url": "https://provider.example.com/v1",
|
||||
"available_models_json": json.dumps([
|
||||
{"id": "model-id", "name": "Display Name"},
|
||||
]),
|
||||
"model": "default-model-id",
|
||||
"model_display_name": "Default Display Name"
|
||||
}
|
||||
|
||||
with open(config_path, "w") as f:
|
||||
yaml.dump(config, f, default_flow_style=False, allow_unicode=True, sort_keys=False)
|
||||
```
|
||||
|
||||
4. **Verify with a test call:**
|
||||
```bash
|
||||
curl -s "https://provider.example.com/v1/chat/completions" \
|
||||
-H "Authorization: Bearer $API_KEY" \
|
||||
-H "Content-Type: application/json" \
|
||||
-d '{"model":"model-id","messages":[{"role":"user","content":"hi"}],"max_tokens":100}'
|
||||
```
|
||||
|
||||
5. **Use the model:** `hermes -m your-provider/model-id` or via `hermes model` picker.
|
||||
|
||||
## Provider Config Fields
|
||||
|
||||
| Field | Required | Notes |
|
||||
|-------|----------|-------|
|
||||
| `api_key` | Yes | Actual key value, not env var reference |
|
||||
| `base_url` | Yes | Must end with `/v1` (or `/v1/`) |
|
||||
| `available_models_json` | Yes | JSON string of `[{id, name}]` array |
|
||||
| `model` | No | Default model ID |
|
||||
| `model_display_name` | No | Human-readable default model name |
|
||||
| `api_mode` | No | Only set if non-standard (e.g. `anthropic-messages` for MiniMax). Omit for OpenAI-compatible. |
|
||||
| `protocol` | No | Usually leave as `''` |
|
||||
|
||||
## Pitfalls
|
||||
|
||||
- **Don't hand-edit YAML** — use Python `yaml.safe_load` + `yaml.dump` to avoid indentation corruption.
|
||||
- **`api_key` must be the actual value**, not `$ENV_VAR` — Hermes doesn't resolve env vars inside provider config (only in `.env`).
|
||||
- **No `api_mode` needed for OpenAI-compatible** — only set this for providers with custom protocols (Anthropic Messages, etc.).
|
||||
- **`reasoning` field in responses** — some providers (SenseNova, DeepSeek) return a `reasoning` field in the message object. Hermes handles this natively for reasoning-capable models.
|
||||
- **Model discovery** — always call `/v1/models` first; don't guess model IDs from documentation (they change).
|
||||
|
||||
## Known Custom Providers
|
||||
|
||||
| Provider | Base URL | Key Env Var | Models |
|
||||
|----------|----------|-------------|--------|
|
||||
| SenseNova | `https://token.sensenova.cn/v1` | `SN_API_KEY` | sensenova-6.7-flash-lite, deepseek-v4-flash, sensenova-u1-fast |
|
||||
@@ -0,0 +1,68 @@
|
||||
# Dashboard Remote Access
|
||||
|
||||
## Problem
|
||||
Dashboard binds to 127.0.0.1:9119 by default. Accessing from a different machine (e.g., local laptop → cloud VPS) requires either SSH tunnel or insecure bind.
|
||||
|
||||
## Recommended: SSH Port Forwarding
|
||||
```bash
|
||||
# On your local machine
|
||||
ssh -L 9119:127.0.0.1:9119 user@server-ip
|
||||
# Then open http://127.0.0.1:9119 in browser
|
||||
```
|
||||
**Pitfall (Windows):** `ssh: connect to host ... port 22: Connection timed out` — almost always a cloud security group issue. Check your cloud provider's security group / firewall rules to allow inbound TCP 22. SSH socket activation (`ssh.socket`) is enabled by default on Ubuntu; the service itself may show `inactive (dead)` — that's normal, socket activation triggers it on connection.
|
||||
|
||||
## Password Protection (Reverse Proxy)
|
||||
|
||||
Dashboard has **no built-in password auth**. Options:
|
||||
|
||||
### Nginx + Basic Auth
|
||||
```bash
|
||||
sudo apt install nginx apache2-utils
|
||||
sudo htpasswd -c /etc/nginx/.htpasswd your-username
|
||||
```
|
||||
```nginx
|
||||
server {
|
||||
listen 8080;
|
||||
location / {
|
||||
auth_basic "Hermes Dashboard";
|
||||
auth_basic_user_file /etc/nginx/.htpasswd;
|
||||
proxy_pass http://127.0.0.1:9119;
|
||||
proxy_set_header Host $host;
|
||||
proxy_set_header X-Real-IP $remote_addr;
|
||||
# WebSocket support for Chat TUI
|
||||
proxy_http_version 1.1;
|
||||
proxy_set_header Upgrade $http_upgrade;
|
||||
proxy_set_header Connection "upgrade";
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### Caddy (simpler config)
|
||||
```
|
||||
:8080 {
|
||||
basicauth * {
|
||||
username $hashed_password
|
||||
}
|
||||
reverse_proxy localhost:9119
|
||||
}
|
||||
```
|
||||
Generate hash: `caddy hash-password --plaintext 'your-password'`
|
||||
|
||||
## Alternative: Insecure Bind (⚠️ exposes API keys)
|
||||
```bash
|
||||
hermes dashboard --insecure --port 9119
|
||||
# Access via http://server-ip:9119
|
||||
```
|
||||
Only use on trusted/private networks. The dashboard exposes `.env` contents including API keys. The `--insecure` flag exists because there's no built-in auth — the warning is intentional.
|
||||
|
||||
## TUI Mode (Embedded Chat)
|
||||
```bash
|
||||
hermes dashboard --tui --no-open
|
||||
```
|
||||
Adds a Chat tab to the web UI — a browser-based `hermes --tui` via PTY/WebSocket. Useful when CLI access is inconvenient.
|
||||
|
||||
## Common Issues
|
||||
- Multiple dashboard processes: `hermes dashboard --stop` kills all
|
||||
- Port conflict: change port with `--port 8080`
|
||||
- Gateway must be running for Kanban dispatch to work (`hermes gateway status`)
|
||||
- SSH connection timeout from Windows: check cloud security group allows inbound TCP 22
|
||||
@@ -0,0 +1,110 @@
|
||||
# Hermes Dashboard Reverse Proxy with Nginx
|
||||
|
||||
## Quick Setup (Nginx + Basic Auth)
|
||||
|
||||
### 1. Install Dependencies
|
||||
```bash
|
||||
sudo apt update && sudo apt install -y nginx apache2-utils
|
||||
```
|
||||
|
||||
### 2. Create Password File
|
||||
```bash
|
||||
# Generate password (will prompt for password twice)
|
||||
sudo htpasswd -c /etc/nginx/.htpasswd <username>
|
||||
|
||||
# Or non-interactive:
|
||||
echo -n '<username>:' | sudo tee /etc/nginx/.htpasswd
|
||||
openssl passwd -apr1 '<password>' | sudo tee -a /etc/nginx/.htpasswd
|
||||
```
|
||||
|
||||
### 3. Nginx Config (`/etc/nginx/sites-available/hermes-dashboard`)
|
||||
```nginx
|
||||
server {
|
||||
listen 80;
|
||||
server_name <your-domain-or-ip>; # e.g., 111.230.53.30 or hermes.example.com
|
||||
|
||||
location / {
|
||||
auth_basic "Hermes Dashboard";
|
||||
auth_basic_user_file /etc/nginx/.htpasswd;
|
||||
|
||||
proxy_pass http://127.0.0.1:9119;
|
||||
# IMPORTANT: Use "localhost" for Host header, NOT $host
|
||||
# Dashboard validates Host header and rejects non-localhost values
|
||||
# This causes "Invalid Host header" error if set to $host
|
||||
proxy_set_header Host localhost;
|
||||
proxy_set_header X-Real-IP $remote_addr;
|
||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||
proxy_set_header X-Forwarded-Proto $scheme;
|
||||
|
||||
# WebSocket support (required for Chat TUI)
|
||||
proxy_http_version 1.1;
|
||||
proxy_set_header Upgrade $http_upgrade;
|
||||
proxy_set_header Connection "upgrade";
|
||||
|
||||
proxy_connect_timeout 60s;
|
||||
proxy_send_timeout 60s;
|
||||
proxy_read_timeout 60s;
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 4. Enable & Reload
|
||||
```bash
|
||||
sudo ln -sf /etc/nginx/sites-available/hermes-dashboard /etc/nginx/sites-enabled/
|
||||
sudo nginx -t && sudo systemctl reload nginx
|
||||
```
|
||||
|
||||
### 5. Ensure Dashboard Running
|
||||
```bash
|
||||
hermes dashboard --no-open --port 9119
|
||||
```
|
||||
|
||||
## Access
|
||||
- URL: `http://<domain-or-ip>`
|
||||
- Auth: Browser popup for username/password
|
||||
|
||||
## Commands
|
||||
```bash
|
||||
sudo systemctl status nginx
|
||||
sudo systemctl restart nginx
|
||||
hermes dashboard --status
|
||||
hermes dashboard --stop
|
||||
```
|
||||
|
||||
## Cleanup (Remove Reverse Proxy)
|
||||
```bash
|
||||
# Stop services
|
||||
hermes dashboard --stop
|
||||
sudo systemctl stop nginx
|
||||
sudo systemctl disable nginx
|
||||
|
||||
# Remove config files
|
||||
sudo rm -f /etc/nginx/sites-available/hermes-dashboard
|
||||
sudo rm -f /etc/nginx/sites-enabled/hermes-dashboard
|
||||
sudo rm -f /etc/nginx/.htpasswd
|
||||
```
|
||||
|
||||
## HTTPS (Optional)
|
||||
Use Certbot for Let's Encrypt:
|
||||
```bash
|
||||
sudo apt install certbot python3-certbot-nginx
|
||||
sudo certbot --nginx -d hermes.example.com
|
||||
```
|
||||
|
||||
## Pitfalls
|
||||
|
||||
### Invalid Host Header Error
|
||||
If you see `{"detail":"Invalid Host header. Dashboard requests must use the hostname the server was bound to."}`:
|
||||
- **Cause**: Nginx is passing `$host` (the public domain/IP) but Dashboard only accepts `localhost`
|
||||
- **Fix**: Change `proxy_set_header Host $host;` to `proxy_set_header Host localhost;`
|
||||
|
||||
### Domain Requires ICP Filing (China)
|
||||
If accessing via domain in China triggers ICP filing requirement:
|
||||
- **Solution**: Use IP address directly instead of domain
|
||||
- Update `server_name` to the server's public IP
|
||||
|
||||
### Security Notes
|
||||
- Dashboard has NO built-in password auth
|
||||
- Without reverse proxy, anyone with network access can see API keys
|
||||
- Always use reverse proxy + basic auth for remote access
|
||||
- Consider SSH port forwarding as a more secure alternative
|
||||
@@ -0,0 +1,76 @@
|
||||
# Hermes Update Autostash Triage
|
||||
|
||||
`hermes update` auto-stashes local changes before pulling. These accumulate as `stash@{N}` with the naming pattern:
|
||||
|
||||
```
|
||||
hermes-update-autostash-YYYYMMDD-HHMMSS
|
||||
```
|
||||
|
||||
## Triage Workflow
|
||||
|
||||
### Step 1: List all stashes
|
||||
|
||||
```bash
|
||||
git stash list
|
||||
```
|
||||
|
||||
### Step 2: Quick scan each stash — file types matter
|
||||
|
||||
```bash
|
||||
git stash show stash@{N} --stat
|
||||
```
|
||||
|
||||
**Lock-file-only stashes** (only `package.json`, `package-lock.json`, `ui-tui/package-lock.json`):
|
||||
- Usually npm dependency resolution artifacts (registry mirror switches, peer dependency reclassification)
|
||||
- Safe to drop: `git stash drop stash@{N}`
|
||||
|
||||
**Source-code stashes** (`.py`, `.ts`, `.tsx` files changed):
|
||||
- Need detailed analysis — these may contain valuable local features
|
||||
|
||||
### Step 3: For source-code stashes — compare against current code
|
||||
|
||||
Don't just `git stash pop`. First check if the features were already merged upstream:
|
||||
|
||||
```bash
|
||||
# Get the full diff
|
||||
git stash show -p stash@{N}
|
||||
|
||||
# For each key feature, search current code:
|
||||
grep -n "feature_keyword" path/to/file.py
|
||||
```
|
||||
|
||||
**Classification:**
|
||||
- ✅ Already in current code → safe to drop
|
||||
- ❌ Missing from current code → candidate for restoration
|
||||
|
||||
### Step 4: Decision matrix
|
||||
|
||||
| Stash type | Action |
|
||||
|------------|--------|
|
||||
| Lock files only | Drop immediately |
|
||||
| Source code, all features merged | Drop |
|
||||
| Source code, some features missing | Selective restore (cherry-pick specific hunks) or apply + resolve conflicts |
|
||||
| Source code, all features missing | `git stash apply stash@{N}` then test |
|
||||
|
||||
### Pitfalls
|
||||
|
||||
- **Don't blindly pop stashes on an active branch** — always `apply` first (preserves stash), test, then `drop` if good.
|
||||
- **Registry mirror changes in lock files** (npmmirror.com, mirrors.tencentyun.com) are local environment artifacts, not valuable code. Drop them.
|
||||
- **`peer: true` removal** in lock files = npm re-resolved peer deps as direct deps. Not meaningful.
|
||||
- **5+ day old stashes** with source changes are likely abandoned experiments. Check if the user still needs them before restoring.
|
||||
- **Merge conflicts** are common after 5+ days — upstream moves fast. Expect to resolve manually.
|
||||
|
||||
### Restoring selectively
|
||||
|
||||
If only some hunks from a stash are needed:
|
||||
|
||||
```bash
|
||||
# Apply but don't drop
|
||||
git stash apply stash@{N}
|
||||
|
||||
# Review conflicts
|
||||
git diff
|
||||
|
||||
# Or use interactive checkout for specific files
|
||||
git checkout stash@{N} -- path/to/specific/file.py
|
||||
```
|
||||
@@ -0,0 +1,61 @@
|
||||
# QQ Bot Rich Media API Reference
|
||||
|
||||
Source: https://bot.q.qq.com/wiki/develop/api-v2/server-inter/message/send-receive/rich-media.html
|
||||
|
||||
## Endpoints
|
||||
|
||||
| Scope | Endpoint | Method |
|
||||
|-------|----------|--------|
|
||||
| 单聊 | `/v2/users/{openid}/files` | POST |
|
||||
| 群聊 | `/v2/groups/{group_openid}/files` | POST |
|
||||
|
||||
## Parameters
|
||||
|
||||
| Field | Type | Required | Description |
|
||||
|-------|------|----------|-------------|
|
||||
| file_type | int | Yes | 1=图片, 2=视频, 3=语音, 4=文件 |
|
||||
| url | string | Yes* | 媒体资源 URL(*url 或 file_data 二选一) |
|
||||
| file_data | string | No | base64 二进制数据 |
|
||||
| srv_send_msg | bool | Yes | true=直接发送(占用主动消息频次),false=仅上传获取 file_info |
|
||||
| file_name | string | No | 文件名(file_type=4 时建议传) |
|
||||
|
||||
## Supported Formats
|
||||
|
||||
- 图片: png/jpg
|
||||
- 视频: mp4
|
||||
- 语音: silk/wav/mp3/flac
|
||||
- 文件: 无限制(群场景暂不开放 file_type=4)
|
||||
|
||||
## Response
|
||||
|
||||
```json
|
||||
{
|
||||
"file_uuid": "...",
|
||||
"file_info": "...", // 用于发送消息接口的 media 字段
|
||||
"ttl": 3600 // 剩余秒数,0=长期有效
|
||||
}
|
||||
```
|
||||
|
||||
## Sending with file_info
|
||||
|
||||
```json
|
||||
{
|
||||
"msg_type": 7,
|
||||
"media": {"file_info": "<file_info_from_upload>"},
|
||||
"msg_seq": 1
|
||||
}
|
||||
```
|
||||
|
||||
## Notes
|
||||
|
||||
- `file_info` 不受目标端影响,可复用到多个群/用户
|
||||
- 用 `/v2/users/{openid}/files` 上传的仅能发单聊,群上传的仅能发群聊
|
||||
- 建议 `srv_send_msg=false`,先获取 file_info 再发送
|
||||
- 文件大小限制:~100MB(分块上传),~10MB(inline base64)
|
||||
|
||||
## Hermes Adapter Implementation
|
||||
|
||||
- Gateway adapter: `gateway/platforms/qqbot/adapter.py` → `_send_media()` (line ~2690)
|
||||
- Chunked upload: `gateway/platforms/qqbot/chunked_upload.py`
|
||||
- Media types defined in `gateway/platforms/qqbot/constants.py` (MEDIA_TYPE_IMAGE=1, VIDEO=2, VOICE=3, FILE=4)
|
||||
- send_message tool gap: `_send_qqbot()` in `tools/send_message_tool.py:1677` is text-only
|
||||
96
autonomous-ai-agents/hermes-agent/references/search-tools.md
Normal file
96
autonomous-ai-agents/hermes-agent/references/search-tools.md
Normal file
@@ -0,0 +1,96 @@
|
||||
# Search Tools for Hermes Agent
|
||||
|
||||
## Current Environment
|
||||
|
||||
| Tool | Type | Usage |
|
||||
|------|------|-------|
|
||||
| **web** (built-in) | Web Search & Scraping | `web_search`, `web_extract` tools |
|
||||
| **session_search** | Session history | `session_search(query="...")` |
|
||||
| **mmx search** | MiniMax CLI | `mmx search query "关键词"` |
|
||||
|
||||
## Popular Search APIs for AI Agents
|
||||
|
||||
| Tool | Best For | Pricing | MCP Server |
|
||||
|------|----------|---------|------------|
|
||||
| **Tavily Search** | AI-native search, structured results | Free tier available | `tavily-mcp` |
|
||||
| **SerpAPI** | Google results scraping | Paid (100 free/month) | `serpapi-mcp` |
|
||||
| **Brave Search** | Privacy-focused, no tracking | Free tier (2000/month) | `brave-search-mcp` |
|
||||
| **Perplexity API** | AI search with citations | Paid | API only |
|
||||
| **Firecrawl** | Web scraping + extraction | Free tier | `firecrawl-mcp` |
|
||||
| **Jina Reader** | URL to Markdown conversion | Free tier | API only |
|
||||
| **SearXNG** | Self-hosted meta search | Free (self-hosted) | `searxng-mcp` |
|
||||
|
||||
## Configuration Examples
|
||||
|
||||
### Tavily Search (Recommended for AI Agents)
|
||||
|
||||
```yaml
|
||||
# ~/.hermes/config.yaml
|
||||
mcp_servers:
|
||||
tavily:
|
||||
command: npx
|
||||
args: ["-y", "tavily-mcp@latest"]
|
||||
env:
|
||||
TAVILY_API_KEY: "tvly-xxxxx"
|
||||
```
|
||||
|
||||
Get API key: https://tavily.com (free tier: 1000 searches/month)
|
||||
|
||||
### Brave Search
|
||||
|
||||
```yaml
|
||||
mcp_servers:
|
||||
brave-search:
|
||||
command: npx
|
||||
args: ["-y", "@anthropic/brave-search-mcp@latest"]
|
||||
env:
|
||||
BRAVE_API_KEY: "BSAxxxxx"
|
||||
```
|
||||
|
||||
Get API key: https://brave.com/search/api/ (free tier: 2000 queries/month)
|
||||
|
||||
### Firecrawl (for Web Scraping)
|
||||
|
||||
```yaml
|
||||
mcp_servers:
|
||||
firecrawl:
|
||||
command: npx
|
||||
args: ["-y", "firecrawl-mcp"]
|
||||
env:
|
||||
FIRECRAWL_API_KEY: "fc-xxxxx"
|
||||
```
|
||||
|
||||
## When to Use Which
|
||||
|
||||
| Scenario | Recommended Tool |
|
||||
|----------|------------------|
|
||||
| Quick factual lookup | `mmx search` or built-in `web_search` |
|
||||
| Structured data extraction | Tavily Search |
|
||||
| Google-specific results | SerpAPI |
|
||||
| Privacy-sensitive search | Brave Search |
|
||||
| Deep web scraping | Firecrawl |
|
||||
| Convert page to markdown | Jina Reader |
|
||||
| Previous conversation context | `session_search` |
|
||||
|
||||
## Pitfalls
|
||||
|
||||
- **mcp_servers: {}** is currently empty in this environment
|
||||
- Tool changes require `/reset` (new session) to take effect
|
||||
- Free tiers have rate limits; monitor usage for production use
|
||||
- Some MCP servers require Node.js (`npx` command)
|
||||
- **mmx search Token Plan 限制**:`mmx search query` 需要 MiniMax Token Plan 支持 `coding-plan-search` 模型。报错 "your current token plan not support model" 表示 plan 不含搜索能力,需升级或换替代方案
|
||||
- **服务器出站网络受限**:某些云服务器/容器的出站网络被限制,`urllib` 报 `Network is unreachable` (errno 101)。此时 curl、Python requests、mimo_search.py 均无法访问外部。排查:`curl -s -o /dev/null -w "%{http_code}" https://www.baidu.com`
|
||||
- **Playwright/Chrome 沙盒问题**:服务器上 Chrome 需要 `--no-sandbox` 参数才能启动。Hermes 的 browser 工具可能因沙盒限制失败。排查:检查 Chrome stderr 中是否有 "No usable sandbox" 错误
|
||||
- **mimo_search.py 环境变量**:脚本依赖 `XIAOMI_API_KEY` 环境变量,但 Hermes config.yaml 中的 key 不会自动 export。需手动 `export XIAOMI_API_KEY=...` 或从 config 提取
|
||||
|
||||
## 搜索降级策略(按优先级)
|
||||
|
||||
当用户请求搜索时,按以下顺序尝试:
|
||||
|
||||
1. **mmx search** — 最简单,但需 Token Plan 支持
|
||||
2. **mimo_search.py** — MiMo 原生搜索,需 XIAOMI_API_KEY 环境变量 + 网络可达
|
||||
3. **web_search 工具** — Hermes 内置,需 web toolset 启用
|
||||
4. **Playwright 浏览器** — 通用兜底,但需 Chrome 可启动
|
||||
5. **curl + 搜索引擎** — 最基础,需服务器出站网络可达
|
||||
|
||||
全部失败时,告知用户具体失败原因(Token Plan 限制 / 网络不通 / 浏览器沙盒问题),并建议用户在本地自行搜索。
|
||||
@@ -0,0 +1,64 @@
|
||||
# Skill Management Pitfalls
|
||||
|
||||
Learned from attempting to optimize the skill library based on SkillRouter paper findings.
|
||||
|
||||
## Pitfall 1: "Same Output" ≠ "Functionally Overlapping"
|
||||
|
||||
**Wrong:** Deleted `pptx-generator` (python-pptx) because `powerpoint` (pptxgenjs) also makes .pptx files.
|
||||
**Right:** Different tech stacks = different fallback options. python-pptx is pure Python, pptxgenjs needs Node.js. Keep both.
|
||||
|
||||
**Rule:** Two skills overlap only when they use the same tools AND serve the same user intent. Same output format is not enough.
|
||||
|
||||
## Pitfall 2: Don't Cross-Reference in Descriptions
|
||||
|
||||
**Wrong:** In arxiv's description: "需要多源学术搜索优先用 sn-search-academic"
|
||||
**Right:** Each skill describes itself only. No competitive recommendations.
|
||||
|
||||
**Why:** Creates circular dependencies. If skill A recommends B, and B recommends A, the LLM loops.
|
||||
|
||||
## Pitfall 3: Don't Expose Implementation Details
|
||||
|
||||
**Wrong:** In sn-infographic description: "需要 SN_API_KEY"
|
||||
**Right:** "需要 SenseNova API"
|
||||
|
||||
**Rule:** Descriptions should express user-facing capabilities, not internal tool/API names.
|
||||
|
||||
## Pitfall 4: Check Hard Dependencies Before Deleting
|
||||
|
||||
**Wrong:** Marked sn-research-planning for deletion because it was "never called."
|
||||
**Right:** sn-deep-research calls it via `skill_view("sn-research-planning")` at runtime. Deleting breaks the pipeline.
|
||||
|
||||
**How to check:**
|
||||
```python
|
||||
# Search all SKILL.md files for references to the target skill name
|
||||
# Only HARD dependencies count: skill_view("target-name") or "读取 target-name"
|
||||
# "Related skills" mentions are SOFT and don't block deletion
|
||||
```
|
||||
|
||||
## Pitfall 5: "Never skill_view'd" ≠ "Unused"
|
||||
|
||||
Skills can be auto-loaded via the system prompt's "MUST load" instruction without explicit `skill_view()` calls. Session data only shows explicit tool calls.
|
||||
|
||||
**Better metric:** Check if the skill is referenced as a runtime dependency by other skills.
|
||||
|
||||
## Pitfall 6: Don't Batch Recommendations Without Verification
|
||||
|
||||
**Wrong:** Generated all 87 recommendations at once, sent to user, then had to fix multiple errors.
|
||||
**Right:** Verify each category before sending. Check dependencies. Then send once.
|
||||
|
||||
**User feedback:** "你这建议就不能确认好之后再发给我吗" (Can you verify before sending?)
|
||||
|
||||
## Description Quality Formula
|
||||
|
||||
Good skill description = **What it does** + **Trigger words** + **Negative boundary**
|
||||
|
||||
Example:
|
||||
```
|
||||
"深度调研全流程编排器(入口 skill)。自动完成:规划→分维度取证→综合→成稿。
|
||||
触发词:深度研究/调研/全面研究/调研报告/deep research。
|
||||
不用于:单点事实问答、一句话摘要。"
|
||||
```
|
||||
|
||||
- What: 深度调研全流程编排器
|
||||
- Triggers: 深度研究/调研/全面研究
|
||||
- Boundary: 不用于单点事实问答
|
||||
@@ -0,0 +1,51 @@
|
||||
# 服务器环境网络搜索指南
|
||||
|
||||
## Playwright 浏览器启动(服务器/容器环境)
|
||||
|
||||
服务器环境(Ubuntu 23.10+、容器、VM)需要 `--no-sandbox` 参数:
|
||||
|
||||
```python
|
||||
from playwright.sync_api import sync_playwright
|
||||
|
||||
with sync_playwright() as p:
|
||||
browser = p.chromium.launch(
|
||||
headless=True,
|
||||
args=['--no-sandbox', '--disable-setuid-sandbox']
|
||||
)
|
||||
page = browser.new_page()
|
||||
page.goto('https://example.com', timeout=30000)
|
||||
# ... 操作页面
|
||||
browser.close()
|
||||
```
|
||||
|
||||
注意:Hermes 内置的 `browser_navigate` 工具不支持传递启动参数,必须直接用 Playwright API。
|
||||
|
||||
## 搜索引擎反爬情况(2026年测试)
|
||||
|
||||
| 搜索引擎 | 状态 | 备注 |
|
||||
|----------|------|------|
|
||||
| 百度 | ❌ 验证码 | 滑块验证,无法绕过 |
|
||||
| 搜狗 | ❌ 验证码 | 图片点选验证 |
|
||||
| 必应(cn.bing.com) | ⚠️ 可用但质量差 | 中文搜索结果常偏离关键词 |
|
||||
| Google | ❌ 超时 | 服务器网络不可达 |
|
||||
| DuckDuckGo | ❌ 超时 | 服务器网络不可达 |
|
||||
|
||||
**结论**:服务器环境下,主流搜索引擎基本不可用。必应是唯一能返回结果的,但质量不稳定。
|
||||
|
||||
## mmx search 限制
|
||||
|
||||
`mmx search` 需要 Token Plan 支持 `coding-plan-search` 模型。如果报错:
|
||||
```
|
||||
your current token plan not support model, coding-plan-search
|
||||
```
|
||||
说明当前计划不支持搜索功能,需要升级或使用其他方式。
|
||||
|
||||
## MiMo 模型搜索能力
|
||||
|
||||
MiMo (mimo-v2.5-pro) 可以声明 `web_search` 工具调用,但**实际上不会真正联网搜索**。它只能基于训练数据回答,无法获取实时信息。
|
||||
|
||||
## 替代方案优先级
|
||||
|
||||
1. Playwright + 必应(唯一可行的浏览器方案)
|
||||
2. 直接访问目标网站(如培训机构官网)
|
||||
3. 用户自行搜索后提供信息
|
||||
Reference in New Issue
Block a user