乐趣区

关于homebrew:使用brew安装历史版本的几种方式

背景

在 mac osx 下, 大部分的软件都是应用 homebrew 进行治理的, 能够不便的进行软件的装置, 更新, 删除等等, 大部分状况下 homebrew 的仓库只会存在一份最新的软件版本, 有时可能也会同时存在多个版本, 比方 python 就会有多个.

在有些状况下咱们可能须要装置某些软件的历史版本, 接下来提供几种形式实现.

Homebrew 术语阐明

在装置历史版本介绍之前, 先简略介绍一下 Homebrew 的一些名词术语及构造, 以便大家能更好的了解, Homebrew 中文能够翻译成 “ 家酿 ”, 所以这个软件是对酒相干的一个形象. 次要的构造包含:

Formula (配方)

: 次要应用 ruby 文件形容的软件信息, 蕴含软件根本信息, 依赖, 编译等等. 如 /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core/Formula/foo.rb

Tap (酒吧)

: 所有 Formula 或命令的 Git 仓库, 比方: /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core 对应 https://github.com/Homebrew/homebrew-core 仓库.
咱们平时应用 brew update 就是应用同步 Tap 对应分支的更新. 应用 brew install formula 也是从所有的 tap 中查找 formula 装置.

Bottle (酒瓶)

: 基于对应操作系统已编译好的二进制包, 能够间接应用, 在 rb 文件中会形容各个版本的二进制. 应用二进制包能够省去本人编译. 如: qt-4.8.4.catalina.bottle.tar.gz

Cask (木酒桶)

: 用于形容和装置 macOs 原生 app 软件, 应用 brew cask 装置就和应用 dmg 装置到 Applications 一样的成果.

Keg (小桶)

: 形容的是一个已装置的软件版本门路, 如: /usr/local/Cellar/foo/0.1

Cellar (酒窖)

: 所有已装置软件的存储仓库门路, 如: /usr/local/Cellar

了解下面的一些概念后, 其实要装置历史版本的话, 咱们只须要拿到历史版本的 formula 就能够装置了.

历史版本装置

存在多版本软件

后面说过, 有的软件在 homebrew 的仓库会存在多个版本, 咱们能够间接指定相应的版本进行装置, 咱们能够应用 brew search 查问下要装置的软件的版本.

➜  brew search python

# output
==> Formulae
ipython                           python-markdown                   python@3.11 ✔                     cython
micropython                       python-tabulate                   python@3.7
pr0d1r2/python2/python@2.7.17 ✔   python-tk@3.10                    python@3.8 ✔
ptpython                          python-tk@3.11                    python@3.9 ✔

如上有很多 python 的其它版本, 如: python@3.7, python@3.8 等. 咱们应用 brew install 指定版本装置.

➜ brew install python@3.7

# output
Running `brew update --auto-update`...

无多版本软件

对于 tap 中无多版本的软件, 咱们能够通过在 tap 对应的 git 仓库中查看历史 formula 版本, 通过下载到本地进行装置.

先通过 https://formulae.brew.sh/ 找到软件信息, 如我须要找到 folly 的历史版本, 可依照上面的步骤.

历史版本的 .rb 文件保留到本地之后, 能够应用 brew install 装置. 如:


# 在 .rb 文件保留的目录执行
# 须要先删除原始版本的链接再进行装置
brew unlink folly
brew install folly.rb

这种形式装置有些状况下会呈现装置失败, 比方我下载了一个 13.0.1_1 llvm.rb 文件装置:


➜  brew install -s llvm.rb
Running `brew update --auto-update`...
==> Auto-updated Homebrew!
Updated 1 tap (homebrew/core).
==> Fetching llvm
==> Downloading https://github.com/llvm/llvm-project/releases/download/llvmorg-13.0.1/llvm-project-13.0.1.src.tar.xz
==> Downloading from https://objects.githubusercontent.com/github-production-release-asset-2e65be/75821432/e556705d-3a90-49f1-909f-c1c5e
######################################################################## 100.0%
...

Warning: llvm 15.0.6 is available and more recent than version 13.0.1_1.

Error: An exception occurred within a child process:
  NoMethodError: undefined method `user' for nil:NilClass

下面会呈现 ruby 的执行谬误, NoMethodError: undefined method 'user' for nil:NilClass, 问题的起因是 llvm.rb 中应用了 tap.user 等变量, 而这些变量来自于以后 formula 对应 tap 的 git 仓库信息, 而你下载的 rb 文件是没有放在仓库中.

# llvm.rb 信息

-DPACKAGE_VENDOR=#{tap.user}
-DBUG_REPORT_URL=#{tap.issues_url}
-DCLANG_VENDOR_UTI=org.#{tap.user.downcase}.clang

解决办法就是批改 llvm.rb 信息, 把变量的值间接写成固定值. 如: tap.user 为 “Homebrew”, tap.issues_url 为 “https://github.com/Homebrew/homebrew-core/issues”, tap.user.downcase 为 “homebrew”.

另外一个解决方案就是把下载的 llvm.rb 文件改名放到 tap 对应的仓库中. 如:

# 进入到本地默认 tap 仓库
cd $(brew --repository homebrew/homebrew-core)/Formula

# 批改 llvm.rb 文件到此仓库
mv ~/Downloads/llvm.rb llvm1.rb

# 装置
brew install llvm1

装置实现后, 记得在仓库中革除这个文件.

References

  • https://docs.brew.sh/Formula-Cookbook
  • https://rubydoc.brew.sh/Tap.html#user-instance_method
退出移动版