# 短信验证码调试与日志说明 本文档说明 thebet365 **玩家端短信验证码**(注册 / 找回密码)在后端的日志行为,以及如何与创蓝控制台对账、排查「收不到码」问题。 相关代码:`apps/api/src/domains/identity/sms/` 创蓝接入总览见 [chuanglan-sms-js-guide.md](./chuanglan-sms-js-guide.md)。 --- ## 一、整体流程 ```text 玩家端 POST /api/player/sms/send → SmsService 生成 6 位验证码 → 拼短信正文,调用创蓝 API(uid = sessionId) → 验证码写入 Redis:sms:code:{sessionId}(默认 5 分钟) 玩家注册 / 找回密码 → POST 携带 phone、sessionId、smsCode → SmsService.verifyCode 校验 Redis 后删除 ``` | 字段 | 说明 | |------|------| | `sessionId` | 后端生成的 UUID,作为创蓝请求的 `uid`,**对应创蓝控制台「批次号」** | | `messageId` | 创蓝返回的发送 ID,出现在成功日志中 | | 验证码 | 仅存于 Redis 与发信正文,**默认不写入 Docker 日志** | --- ## 二、环境变量 在 `.env.docker`(或本地 API 的 `.env`)中配置: | 变量 | 默认值 | 说明 | |------|--------|------| | `CHUANGLAN_ACCOUNT` | — | 创蓝账号(必填) | | `CHUANGLAN_PASSWORD` | — | 创蓝密码(必填) | | `CHUANGLAN_ENDPOINT` | `https://sgap.253.com/send/sms` | 国际短信网关 | | `SMS_CODE_TTL_SECONDS` | `300` | 验证码 Redis 有效期(秒) | | `SMS_RATE_LIMIT_SECONDS` | `60` | 同一手机号 / IP 发码冷却(秒) | | `SMS_DEBUG_LOG_CODE` | `false` | **联调专用**:为 `true` 时在日志输出验证码明文 | `docker-compose.prod.yml` 已将 `SMS_DEBUG_LOG_CODE` 传入 `thebet365-api` 容器。 --- ## 三、常规日志(生产推荐) `SMS_DEBUG_LOG_CODE=false` 时,API 会记录短信链路,但**不打印验证码**,手机号仅保留末 4 位(如 `****1234`)。 ### 3.1 查看方式 **SSH(推荐):** ```bash docker logs -f thebet365-api 2>&1 | grep -E 'SMS|Chuanglan|Player registered|Password reset' ``` **宝塔:** 容器 `thebet365-api` → **容器日志** → 搜索 `SMS` 或 `Chuanglan`。若显示「暂无日志」,优先用上面 `docker logs` 命令;面板偶发读不到 Docker 日志。 ### 3.2 日志含义 | 日志前缀 / 关键字 | 级别 | 含义 | |-------------------|------|------| | `SMS send request purpose=...` | log | 收到发码请求 | | `SMS rate limited` | warn | 触发频控(手机号或 IP) | | `SMS reset_password blocked reason=...` | warn | 找回密码前置校验失败 | | `Chuanglan request` | log | 已向创蓝发起 HTTP 请求 | | `Chuanglan success` | log | 创蓝受理成功(含 `messageId`) | | `Chuanglan rejected` | warn | 创蓝返回业务错误码 | | `Chuanglan HTTP error` | error | 网络超时或 HTTP 异常 | | `SMS sent` | log | 发码完成并已写入 Redis | | `SMS send failed` | error | 创蓝失败,前端收到 `SMS_SEND_FAILED` | | `SMS verify request / success / failed` | log / warn | 验码过程 | | `Player registered` | log | 注册成功(已通过验码) | | `Password reset by phone` | log | 短信找回密码成功 | `session=abc12345…` 为 `sessionId` 前 8 位,便于与创蓝批次号前缀对照。 --- ## 四、调试模式(输出验证码) 联调或排查「创蓝有记录、手机没收到」时,可临时开启: ```env SMS_DEBUG_LOG_CODE=true ``` 修改 `.env.docker` 后重启 API: ```bash cd /www/wwwroot/thebet365 # 按实际路径 docker compose -f docker-compose.prod.yml --env-file .env.docker up -d api ``` 若代码有更新,需先重新构建并加载 API 镜像再执行上述命令。 ### 4.1 调试日志格式 每次发码会多一行 **warn** 级别日志: ```text [SmsService] [SMS_DEBUG] purpose=register phone=0088613812345678 sessionId=ed08fe51-e9b7-4fec-9381-8249f4c65f4b code=123456 content=您的验证码是:123456。5分钟内有效。 ``` ### 4.2 与创蓝控制台对照 | 后端 `[SMS_DEBUG]` 字段 | 创蓝控制台列 | |-------------------------|--------------| | `sessionId` | **批次号** | | `phone` | **手机号** | | `code` / `content` | **发送内容** | | 常规日志 `messageId` | 发送详情中的 messageId | 筛选调试行: ```bash docker logs -f thebet365-api 2>&1 | grep SMS_DEBUG ``` ### 4.3 安全提醒 - **生产环境务必保持 `SMS_DEBUG_LOG_CODE=false`**,否则验证码会留在 Docker 日志中。 - 调试结束后改回 `false` 并重启 API。 - 不要将含 `[SMS_DEBUG]` 的日志片段对外分享。 --- ## 五、从 Redis 读取验证码(可选) 发码接口返回的 `sessionId` 有效期内,可直接查 Redis: ```bash docker exec thebet365-redis redis-cli GET "sms:code:完整的sessionId" ``` 返回示例: ```json {"phone":"0088613812345678","code":"123456","purpose":"register"} ``` 列出当前未过期的 key(key 带 TTL,验码或过期后会消失): ```bash docker exec thebet365-redis redis-cli KEYS "sms:code:*" ``` --- ## 六、常见问题 ### 6.1 创蓝显示「接收失败 / UNDELIV」 表示创蓝已提交,但**运营商未送达终端**。常见原因: - 号码格式或国家码不正确(国际号需与创蓝侧开通路由一致) - 目标地区/运营商拦截或关机、空号 - 短信内容或发送频率触发运营商过滤 后端若已有 `Chuanglan success` + `SMS sent`,说明 **API 与创蓝侧发送正常**,需从号码与运营商侧继续排查。 ### 6.2 日志里完全没有 `SMS` / `Chuanglan` 1. 确认请求是否打到 API(Nginx 是否将 `/api` 反代到 `thebet365-api:3000`)。 2. 确认运行的是**包含短信日志的新版 API 镜像**。 3. 用 `docker logs thebet365-api --tail 200` 查看,不要仅依赖宝塔面板。 ### 6.3 创蓝控制台有记录,后端没有 `[SMS_DEBUG]` - 检查 `.env.docker` 是否 `SMS_DEBUG_LOG_CODE=true`。 - 重启后是否加载了新环境变量:`docker exec thebet365-api printenv SMS_DEBUG_LOG_CODE`。 ### 6.4 发码太频繁 日志会出现 `SMS rate limited`。默认同一手机号 60 秒内只能发一次,可在 `.env.docker` 调整 `SMS_RATE_LIMIT_SECONDS`(生产不建议设过小)。 ### 6.5 找回密码发码失败 关注 `SMS reset_password blocked reason=`: | reason | 含义 | |--------|------| | `phone_not_registered` | 手机号未注册玩家 | | `account_disabled` | 账号已禁用 | | `password_change_disabled` | 系统配置禁止改密 | --- ## 七、相关 API | 方法 | 路径 | 说明 | |------|------|------| | POST | `/api/player/sms/send` | 发验证码,`purpose`: `register`(默认)或 `reset_password` | | POST | `/api/player/auth/register` | 注册,需 `sessionId` + `smsCode` | | POST | `/api/player/auth/forgot-password` | 找回密码,需 `sessionId` + `smsCode` | Swagger(API 容器启动后):`http://<主机>:3000/api/docs` --- ## 八、检查清单 联调短信时建议按序确认: - [ ] `.env.docker` 中 `CHUANGLAN_ACCOUNT` / `CHUANGLAN_PASSWORD` 正确 - [ ] 需要看验证码时 `SMS_DEBUG_LOG_CODE=true`,上线前改回 `false` - [ ] `docker logs` 能看到 `Chuanglan success` 与 `SMS sent` - [ ] 创蓝批次号与日志 `sessionId` 一致 - [ ] 若创蓝为 `DELIVRD` 仍收不到,检查手机拦截 / 地区路由 - [ ] 若创蓝为 `UNDELIV`,优先排查号码与运营商,而非后端代码