WeChat local data CLI with daemon architecture
 
 
 
 
Go to file
jakevin 6b7285c730
Merge pull request #1 from jackwener/feat/rust-rewrite
feat: Rust 完整重写 + 全部 bug 修复
2026-04-16 15:17:38 +08:00
.claude chore: 从 index 移除编译产物和 worktree 子仓库 2026-04-16 14:48:19 +08:00
.github/workflows fix: 修复全部 medium/low 优先级问题 2026-04-16 15:12:33 +08:00
docs fix: replace nonexistent scan_keys.js with actual tools in quickstart 2026-03-05 23:26:55 +08:00
src fix: 修复全部 medium/low 优先级问题 2026-04-16 15:12:33 +08:00
tests feat: wx init/export,修复全局搜索聊天名称 2026-04-16 01:38:40 +08:00
.gitignore chore: 从 index 移除编译产物和 worktree 子仓库 2026-04-16 14:48:19 +08:00
Cargo.lock feat: Rust 完整重写 wx-cli(单一二进制,支持 macOS/Linux/Windows) 2026-04-16 14:37:10 +08:00
Cargo.toml fix: 修复 review 发现的 4 个高优先级 bug 2026-04-16 14:48:03 +08:00
README.md docs: 恢复 ad-hoc 签名为必要前置步骤 2026-04-16 10:11:17 +08:00
config.example.json feat: add Linux support with cross-platform memory scanning 2026-03-07 21:35:24 +08:00
config.py fix: 统一所有 JSON 文件读写为 UTF-8 编码 2026-03-20 14:32:37 +08:00
find_all_keys_macos.c fix: use forward slashes in JSON output and add size==0 guard 2026-03-05 23:19:22 +08:00
key_utils.py refactor(find_all_keys): extract shared key scan logic 2026-03-07 21:35:24 +08:00
pyproject.toml feat: daemon + CLI 架构 (wx_daemon.py + wx.py) 2026-04-16 01:28:22 +08:00
uv.lock feat: daemon + CLI 架构 (wx_daemon.py + wx.py) 2026-04-16 01:28:22 +08:00
wx.py feat: wx init/export,修复全局搜索聊天名称 2026-04-16 01:38:40 +08:00
wx_daemon.py feat: wx init/export,修复全局搜索聊天名称 2026-04-16 01:38:40 +08:00

README.md

wx-cli

微信 4.x (macOS) 本地数据 CLI 工具。从运行中的微信进程内存提取加密密钥,后台常驻 daemon 持久缓存解密数据库CLI 毫秒级响应。

架构

wx (CLI) ──Unix socket──▶ wx-daemon (后台进程)
                              │
                    ┌─────────┼─────────┐
               DBCache     联系人缓存   WAL 监听
            (mtime 感知)              (500ms polling)
  • wx-daemon:后台常驻,持有解密后的 DB 热缓存首次解密后跨重启复用mtime 不变则不重解密)
  • wx (CLI):发 JSON 请求到 Unix socket获得响应后格式化输出首次调用自动启动 daemon

快速开始

环境要求

  • macOS (Apple Silicon / Intel)
  • WeChat 4.x (macOS 版,需 ad-hoc 签名,见下文)
  • Python 3.12+
  • uvPython 包管理)
  • Xcode Command Line Toolsxcode-select --install

安装

git clone git@github.com:jackwener/wx-cli.git
cd wx-cli
uv sync

初始化(首次使用)

微信需要 ad-hoc 签名才能被扫描内存:

sudo codesign --force --deep --sign - /Applications/WeChat.app

然后打开微信并登录,运行初始化:

uv run python wx.py init

wx init 自动完成:

  1. 检测微信数据目录(~/Library/Containers/.../xwechat_files/<wxid>/db_storage
  2. 编译 C 内存扫描器(如未编译)
  3. sudo 扫描微信进程内存,提取所有数据库密钥 → all_keys.json
  4. 更新 config.json

使用

# 最近会话
uv run python wx.py sessions

# 聊天记录
uv run python wx.py history "张三"
uv run python wx.py history "AI群" --since 2026-04-01 --until 2026-04-15

# 搜索消息
uv run python wx.py search "Claude"
uv run python wx.py search "会议" --in "工作群" --since 2026-01-01

# 联系人
uv run python wx.py contacts
uv run python wx.py contacts -q "李"

# 导出聊天记录
uv run python wx.py export "张三" --format markdown -o chat.md
uv run python wx.py export "AI群" --since 2026-01-01 --format json -o chat.json

# 实时监听新消息Ctrl+C 退出)
uv run python wx.py watch
uv run python wx.py watch --chat "AI交流群"
uv run python wx.py watch --json | jq .content

# daemon 管理
uv run python wx.py daemon status
uv run python wx.py daemon stop
uv run python wx.py daemon logs
uv run python wx.py daemon logs --follow

daemon 在首次 CLI 调用时自动启动,无需手动运行。

可选:设置别名

echo 'alias wx="uv run --directory /path/to/wx-cli python wx.py"' >> ~/.zshrc
source ~/.zshrc

# 之后可以直接用
wx sessions
wx history "张三"
wx watch

命令参考

wx init [--force]

首次初始化:检测数据目录、编译扫描器、提取密钥、写入配置。--force 强制重新扫描(微信更新后使用)。

wx sessions [-n N] [--json]

列出最近 N 个会话(默认 20显示未读数、最后消息摘要。

wx history CHAT [-n N] [--offset N] [--since DATE] [--until DATE] [--json]

查看指定聊天的消息记录。DATE 格式:YYYY-MM-DDYYYY-MM-DD HH:MM

wx search KEYWORD [--in CHAT]... [-n N] [--since DATE] [--until DATE] [--json]

全库搜索消息,--in 可指定多个聊天范围。

wx contacts [-q QUERY] [-n N] [--json]

列出或搜索联系人。

wx export CHAT [-f FORMAT] [-o FILE] [-n N] [--since DATE] [--until DATE]

导出聊天记录。-f 支持 markdown(默认)、txtjson-o 指定输出文件,不指定则输出到 stdout。

wx watch [--chat CHAT] [--json]

实时监听新消息WAL 变化推送,约 500ms 延迟)。--json 输出 JSON lines方便 jq 处理。

wx daemon status / stop / logs [-f] [-n N]

管理后台 daemon。logs --follow 等同 tail -f

原理

密钥提取

微信 4.x 使用 SQLCipher 4 加密本地数据库:

  • 加密AES-256-CBC + HMAC-SHA512
  • KDFPBKDF2-HMAC-SHA512256,000 次迭代
  • 页结构4096 bytes/pagereserve = 80IV 16 + HMAC 64

WCDB 在进程内存中缓存派生后的 raw key格式为 x'<64hex_enc_key><32hex_salt>'。C 扫描器(find_all_keys_macos.c)通过 macOS Mach VM API 扫描微信进程内存,匹配此模式,再用 HMAC 校验 page 1 确认密钥正确性,输出到 all_keys.json

DBCachemtime 感知缓存)

daemon 首次解密后将结果(及 DB/WAL 的 mtime持久化到 ~/.wechat-cli/cache/_mtimes.json。重启时若 mtime 未变,直接复用已解密文件,无需重新解密。

WAL 监听

微信使用 SQLite WAL 模式WAL 文件固定预分配 4MB不能靠文件大小判断变化。daemon 每 500ms 检测 session.db-wal 的 mtime有变化时重新解密并广播新消息给所有 watch 客户端。

数据文件路径

~/.wechat-cli/
├── daemon.sock       # Unix socket
├── daemon.pid        # PID 文件
├── daemon.log        # daemon 日志
└── cache/
    ├── _mtimes.json  # mtime 持久化索引
    └── *.db          # 解密后的数据库缓存

数据库结构

解密后约 26 个数据库:

路径 内容
session/session.db 会话列表(最新消息摘要、未读数)
message/message_*.db 聊天记录(按 Msg_<md5(username)> 分表)
contact/contact.db 联系人username、nick_name、remark
media_*/media_*.db 媒体文件索引

测试

uv run python -m pytest tests/ -v

免责声明

本工具仅用于学习和研究目的,用于解密自己的微信数据。请遵守相关法律法规,不要用于未经授权的数据访问。