🌐 其他语言 / Other Languages: English · 日本語 · 한국어 · Français · Deutsch · Русский · Español
一款微信风格的端对端加密即时通讯应用,采用无状态 ECDH + XSalsa20-Poly1305 逐消息加密,支持 iOS PWA 永久免签与 Cloudflare R2 文件存储。
| 功能 | 说明 |
|---|---|
| 🔐 端对端加密 | 无状态 ECDH + XSalsa20-Poly1305,逐消息临时密钥,前向保密,Signal 风格安全号码验证 |
| 🗝️ 零知识服务器 | 服务器只存储密文,私钥仅在设备本地(四层持久化) |
| 📹 视频/语音通话 | WebRTC P2P(1:1)+ Mesh(多人),Cloudflare TURN 穿透 |
| 👥 群聊 | 最多 2000 人群组,纯文本消息(无加密),免打扰模式,成员管理 |
| 👫 好友系统 | 添加好友需对方审核,支持 512 字验证消息;备注名称;好友标签分组 |
| ⏱️ 消息自动删除 | 5 档可选(永不/1天/3天/1周/1月),私聊双方均可设置,群聊群主专属 |
| 🔔 消息推送 | Web Push (VAPID) + OneSignal 双通道,离线也能收到通知 |
| 🌐 多语言 | 中文、英文、日语、韩语、法语、德语、俄语、西班牙语(自动检测 + 手动切换) |
| 📱 iOS 永久免签 | PWA H5 → Safari「添加到主屏幕」,无需企业证书 |
| 💬 消息功能 | 文字、图片、视频、文档文件(PDF/DOCX/XLSX 等带类型图标)、语音消息、Emoji 面板(200+,8 分类)、Telegram 贴纸包、已读状态 |
| 📤 文件上传 | 单文件最大 500MB,支持 Cloudflare R2 或本地存储,带进度条动画 |
| 🌐 朋友圈 | 发动态(文字+最多9张图或1个视频≤10分钟)、点赞(显示好友头像)、评论、标签可见性控制 |
| 👤 个人资料 | 联系人资料页(头像/昵称/朋友圈动态),支持「不看此人朋友圈」与「不让他看我的朋友圈」双向隐私控制 |
| 📰 时间线 | 小红书风格公开发帖区——双列瀑布流布局,图片/视频+文字(最多50个媒体、2000字),支持匿名发帖、点赞、评论 |
| 🏷️ 好友标签 | 为好友设置多个标签(12色预设调色板),按标签分类筛选通讯录 |
| 🗂️ R2 对象存储 | Cloudflare R2 存储图片/语音,可选公开 CDN 直链 |
| 🔑 两步验证 (2FA) | Google Authenticator 兼容 TOTP 验证,8 个一次性恢复码,登录时强制验证 |
| 📷 扫码加好友/入群 | 扫一扫二维码添加好友、加入群聊,群二维码可设置有效期(1 周/1 月/3 月) |
| 🏗️ 可自托管 | Docker Compose 一键部署,Zeabur 一键云部署,前端可部署至 Vercel |
后端 (server/)
Rust (Axum 0.8) — 高性能异步 Web 框架
sqlx + MySQL 8.0 — 用户/消息持久化
deadpool-redis + Redis 7 — 在线状态 + 跨节点路由
aws-sdk-s3 — Cloudflare R2 文件存储(S3 兼容 API)
argon2 + jsonwebtoken 认证
前端 (client/)
React 19 + TypeScript + Vite 6
Zustand 状态管理
libsodium-wrappers-sumo (WebAssembly, Curve25519 / XSalsa20-Poly1305)
WebRTC API — 视频/语音通话
PWA: manifest.json + Service Worker
加密层
无状态 ECDH + XSalsa20-Poly1305 — 逐消息临时 ECDH 密钥对,前向保密
私钥四层持久化: 内存 → localStorage → sessionStorage → IndexedDB
私钥全程存储在设备本地,从不上传服务器
Note
部署完成后需手动完成一步配置,否则注册/登录无法使用:
- 进入 Zeabur 控制台 → server 服务 → Environment Variables → 复制
ZEABUR_WEB_URL的值(如http://10.43.x.x:3000) - 进入 client 服务 → Environment Variables → 添加变量
VITE_API_URL= 上一步复制的值 - Restart client 服务
Tip
进阶方案:Zeabur + Vercel 混合部署
可以在 Zeabur 部署完成后,手动删除 Zeabur 上的 client 服务,改用 Vercel 部署前端(参见下方“方式二”)。
这样 server/MySQL/Redis 由 Zeabur 托管,前端由 Vercel CDN 加速,全球访问更快。
Vercel 中设置环境变量 VITE_API_URL 指向 Zeabur 上 server 服务的公网域名即可。
已知注意事项:
- 首次启动 server 会自动创建数据库表(
CREATE TABLE IF NOT EXISTS),无需手动导入 schema - Redis 在集群内无需密码,已默认关闭认证
- 若需配置 MySQL root 密码,可在 server 服务的
DB_PASS里手动填写 MySQL 服务的MYSQL_ROOT_PASSWORD
# 克隆仓库
git clone <repo-url> && cd paperphone-plus
# 复制并编辑环境变量
cp server/.env.example server/.env
# 按需编辑:DB_PASS / JWT_SECRET / CF_CALLS_APP_ID 等
# 一键启动(含前端、后端、MySQL、Redis)
docker compose up -d
# 查看服务状态
docker compose ps
# 访问
open http://localhost注意:server 首次启动会自动初始化数据库 schema,无需手动导入 SQL 文件。
# 1. Fork 本仓库
# 2. 在 Vercel 中导入前端项目
# - Root Directory: client/
# - Build Command: npm run build
# - Output Directory: dist/
# - 环境变量: VITE_API_URL = https://your-server-domain.com
# 3. 后端使用 Docker 部署或 Zeabur 部署cd server
cp .env.example .env # 编辑环境变量
cargo run --release # → http://localhost:3000cd client
npm install
npm run dev # → http://localhost:5173视频通话使用 WebRTC P2P,局域网内开箱即用。跨网络通话需要配置 TURN 服务器(用于 NAT 穿透)。
- 登录 Cloudflare Dashboard → Workers & Pages → Calls → 创建 App
- 获取 App ID 和 App Secret(Token Key)
- 填入
server/.env:
CF_CALLS_APP_ID=your_app_id_here
CF_CALLS_APP_SECRET=your_app_secret_here- 重启后端,TURN 凭据会在每次通话时自动刷新(TTL 86400s)
未配置时:自动降级为 STUN only(Google + Cloudflare 公共 STUN),局域网内可正常通话。
| 类型 | 技术方案 | 适用场景 |
|---|---|---|
| 私聊 1:1 视频 | WebRTC P2P + TURN | 所有场景 |
| 私聊 1:1 语音 | WebRTC P2P + TURN | 所有场景 |
| 群组多人语音/视频 | WebRTC Mesh(全连接) | ≤ 6 人 |
离线消息通知通过双通道推送,最大化消息送达率:
| 通道 | 适用场景 | 配置 |
|---|---|---|
| Web Push (VAPID) | 浏览器 (Chrome/Edge/Firefox) + iOS PWA (Safari 16.4+) | VAPID 密钥 |
| OneSignal | Median.co 打包的原生 Android/iOS App | OneSignal App ID + REST Key |
- 生成 VAPID 密钥(仅需一次):
npx web-push generate-vapid-keys- 填入
server/.env:
VAPID_PUBLIC_KEY=your_public_key_here
VAPID_PRIVATE_KEY=your_private_key_here
VAPID_SUBJECT=mailto:admin@your-domain.com- 重启服务器,用户可在设置页开启通知
iOS 用户需先将应用「添加到主屏幕」,且仅 iOS 16.4+ 支持。
- 在 OneSignal Dashboard 创建 App 并配置 Firebase
- 在 Median.co 中启用 OneSignal 并填入 App ID
- 将 OneSignal 的 App ID 和 REST API Key 填入
server/.env:
ONESIGNAL_APP_ID=your_onesignal_app_id
ONESIGNAL_REST_KEY=your_onesignal_rest_api_key未配置时:推送功能静默禁用,不影响其他功能。
- 部署到有 HTTPS 域名的服务器(WebRTC 和加密 API 需要 HTTPS)
- 用 Safari 打开
https://your.domain.com - 点击底部分享按钮 ⬆️
- 选择「添加到主屏幕」→「添加」
即可获得与原生 App 相同的体验,无需 Apple 企业证书,永久有效!
server {
listen 443 ssl http2;
server_name your.domain.com;
ssl_certificate /path/to/cert.pem;
ssl_certificate_key /path/to/key.pem;
# 前端静态文件(或使用 Vercel 单独部署)
location / {
root /path/to/paperphone-plus/client/dist;
try_files $uri /index.html;
}
# API
location /api/ {
proxy_pass http://localhost:3000;
proxy_set_header Host $host;
}
# WebSocket 信令
location /ws {
proxy_pass http://localhost:3000;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
proxy_read_timeout 3600s;
}
}paperphone-plus/
├── docker-compose.yml
├── zeabur.yaml
├── server/ # Rust (Axum) 后端
│ ├── Cargo.toml
│ ├── Dockerfile
│ ├── .env.example
│ └── src/
│ ├── main.rs # Axum 应用入口
│ ├── config.rs # 环境变量管理
│ ├── db/
│ │ ├── mysql.rs # MySQL 连接池 (sqlx)
│ │ ├── redis.rs # Redis 连接池 (deadpool-redis)
│ │ └── schema.sql # 数据库 schema(幂等)
│ ├── auth/
│ │ ├── jwt.rs # JWT 签名/验证(含 2FA pending token)
│ │ └── middleware.rs # Axum 鉴权中间件
│ ├── routes/
│ │ ├── auth.rs # 注册/登录(含 X3DH 公钥上传)
│ │ ├── users.rs # 用户搜索 / Prekey 下载
│ │ ├── friends.rs # 好友申请 / 接受
│ │ ├── groups.rs # 群组管理
│ │ ├── messages.rs # 消息历史分页
│ │ ├── upload.rs # Cloudflare R2 文件上传
│ │ ├── files.rs # 文件代理(R2_PUBLIC_URL 未设时)
│ │ ├── moments.rs # 朋友圈(动态/点赞/评论/隐私控制)
│ │ ├── timeline.rs # 时间线(公开发帖/点赞/评论/匿名)
│ │ ├── calls.rs # TURN 凭据派发
│ │ ├── push.rs # 推送订阅管理
│ │ ├── stickers.rs # Telegram 贴纸包代理(缓存)
│ │ ├── totp.rs # TOTP 两步验证
│ │ ├── sessions.rs # 会话管理(多设备登录)
│ │ └── tags.rs # 好友标签 CRUD
│ ├── services/
│ │ ├── push.rs # Web Push VAPID 服务
│ │ └── onesignal.rs # OneSignal REST API 服务
│ └── ws/
│ └── server.rs # WebSocket 路由(消息/通话信令/已读/推送)
│
└── client/ # React + TypeScript + Vite 前端
├── package.json
├── Dockerfile
├── vercel.json # Vercel 部署配置
├── vite.config.ts
├── tsconfig.json
├── index.html
└── src/
├── main.tsx # React 入口
├── App.tsx # 路由 + 鉴权守卫
├── index.css # Premium 设计系统(暗色/亮色,玻璃拟态)
├── vite-env.d.ts
├── store/
│ └── index.ts # Zustand 全局状态
├── api/
│ ├── http.ts # HTTP 客户端(JWT 拦截)
│ └── socket.ts # WebSocket 客户端(自动重连 + 心跳)
├── i18n/
│ ├── index.ts # 多语言引擎
│ └── locales/ # zh/en/ja/ko/fr/de/ru/es
├── crypto/
│ ├── ratchet.ts # ECDH + XSalsa20-Poly1305 加密
│ └── keystore.ts # 四层私钥持久化
├── hooks/
│ ├── useAuth.ts
│ ├── useI18n.ts
│ └── useSocket.ts
├── components/
│ └── TabBar.tsx # 底部导航栏
└── pages/
├── Login.tsx # 登录/注册(含密钥生成、2FA、语言切换)
├── Chats.tsx # 会话列表
├── Chat.tsx # 聊天窗口(消息气泡、输入框)
├── Contacts.tsx # 通讯录(好友/群组/好友请求)
├── Discover.tsx # 发现页
├── Profile.tsx # 设置(深色模式、语言、退出)
├── UserProfile.tsx # 联系人资料页
├── GroupInfo.tsx # 群信息
├── Moments.tsx # 朋友圈(图片网格、点赞、评论)
└── Timeline.tsx # 时间线(瀑布流布局)
共 19+ 张表,首次启动自动创建(CREATE TABLE IF NOT EXISTS):
| 表名 | 说明 |
|---|---|
users |
用户信息 + ECDH/OPK 公钥 |
prekeys |
X3DH 一次性预密钥池 |
friends |
好友关系(pending/accepted/blocked) |
groups / group_members |
群组 + 成员(含免打扰状态) |
messages |
加密消息(离线缓冲,送达后可删) |
moments |
朋友圈动态(文字 ≤1024 字) |
moment_images |
动态图片(每条最多 9 张) |
moment_videos |
动态视频(封面图+时长,每条最多 1 个,≤10 分钟) |
moment_likes |
点赞(每用户每条唯一) |
moment_comments |
评论(最多 512 字/条) |
moment_visibility |
动态可见性规则 |
moment_privacy |
朋友圈用户级隐私设置(不看/不让看) |
push_subscriptions |
Web Push 推送订阅(VAPID) |
onesignal_players |
OneSignal 设备注册(Median.co) |
user_totp |
TOTP 两步验证密钥与恢复码 |
sessions |
多设备会话管理 |
friend_tags / friend_tag_assignments |
好友标签系统 |
timeline_posts |
时间线帖子(文字 ≤2000 字,支持匿名) |
timeline_media |
时间线媒体(图片/视频,每帖最多 50 个) |
timeline_likes |
时间线点赞 |
timeline_comments |
时间线评论(支持匿名) |
group_invites |
群邀请链接(含有效期,用于二维码加群) |
注册时:
设备本地生成 IK(身份密钥)+ SPK(签名预密钥)+ 20x OPK(一次性预密钥)
公钥上传服务器,私钥四层持久化,永不离开设备
发送消息时:
发送方下载接收方 IK 公钥
生成临时 ECDH 密钥对(每条消息独立)
X25519 ECDH → 共享秘密 → XSalsa20-Poly1305 加密
临时公钥附在消息 header 中,接收方解密后销毁
服务器所见:
✅ 密文 blob + 路由元数据(发件人/收件人 UUID)
❌ 明文 / 私钥 / 临时密钥 / 通话内容
| 变量 | 说明 | 默认值 |
|---|---|---|
PORT |
服务端口 | 3000 |
JWT_SECRET |
JWT 签名密钥(生产必改) | dev_secret |
DB_HOST / DB_PASS / DB_NAME |
MySQL 连接配置 | — |
REDIS_HOST / REDIS_PASS |
Redis 连接配置 | — |
R2_ACCOUNT_ID |
Cloudflare 账号 ID | — |
R2_ACCESS_KEY_ID |
R2 API Token 的 Access Key | — |
R2_SECRET_ACCESS_KEY |
R2 API Token 的 Secret Key | — |
R2_BUCKET |
R2 Bucket 名称 | — |
R2_PUBLIC_URL |
R2 公开 URL(可选),设置后文件走 CDN 直链 | — |
CF_CALLS_APP_ID |
Cloudflare Calls App ID(可选) | — |
CF_CALLS_APP_SECRET |
Cloudflare Calls App Secret(可选) | — |
VAPID_PUBLIC_KEY |
Web Push VAPID 公钥(可选) | — |
VAPID_PRIVATE_KEY |
Web Push VAPID 私钥(可选) | — |
VAPID_SUBJECT |
VAPID 联系邮箱(可选) | mailto:admin@paperphone.app |
ONESIGNAL_APP_ID |
OneSignal App ID(可选,Median.co) | — |
ONESIGNAL_REST_KEY |
OneSignal REST API Key(可选) | — |
TELEGRAM_BOT_TOKEN |
Telegram Bot Token(可选,贴纸包代理) | — |
STICKER_PACKS |
自定义贴纸包列表(可选,逗号分隔 包名:显示名) |
内置 8 个默认包 |

















