大家好我叫乌图米,我会在这里陆续跟大家分享一些有的没的数码体验、软件技巧、系统知识,欢迎大家留言与我交流~如果你喜欢文章的内容,可以在文末点个赞 ,你的支持就是我最大的动力 !
这篇文章介绍一下 Homebrew 如何安装历史软件包。有两种方式,可以添加「homebrew-versions」仓库得到历史版本软件包,也可以主动搜寻历史版本的.rb
安装文件进行安装。
如果你还不知道什么是 Homebrew,笔者这里有一篇介绍它的文章:
乌图米:macOS Homebrew 马克zhuanlan.zhihu.com0 软件包迭代与依赖
软件包随着时间推移,会逐渐更新功能、修复问题,呈现为版本号的向上迭代。brew install
安装的时候是一个版本,隔段时间可能软件包更新了,便可以使用brew upgrade
去更新它。
另一方面,软件包之间存在着依赖关系。开发者可以引用已经提供的软件,使用其中的一些功能,满足自己软件包的需求。由于被依赖存在变动,因此往往会指定一个依赖软件包的版本,避免意料之外的错误。
已知 Homebrew 安装的软件包只能是目前最新的,那么在这样的大环境下就会造成一些问题:如果安装软件包 A 时,需要依赖软件包 B 版本v1.2
,但此时软件包 B 已经更新到了v2.0
了,无法再用 Homebrew 安装v1.2
版的软件包 B。
引出的需求是:如何使用 Homebrew 安装历史版本的软件包?
1 自带仓库的历史版本
事实上,多数软件包之间设定依赖关系时,没有非常严格的版本号限制,往往是比较大的版本要求,例如:不低于v1.2
,不高于v2.0
。严格一点就会直接指定软件版本:必须是v1.2.4
。
肯定有相当多的软件包依赖于许多大型软件包的历史版本。因此,随着 Homebrew 的发展,许多主流的历史版本软件包渐渐被被官方仓库收录,例如python@2
,node@12
等等。
这些软件包只是大的历史版本,没有细致的划分。Homebrew 收录的python@2
的版本是v2.7.17
,node@12
的版本是v12.16.0
。并且这些软件包依然可以安装较小的版本更新。这有点类似于 Windows 10 推出后,微软依旧在为 Windows 7 提供更新。
- 以安装历史版本的
node.js
为例,直接搜索node
:
~ > brew search node
==> Formulae
libbitcoin-node node node-sass node@12 ✔ nodebrew nodenv
llnode node-build node@10 node_exporter nodeenv==> Casks
font-ubuntu-mono-derivative-powerline nodeclipse
nodebox soundnode
- 可以看到一共有三个
node
版本。有最新的node
,也有两个历史版本node12
和node10
。直接安装对应的历史版本即可。
brew info node@12
brew install node@12
2 添加仓库「homebrew-versions」获取历史版本
顺便给出笔者介绍为 Homebrew 添加更多仓库的文章:
乌图米:Homebrew Taps 更多仓库马克zhuanlan.zhihu.com对于「Casks」类软件包,可以使用添加「homebrew-versions」仓库的方式得到一些比较常用的历史版本。其中的历史版本非常有限。因其收录的要求比较严格,时效也比较短。
- 可以这样添加这个仓库:
brew tap homebrew/homebrew-versions
- 之后直接搜索需要的软件包即可。如果仓库收录了其历史版本,将出现在「Casks」分类下。其名称后面往往直接跟着对应的版本号,例如
alfred2
。
~ > brew search alfred
==> Casks
alfred alfred2 alfred3
- 值得注意的是,虽然仓库提供的是更多的历史版本,但也提供
beta
版本的内容。
~ > brew search github-beta
==> Casks
github-beta
你可以在这里看到所有收录的软件包历史版本或测试版本。的确没有多少的内容。
3 根据git
历史获取精准历史版本
如果需求是特别精确的软件包版本,那么很可能没有被任何仓库收录。这时候可以通过访问软件包安装文件.rb
的历史版本,来安装对应时间的软件包。
原理是个啥?
查看软件包的信息时,输出信息如下:
~ > brew info when
...
From: https://mirrors.tuna.tsinghua.edu.cn/git/homebrew/homebrew-core.git/Formula/when.rb
...
注意到图中高亮的一句话,显示的是一个.rb
后缀的文件,是使用ruby
语言编写的文件。ruby
也是 Homebrew 主要的编程语言。如果想了解更多ruby
相关的内容,可以到其社区查看。
- 打开浏览器,复制粘贴刚刚的文件地址。
注意!如果替换过软件源「介绍 Homebrew文章提到过,以此改善 Homebrew 的下载速度」,显示.rb
文件地址将会是替换后的源地址,此时我们需要替换回原来的地址进行查看。例如:
# 现有地址
https://mirrors.tuna.tsinghua.edu.cn/git/homebrew/homebrew-core.git/Formula/when.rb
# 实际地址
https://github.com/Homebrew/homebrew-core/blob/master/Formula/when.rb
替换的规则,使用下面的地址,替换最后的软件包名称是:
# 替换最后的 [formulae] 为软件包的名称
https://github.com/Homebrew/homebrew-core/blob/master/Formula/[formulae].rb
- 浏览器显示是 GitHub 中的对应文件。观察一下文件内容。这其实是软件安装的相关信息与指令。比较熟悉的就是版本号「重点」,下载链接与校验码。
class When < Formula...url "https://deb.debian.org/debian/pool/main/w/when/when_1.1.38.orig.tar.gz"sha256 "139834945142f5e3ea6b20f43ba740d30b4a87b42ce5767026094e633dca999f"...
end
其实每次执行brew install
指令时,Homebrew 做的事情就是去仓库里面找到对应的.rb
文件,这就是安装软件包的执行文件。因此,只要有对应版本的.rb
文件,就可以进行对应版本的软件包安装。
现在要做的是:找到历史版本的.rb
文件,下载到本地;使用 Homebrew 直接执行这份.rb
文件,进入历史版本的安装流程。
- 熟悉
git
的盆友应该知道,这份文件是最新提交的版本。既然有最新提交的版本,当然有历史版本。点击右上角的「HIstory」按钮,查看历史版本。
- 原本正常的流程是:从提交历史中找到对应的历史版本文件,查看
raw
文件,保存文件到本地。然而实际上,由于这个仓库过于庞大,直接查看历史版本,通常会遇到无法加载的情况。
那就下来还有两种方法可以尝试,用于获取历史版本.rb
文件。
- 第二种是利用本地历史仓库,查看本地历史的提交记录,获取其中旧版本文件的
url
与sha256
片段。 - 第三种是直接主动修改当前
.rb
文件的url
与sha256
片段,欺瞒 Homebrew 进行安装。
目前以上三种方法的前两种,笔者都没有找到when
这个软件包的历史版本。陷入沉思,应该与换源有很大关系。等回头搞清楚了再来更新。
但是第三种可以「耍赖」地尝试一下:
- 保存当前最新的
.rb
文件到本地,当然里面的版本号是v1.1.38
。编辑when.rb
,我们耍赖地其中版本号修改为v1.1.36
。当然,可以修改为任意存在的版本,可以自己尝试。改完记得保存一下。
- 回到 Homebrew,使用这份修改过的
when.rb
文件来安装when
软件包。你可以选中文件之后,cmd + option + c
复制文件路径。
brew install /Users/tommy/Downloads/when.rb # 注意这里是你下载文件的路径!
- 「毫不意外」的回报错:第一,我替换的清华镜像源中没有该历史版本,不过它会自动转为开发者提供的下载链接;第二,文件的校验码不对,根据文件期望得到的值与实际值不吻合。
- 我们继续耍赖。将校验码实际值复制下来,编辑
when.rb
,替换其中的校验码为我们刚刚复制到的、所下载文件的实际校验码。这是在规避校验码无法匹配的问题。
- 重新执行上面的安装指令。现在可以看到,文件可以正常安装了。
- 最后执行一下
brew info when
。可以看到,此时安装的版本是v1.1.36
,不是最新的v1.1.38
。
总结一下,上面三种方法都是利用旧的本地.rb
文件,实现旧版本的安装。其中第三种是比较通用的,因为你可以「耍赖」地无限尝试。尤其是当有依赖包指定了具体的版本号,就更容易了。
4 锁定软件包
好不容易安装好了历史版本的软件包,当然最好是锁定它,避免其被更新。
- 使用
pin
指令锁定以安装的软件包。之后不会在upgrade
指令中更新被锁定的软件包。
brew pin when # 锁定
- 在
ls
指令后添加--pinned
参数查看当前全部被锁定的软件包与版本号。
brew ls --pinned # 查看当前锁定的软件包版本
- 使用
unpin
指令解除软件包的锁定。然后可以正常在outdated
和upgrade
指令中查看和更新落伍的软件包。
brew unpin when # 解锁
尤其注意,如果你对这个软件包的依赖是长期的,那么尽量锁定;如果只是暂时的需求,可以不用锁定。
不过,可以看到,upgrade
指令发现pinned
软件包时,将直接提示报错。这说明 Homebrew 是非常不提倡固定软件包版本的。
5 注意事项
有一些需要注意的地方:
- 不建议使用历史软件包,Homebrew 也是同样的态度。更新总是有原因的,尽量向新版本看齐。
- 如果已经安装了更新版本的软件包、再安装旧版,是不可以覆盖安装的。安全的操作是先
brew unlink
取消软件包的关联,再卸载新版本软件包,最后尝试安装更旧的版本。 - 理论上来说,只要可以获取某个时期的安装包及其对应的校验码,都是可以甩给 Homebrew 安装的。但获取正确的校验码及文件,还是需要参考仓库的提交历史。
- 「耍赖」方法一时爽,一直耍赖小心出问题 。
共同学习,武汉加油!