现象速描:三句话看懂“音色去哪儿了”
- 终端执行
python -m cosyvoice.cli.tts --list-voices回显[],列表空空如也。 - 日志里反复出现
WARNING: Pre-trained voices not found under ./pretrained_voices,却明明记得安装包自带了 12 个官方音色。 - WebUI 下拉框只剩一个“--None--”,播放按钮灰掉,系统毫无报错,只剩一脸懵。
技术拆解:音色加载链路与常见坑位
1. 默认加载逻辑
CosyVoice 在 Linux 下的音色搜索顺序写死在cosyvoice.utils.voice_loader:
VOICE_ROOT = os.getenv("COSYVOICE_VOICE_ROOT") or \ Path(__file__).resolve().parent.parent / "pretrained_voices"若环境变量未设置,则退到安装目录的pretrained_voices子目录;该目录预期结构:
pretrained_voices/ ├── g1/ │ ├── config.yaml │ └── generator.ckpt ├── g2/ ...2. 缺失三大元凶
- 权限不足:pip 默认装在
/usr/local/lib/python3.x/site-packages,普通用户无写权限,导致解压后音色目录 inode 为root:root 750,进程无法遍历。 - 路径漂移:conda 虚拟环境或
venv把包挪到$CONDA_HOME/envs/xxx/...,而你在全局/usr/share/cosyvoice硬链接了一份旧数据,加载器优先读到空目录。 - 依赖缺失:音色压缩包采用
zstd,系统未安装libzstd.so,解压阶段静默失败,留下空壳文件夹。
3. strace 快速定位
用strace追踪文件访问,比读日志更直接:
strace -e trace=file -f -o trace.log \ python -m cosyvoice.cli.tts --list-voices搜索pretrained_voices关键字,若出现EACCES (Permission denied)或ENOENT (No such file or directory),即可锁定是权限还是路径问题。
分步解决方案
Step 1 确认目录结构
# 找到包真实路径 python -c "import cosyvoice,os,sys; print(os.path.dirname(cosyvoice.__file__))" # 假设输出 /opt/miniconda/envs/cosy/lib/python3.10/site-packages/cosyvoice ls -lh /opt/miniconda/envs/cosy/lib/python3.10/site-packages/cosyvoice/pretrained_voices预期输出应包含g1、g2...若干子目录;若为空,继续 Step 2。
Step 2 修复权限(755 vs 777)
# 仅授予目录遍历与文件读取,最小权限原则 sudo chmod -R 755 /opt/miniconda/envs/cosy/lib/python3.10/site-packages/cosyvoice/pretrained_voices区别:
755= 所有者 rwx,组与其他 rx,可进目录但不可写,满足推理阶段只读需求;777= 任意用户可写,生产环境易遭篡改,除非容器只读文件系统,否则不推荐。
Step 3 配置环境变量
临时方案(当前 shell 生效):
export COSYVOICE_VOICE_ROOT=$HOME/mycosyvoices python -m cosyvoice.cli.tts --list-voices永久方案(推荐写入~/.bashrc或/etc/profile.d/cosyvoice.sh):
echo 'export COSYVOICE_VOICE_ROOT=$HOME/mycosyvoices' >> ~/.bashrc source ~/.bashrc注意:若使用 systemd 启动服务,需在 unit 文件中通过Environment=显式传入,.bashrc对 systemd 无效。
Step 4 Python 端自检脚本
# check_voices.py import os, sys from pathlib import Path try: from cosyvoice.utils.voice_loader import list_voices voices = list_voices() if not voices: print("WARN: 音色列表为空,请检查 COSYVOICE_VOICE_ROOT 或目录权限") sys.exit(1) print("OK: 已加载音色", voices) except Exception as e: print("ERROR:", e) sys.exit(2)运行:
python check_voices.py && echo "加载成功" || echo "加载失败"生产环境建议
1. 容器化部署要点
Dockerfile 片段:
COPY pretrained_voices /app/pretrained_voices ENV COSYVOICE_VOICE_ROOT=/app/pretrained_voices RUN chmod -R 755 /app/pretrained_voicesdocker-compose.yml 示例:
volumes: - ./pretrained_voices:/app/pretrained_voices:ro # 只读挂载,防止运行时篡改2. 校验文件完整性
官方提供voices.md5:
cd $COSYVOICE_VOICE_ROOT md5sum -c voices.md5 # 输出 OK 即完整;若失败,重新下载并解压小结
- 音色缺失 90% 是权限或路径问题,strace 能在 30 秒内告诉你“缺在哪”。
- 目录结构、环境变量、只读挂载三件套配齐后,基本可一次性解决。
- 生产环境务必加 md5 校验与只读卷,防止运行时模型被意外覆盖。
开放讨论
当音色库膨胀到上百个、总大小超 10 GB 时,一次性全量加载显然会吃光内存。你打算如何实现按需加载(lazy load)——是改写voice_loader在首次调用时再torch.load,还是把音色拆成独立微服务远程调用?欢迎留言交换思路。