tg-claude
Health Pass
- License — License: MIT
- Description — Repository has a description
- Active repo — Last push 0 days ago
- Community trust — 13 GitHub stars
Code Fail
- rm -rf — Recursive force deletion command in approval-hook.mjs
- process.env — Environment variable access in approval-hook.mjs
- network request — Outbound network request in approval-hook.mjs
- process.env — Environment variable access in notify-hook.mjs
- network request — Outbound network request in notify-hook.mjs
- process.env — Environment variable access in stop-hook.mjs
- network request — Outbound network request in stop-hook.mjs
- execSync — Synchronous shell command execution in worker/executor.mjs
- process.env — Environment variable access in worker/executor.mjs
- execSync — Synchronous shell command execution in worker/index.mjs
- network request — Outbound network request in worker/index.mjs
- network request — Outbound network request in worker/mcp-telegram.mjs
Permissions Pass
- Permissions — No dangerous permissions requested
This tool acts as a remote control interface, allowing users to operate a local Claude Code terminal session directly through a Telegram bot. It enables full command execution, file handling, and voice input from a mobile device.
Security Assessment
Overall risk: High. The application is fundamentally designed to expose a local shell to an external chat service, which carries inherent risks. Static analysis reveals multiple critical concerns. The codebase relies heavily on synchronous shell command execution (`execSync`) and contains recursive force deletion (`rm -rf`) commands. It actively reads environment variables, potentially exposing local system secrets. While no hardcoded secrets were detected, the tool routinely makes outbound network requests to facilitate the Telegram bot communication and external voice transcription APIs (Groq Whisper). Users must rigorously secure their Telegram bot credentials, as any compromise would grant an attacker direct access to the host machine's command line.
Quality Assessment
The project is relatively new and has a very small footprint, reflected by its 13 GitHub stars, meaning it lacks widespread community trust or extensive peer review. However, basic maintenance appears excellent, with the most recent code push occurring today. It is properly licensed under the standard MIT agreement and includes a highly detailed, comprehensive README.
Verdict
Use with extreme caution; while well-documented and active, the tool's core functionality involves exposing a powerful local shell to an external messaging platform.
Turn your Mac/Linux into a remote Claude Code terminal via Telegram. Spawns real CLI — sessions, streaming, approvals, voice, git panel, 30+ commands.
TG Claude
Turn your Mac or Linux machine into a remote Claude Code terminal — controlled from your phone via Telegram.
Send a message from Telegram → bot spawns real Claude Code CLI → streams progress → sends response back. Not an API wrapper — runs the actual CLI with full tool access (Bash, Read, Write, Edit, Grep, Glob, etc).
You (Telegram) → Bot (Node.js) → Claude Code CLI → tools → response → You
| Voice input + streaming | Approval buttons | Full approval flow |
|---|---|---|
![]() |
![]() |
![]() |
| /status | /sessions |
|---|---|
![]() |
![]() |
Why This Exists
Anthropic's official Telegram plugin is a minimal MCP bridge — 3 tools, no history, no session management.
TG Claude is a full remote terminal: sessions, streaming progress, approval system for dangerous ops, git panel, voice input, file/photo handling, Mac remote control, and 30+ commands — all from your phone.
| Official Plugin | TG Claude | |
|---|---|---|
| Architecture | MCP bridge | Full bot wrapper around CLI |
| Session management | None | Auto-resume, rotate, switch, rename |
| Streaming progress | None | Live tool activity display |
| Dangerous op approval | None | Inline buttons (approve/deny) |
| Voice input | None | Groq Whisper STT |
| Git integration | None | Full panel (status/diff/log/push/pull) |
| Quick commands | None | Shell, system info, clipboard, files |
| i18n | None | EN + RU (auto-detected) |
Quick Start
git clone https://github.com/Imolatte/tg-claude.git
cd tg-claude && cp config.example.json config.json
# Edit config.json — add bot token, chat ID, Groq API key
cd worker && npm install && node index.mjs
On first /start, a setup wizard walks you through OS, output mode, and token rotation settings.
Features
Input Types
- Text → Claude Code with full tool access
- Voice → Groq Whisper STT → Claude
- Photos → downloaded, passed to Claude for vision analysis
- Files → downloaded with original extension, passed to Claude
- Forwards → analyzed by Claude (text before forward auto-combined)
Live Streaming Progress
While Claude works, see exactly what's happening:
🔧 Read: /src/auth.ts
🔧 Edit: /src/auth.ts
🔧 Bash: npm run build
📝 Writing response...
With /codediff enabled, edits show inline diffs:
🔧 Edit: /src/auth.ts
- import { getRegion } from './geo'
+ import { getRegion, USDT_PRICE } from './geo'
Token usage shown after each response: ↓3.2k ↑17k · 4.5s · ████░░░░ 52k/100k
Session Management
- Auto-resume sessions across messages (
--resume) /sessions— inline keyboard to switch/delete/new [name]— start fresh;/name <title>— rename/detach— disconnect;/cd <path>— change working directory- Token usage tracking with warning at 190k (auto-compaction by Claude Code CLI)
Terminal ↔ Telegram Session Sync (experimental — testing)
Status: feature is being tested. The problem: terminal and Telegram use separate Claude Code sessions, so progress/context is lost when switching between them.
One shared session between terminal and Telegram. Both read/write to worker/state.json. The session itself is a single .jsonl file on disk — not a copy, the same conversation.
How it works
Telegram → Terminal:
- You work in terminal → session
.jsonlfile gets updated - You switch to Telegram and send a message
- Bot detects a newer session exists → asks "Switch to session-name?" with ✅/❌ buttons
- ✅ — bot resumes the terminal session. ❌ — stays in current, pins it (no more prompts)
Terminal → Telegram:
- You work in Telegram → bot saves session to
state.json - You switch to terminal and run
cc ccreadsstate.json, finds the freshest session,cds into its project directory, and runsclaude --resume <id>
New session in Telegram:
/newin Telegram creates a session in the same project directory as the previous one (or whatever you set via/cd)- After first message,
ccin terminal will pick it up
The cc command
Use cc instead of claude to continue the shared session. Works from any directory — auto-navigates to the project.
cc # resume the most recent shared session
cc "fix the bug" # resume with a prompt
If no shared session found — falls back to claude -c (standard "continue last").
Pinned sessions
When you manually select a session via /sessions in Telegram, it gets pinned — the bot won't suggest switching away from it. The pin is cleared after the next Claude run, so the auto-detect resumes naturally.
Declining a switch (❌) also pins the current session.
Setup
- Copy
ccto your PATH:cp scripts/cc ~/.local/bin/cc && chmod +x ~/.local/bin/cc - Add the Stop hook to
~/.claude/settings.json(syncs terminal sessions tostate.json):{ "hooks": { "Stop": [{ "hooks": [{ "type": "command", "command": "node /path/to/tg-claude/stop-hook.mjs", "timeout": 10 }] }] } }
Known limitations
ccpicks the most recent session across all projects — if you have multiple active projects, it may pick the wrong one- Terminal sessions only sync to
state.jsonwhen Claude finishes a turn (Stop hook), not mid-conversation - If a session is deleted in Telegram, terminal gets "session not found" — just run
claudeto start fresh
Approval System
Dangerous operations require your approval via Telegram inline buttons:
git push/reset/clean,rm -rf, Docker commands- DB migrations (
prisma migrate,DROP TABLE) - Deploys (
vercel --prod,npm publish) - Sensitive file edits (
.env,docker-compose, CI configs)
Output Modes
| Mode | Responses | Approvals | Use case |
|---|---|---|---|
terminal |
Terminal | Terminal | At desk |
hybrid |
Terminal | Telegram | Away, approvals on phone |
telegram |
Telegram | Telegram | Fully remote |
Switch with /mode in Telegram or node mode.mjs <mode> in terminal.
Commands
Main
| Command | Description |
|---|---|
/help |
Full command reference |
/status |
Mode, model, session, cwd, token progress bar |
/setup |
Re-run first-time setup wizard |
/stop |
Kill running Claude process |
/plan |
Toggle Plan / Build mode |
Sessions
| Command | Description |
|---|---|
/sessions |
Session list with connect/delete buttons |
/new [name] |
Start a new session |
/name <title> |
Rename current session |
/detach |
Disconnect from session |
/cd <path> |
Set working directory (- to reset) |
Display & Output
| Command | Description |
|---|---|
/codediff |
Toggle inline code diff in tool updates |
/mode |
Output mode: terminal / hybrid / telegram |
/model |
Switch model: sonnet / opus / haiku |
/botlang |
Bot UI language: en / ru |
/lang |
Voice recognition language: ru / en / auto |
Git
| Command | Description |
|---|---|
/git |
Git panel — Status, Diff, Log, Stage, Commit (AI message), Push, Pull |
/git <args> |
Direct git command, e.g. /git log --oneline -5 |
/undo |
Rollback last commit (soft/hard, with confirmation) |
/diff [ref] |
Show git diff with pagination |
Quick Commands (instant, no Claude)
| Command | Description |
|---|---|
/sh <cmd> |
Run shell command |
/sys |
CPU, RAM, disk, battery, Wi-Fi, IP, uptime |
/clip |
Get/set clipboard |
/dl <path> |
Download file to Telegram |
/recent |
Recently edited files with one-tap download |
/screenshot <url> |
Screenshot via Puppeteer |
/cron <Xh/Xm> <text> |
Set a reminder |
Mac Remote Control
| Command | Description |
|---|---|
/sleep |
Put Mac to sleep |
/lock |
Lock screen |
/shutdown |
Shut down (with confirmation) |
/reboot |
Restart (with confirmation) |
/battery |
Battery status with low-battery alerts |
MCP: Claude → Telegram
Claude can proactively message you:
send_telegram(text)— send text messagesend_file_telegram(file_path, caption?)— send file
Group Chat Support
Add the bot to any Telegram group:
- Only the owner + users on the allowed list can interact
- Bot responds when @mentioned or when replying to its messages
- Dangerous operations from non-owners are automatically denied (owner gets a DM notification)
Setup
Prerequisites
- Node.js 20+
- Claude Code CLI installed and in PATH
- Telegram Bot — create via @BotFather
- Groq API key (free): groq.com → API Keys
- Optional: FFmpeg (
brew install ffmpeg) for voice fallback
Install
git clone https://github.com/Imolatte/tg-claude.git
cd tg-claude/worker && npm install
cp ../config.example.json ../config.json
Configure
Edit config.json:
{
"botToken": "YOUR_BOT_TOKEN",
"chatId": "YOUR_TELEGRAM_CHAT_ID",
"groqApiKey": "YOUR_GROQ_API_KEY",
"timeoutMs": 300000,
"claudeTimeoutMs": 1800000,
}
| Field | Description | Default |
|---|---|---|
botToken |
Telegram bot token from @BotFather | required |
chatId |
Your Telegram user ID (owner) | required |
groqApiKey |
Groq API key for voice STT | required |
timeoutMs |
Approval request timeout (ms) | 300000 (5 min) |
claudeTimeoutMs |
Max time for a single Claude task (ms) | 1800000 (30 min) |
Get your chatId: send any message to your bot, then open https://api.telegram.org/bot<TOKEN>/getUpdates — find chat.id.
Claude Code Hooks
Add to ~/.claude/settings.json for approval forwarding and completion notifications:
{
"hooks": {
"PreToolUse": [{
"matcher": "",
"hooks": [{ "type": "command", "command": "node /full/path/to/tg-claude/approval-hook.mjs", "timeout": 310 }]
}],
"Stop": [{
"hooks": [{ "type": "command", "command": "node /full/path/to/tg-claude/stop-hook.mjs", "timeout": 10 }]
}]
}
}
Start
cd worker && node index.mjs
Autostart (macOS)
Create ~/Library/LaunchAgents/com.tg-claude.worker.plist pointing to node index.mjs in the worker directory. See launcher.sh for reference.
Logs: /tmp/tg-claude.log
Architecture
worker/
index.mjs Bot core: polling, commands, streaming, task queue
executor.mjs Spawns Claude Code CLI, parses stream-json events
sessions.mjs Session state management
locale.mjs i18n: EN + RU strings
voice.mjs Groq STT with local Whisper fallback
mcp-telegram.mjs MCP server: send_telegram + send_file_telegram
approval-hook.mjs PreToolUse hook → Telegram approval buttons
stop-hook.mjs Stop hook → Telegram completion notification
mode.mjs CLI: switch output mode (terminal/hybrid/telegram)
bot-system-prompt.md System prompt appended to Claude Code
config.json Credentials (gitignored)
Design principles:
- Zero runtime dependencies (except Puppeteer for optional screenshots)
- No frameworks — vanilla Node.js, ESM modules
- Single-process architecture — one
node index.mjsruns everything - File-based IPC between hooks and bot (via
/tmp/files) - Works on macOS and Linux
i18n
Bot UI supports English and Russian. Language is auto-detected from your Telegram profile on first /start. Switch anytime with /botlang.
Adding a new language: copy the en object in worker/locale.mjs, translate all values, submit a PR.
Security
- Owner-only — single
chatId, all other users silently ignored - Group access control — explicit allowlist per user ID
- No credentials in repo —
config.jsonis gitignored - Duplicate protection — kills previous instances on startup
- Stale session recovery — auto-retries without
--resumeon failure
Contributing
See CONTRIBUTING.md for guidelines.
License
MIT — Andrey Petrushikhin
Reviews (0)
Sign in to leave a review.
Leave a reviewNo results found




