事情是这样的。
最近在 GitHub 上刷到一个项目叫 code-review-graph,能把代码 review 的过程画成知识图谱。我对这种 AI 开发工具链没什么抵抗力,第一反应就是装上试试。
官方文档说安装就一行命令:
pipx install code-review-graph我敲下去,回车,等它跑完。
结果直接糊了一脸报错:
error: Failed to inspect Python interpreter from provided path at/opt/homebrew/opt/python@3.14/libexec/bin/pythonCaused by: Can't use Python at /opt/homebrew/opt/python@3.14/libexec/bin/pythonCaused by: Broken Python installation,platform.mac_ver() returned an empty value我愣了一下。
Broken Python installation?Python 装坏了?
前两天还好好的,我也没动过什么。第一反应是 Homebrew 又抽风了,但再看报错,指向的是一个很具体的函数 platform.mac_ver(),不是那种“你的 Python 炸了”的模糊提示。
有意思。决定追下去。
报错说 platform.mac_ver() 返回空值,那就直接测:
/opt/homebrew/opt/python@3.14/libexec/bin/python -c "import platformprint(platform.mac_ver())"输出:
('', ('', '', ''), '')确实是空的。
正常应该是这样:
('26.2', ('', '', ''), 'arm64')至少这个函数确实出问题了。但光一个函数异常,不代表整个 Python 都坏了。
先看系统版本:
sw_vers输出:
ProductName: macOSProductVersion: 26.2BuildVersion: 25C56是 macOS 26.2,Tahoe。
再确认:
sw_vers -productVersion26.2系统版本号正常,macOS 本身没问题。
同一个 Python 继续检查:
python3.14 -c "import platformimport osprint(platform.platform())print(platform.uname())print(os.uname())"输出:
Darwin-25.2.0-arm64-arm-64bit-Mach-Ouname_result(...)posix.uname_result(...)uname 正常,架构识别正常,Darwin 内核版本也正常。
也就是说,Python 并不是“坏了”,只是 platform.mac_ver() 这一个函数出了问题。
Python 3.14 是预发布版本,macOS 26 也是新系统。两个新东西凑一块,出兼容性问题太正常了。
于是我装了一个 Python 3.13:
brew install python@3.13然后测试:
/opt/homebrew/bin/python3.13 -c "import platformprint(platform.mac_ver())"结果:
('', ('', '', ''), '')一样。
排除了“Python 3.14 独有的 bug”。问题出在别的地方。
我机器上还有一套 Claude Code 自带的 Python,在 ~/.claude/.venv/ 下面。这套 Python 是我用 uv 装的,uv 下载的是官方 CPython 二进制,跟 Homebrew 自己编译的不是一回事。
抱着试试看的心态跑了一下:
/Users/xxx/.claude/.venv/bin/python3 -c "import platformprint(platform.mac_ver())"输出:
('26.2', ('', '', ''), 'arm64')完全正常。
这下范围小了。macOS 26.2 没问题,platform.mac_ver() 也没问题,出问题的是 Homebrew 构建出来的 Python。
同样的代码,同一个操作系统,uv 下载的官方 CPython 能正常返回版本号,Homebrew 自己编译的 Python 返回空值。
原因大概是这样:uv 下载的官方 CPython 是用较新 SDK 编译的,读取 SystemVersion.plist 时不会被 Apple 的版本号兼容机制误导。而 Homebrew 在本地编译 Python 时,用的 SDK 版本跟 macOS 26 之间可能存在不匹配,导致 mac_ver() 解析失败,直接抛空。
继续排查时跑了一个最朴素的命令:
which python3输出:
/usr/bin/python3居然是系统自带的 Python 3.9.6。
查了一圈,这台机器上至少有四套 Python:
/usr/bin/python3,系统自带,Python 3.9/opt/homebrew/bin/python3.13,Homebrew 装的~/.claude/.venv/bin/python3,Claude Code 虚拟环境,使用 uv 安装~/.local/bin/python3.13,uv 管理的独立 Python典型的 Python 环境地狱。
你以为自己在用 A Python,实际上工具链里的某个环节调的是 B Python。pipx 底层用 uv 创建虚拟环境,uv 在检测 Python 解释器时选了 Homebrew 那个有问题的 Python,然后 platform.mac_ver() 返回空值,uv 就直接判定“这个 Python 装坏了”。
链条是这样的:
pipx → uv venv → 选了 Homebrew Python → platform.mac_ver() 返回空 → uv 判定 Broken Python → 安装失败每一步看起来都合理,但组合在一起就是个诡异的报错。
追到根上,这是 Apple 的一个“兼容性设计”。
macOS 的版本号从 15.x 直接跳到了 26.x,Apple 改用了年份版本号。但很多老程序检查版本号时写的是“如果主版本号大于 15 就怎样怎样”,这种假设在新系统上会炸。
Apple 的解决方案很粗暴:对不同 SDK 编译的程序,系统返回不同的版本号。
用 SDK 26 编译的程序去读 SystemVersion.plist,返回 26.0。用旧版 SDK 编译的程序去读同一个文件,系统会在背后偷偷换成 16.0。
这不是 bug,是 Apple 的 feature。
CPython 社区在 GitHub 上有一个专门的 issue(#135675)讨论这件事。有人在 macOS 26 上跑 platform.mac_ver(),返回的是 ('16.0', ...) 而不是 ('26.0', ...)。
但我的情况更诡异,不是返回错误版本号,而是直接返回空值。
我怀疑是 Homebrew 在构建 Python 时,SDK 版本跟目标系统版本之间存在某种不匹配,导致 Python 读 SystemVersion.plist 时读到了一个无法解析的结果,然后直接抛空了。
同样的套路之前也玩过。macOS 从 10.15 跳到 11.0(Big Sur)时,旧 SDK 编译的程序读到的是 10.16,新 SDK 读到的才是 11.0。
五年过去了,同样的坑,换了个版本号,又踩一遍。
如果也遇到类似问题,可以用这个环境变量临时绕过:
SYSTEM_VERSION_COMPAT=0 python -c "import platform; print(platform.mac_ver())"设置 SYSTEM_VERSION_COMPAT=0 后,系统会强制返回真实版本号,不管 Python 是用什么 SDK 编译的。
找到根因后,方向就清晰了:换一套干净的、不受 Homebrew 影响的 Python。
这是我最终选的路。以后开发环境全部交给 uv,不再依赖 Homebrew 的 Python。
先卸载 Homebrew 的 Python:
brew uninstall python@3.14brew uninstall python@3.13然后用 uv 安装官方 CPython:
uv python install 3.13uv 会直接从 Python 官方下载预编译的二进制包,放到 ~/.local/bin/python3.13。这个版本的 Python 是用新版 SDK 编译的,不会有 platform.mac_ver() 的兼容问题。
装完确认路径:
which python3.13/Users/xxx/.local/bin/python3.13干净的,跟 Homebrew 没关系。然后正常安装 code-review-graph:
pip3.13 install code-review-graph验证:
code-review-graph --helpusage: code-review-graph ...搞定。
这条路走通之后,pipx 再装任何东西也不会踩这个坑了。因为 Homebrew 的 Python 已经卸了,uv 创建虚拟环境时会自动选 ~/.local/bin/ 下面这套官方 CPython。
不想引入 uv 的话,也可以直接从 python.org 下载 macOS 安装包。
Python 官方 installer 是用最新 Xcode 和 SDK 构建的,跟系统版本号兼容没问题。装完记得调整 PATH,确保 which python3 指向官方版本,而不是 Homebrew 那个。
这条路更传统,但以后升级版本、切换版本都得手动来。
我选的是方法一,一次性把 Python 的管理权交给 uv,以后少操心。
Broken Python installation 这个报错信息挺坑的。任何人看到这句话,第一反应都是“Python 装坏了,重装吧”。
但实际上 Python 没有坏,它只是一个函数返回了空值。而这个空值跟 Python 本身没关系,是 Apple 的版本兼容机制、Homebrew 的构建方式、uv 的检测逻辑三者叠加出来的结果。
更深的感受是另一件事。
在 Claude Code、Cursor、Aider、OpenHands、code-review-graph 这些 AI 开发工具越来越依赖 Python 的今天,Python 环境管理本身已经成了开发效率的一大绊脚石。
一台机器上同时存在系统 Python、Homebrew Python、uv Python、pyenv Python、各种虚拟环境 Python,这种情况太普遍了。然后某个工具链的某个环节选错了 Python,你就会看到一个完全无法理解的报错。
先搞清楚这几个命令的输出:
which pythonwhich python3which pipwhich pip3往往能省掉一大半排查时间。
还有就是,尽量不要让 Homebrew 管理你的 Python。Homebrew 是个很好的包管理器,但它的 Python 是本地编译的,编译时用的 SDK 版本跟 macOS 系统版本之间可能出现不匹配,在一些边缘场景下(比如 platform.mac_ver())就会出问题。
更好的做法是让 uv 统一管理 Python 版本。uv 下载的是 Python 官方预编译二进制,SDK 版本跟系统匹配度更高,安装速度也比 Homebrew 编译快得多。
如果已经在用 Homebrew 管理 Python,迁移到 uv 也很简单:
# 卸载 Homebrew 的 Pythonbrew uninstall python@3.13brew uninstall python@3.14# 用 uv 安装 Pythonuv python install 3.13# 确认路径which python3.13# 应该输出 /Users/xxx/.local/bin/python3.13以后 Python 环境都交给 uv(或者去官网下官方安装包),不会再有这种“Homebrew Python 跟 macOS 版本号打架”的诡异问题了。
最后,如果你在 macOS 26 上也遇到类似报错,建议按这个顺序排查:
先直接测 platform.mac_ver(),确认是不是返回空值。然后用 sw_vers 确认系统版本。接着 which python3 看当前用的是不是 Homebrew 的 Python,顺便检查机器上有没有多套 Python 并存。如果急用,SYSTEM_VERSION_COMPAT=0 可以临时绕过。最终方案还是换掉 Homebrew 的 Python,用 uv 管理或者去 python.org 下官方安装包。
这次踩坑大概就是这样。
从晚上九点多开始排查,到凌晨一点多才搞定。中间无数次想放弃,直接重装系统算了。但每次看到报错信息里的 platform.mac_ver() returned an empty value,又觉得这事肯定没那么简单。
最后搞明白了,还挺值的。