当我需要一个不同的版本时,如何解决需要最新版本 gem 的 bundler 的问题?

Arr*_*ott 8 ruby ubuntu ruby-on-rails bundler puma

近 2 周以来,我一直在为这个问题挠头。我有一个安装了 rbenv 的 Ubuntu 14.04 服务器,运行许多不同的 Rails 网站,其中一些在旧版本的 Rails 上,其中一些在最新版本上。

我有 2 个网站,它们都需要不同版本的 puma_worker_killer,1 个需要 0.1.0,另一个需要 0.1.1。这两个网站都使用 Ruby 2.5.3。

当我启动服务器时,RAILS_ENV=dev3 bundle exec pumactl -F ./config/puma.rb start我在日志中收到以下错误并且网站挂起:

You have already activated puma_worker_killer 0.1.1, but your Gemfile requires puma_worker_killer 0.1.0. Prepending `bundle exec` to your command may solve this. (Gem::LoadError)
Run Code Online (Sandbox Code Playgroud)

起初我认为这可能是 rbenv 的问题,因为我将 gems 安装在 ~/.gem 而不是 ~/.rbenv 中,所以我在 ~/.gem 中对所有红宝石进行了核处理,并将它们重新安装到正确的位置rbenv 文件夹,bundle install我仍然遇到同样的问题。

现在我想澄清一下,我已经在网上对这个主题进行了广泛的研究,我知道我可以做很多事情来解决这个问题。

我知道我可以更改版本和bundle update puma_worker_killer.

我也知道我可以通过执行gem uninstall puma_worker_killer和选择 0.1.1来删除最新版本,但这意味着不会满足对其他网站的依赖。

我已经深入研究了 bundler 的源代码,可以看到它是由以下代码行引起的:

You have already activated puma_worker_killer 0.1.1, but your Gemfile requires puma_worker_killer 0.1.0. Prepending `bundle exec` to your command may solve this. (Gem::LoadError)
Run Code Online (Sandbox Code Playgroud)

当在 bundler 的上下文中bundle exec同时使用theactivated_specspecmatch 时,这意味着该方法 ( check_for_activated_spec!) 中的以下代码不会运行。出于某种原因,当运行上面的命令启动服务器时,activated_spec(激活的 gem)是最新版本(0.1.1)而不是 Gemfile(0.1.0)中列出的版本,这意味着它不会返回并抛出上面的错误。

我还应该提到,get_process_mem 似乎也存在同样的问题,它是 puma_worker_killer 的依赖项之一。它抱怨已经激活 0.2.5 但我的 Gemfile 想要 0.2.4:

You have already activated get_process_mem 0.2.5, but your Gemfile requires get_process_mem 0.2.4. Prepending `bundle exec` to your command may solve this. (Gem::LoadError)
Run Code Online (Sandbox Code Playgroud)

我对 bundler 的理解是,它应该加载 Gemfile 中列出的版本,bundle exec以解决具有相同 gem 的多个版本的问题。

我知道我也可以创建一个单独的 gemset(显然可以用 rbenv 完成),其中包含不同版本的 puma_worker_killer,然后运行rbenv local 2.5.3-pwk0.1.0rbenv local 2.5.3-pwk0.1.1根据我想要的版本,在项目中,但这对于我想要的来说似乎太过分了达到。

按照这个速度,我很想只用 puma_worker_killer 和 get_process_mem 的最新版本更新所有网站,然后将它们锁定并删除服务器上的所有旧版本,但我认为我不应该这样做。

有谁知道这里发生了什么,或者我是否在做一些明显错误的事情?

下面是我用来在 puma 配置中使用 puma_worker_killer 的一段代码。

return if activated_spec.version == spec.version
Run Code Online (Sandbox Code Playgroud)

ach*_*ion 1

这里发生的事情基本上是你的系统中有多个版本的 gem。

大多数时候它不会引起问题,因为bundle exec会动态加载应用程序所需的版本。

在某些情况下,gems 会包含二进制文件。这种情况bundle exec不会有任何帮助,因为您在一瞬间只能链接一个版本。

基本上,如果您想通过别名调用二进制文件,则必须为每个应用程序使用单独的 gemset。

如果你想将所有的 gems 保存在一个地方,你可以直接调用二进制文件。

在你的情况下它将是:

RAILS_ENV=dev3 bundle exec pumactl _0.1.0_ -F ./config/puma.rb
Run Code Online (Sandbox Code Playgroud)

_<version>_构造允许您指定要运行的二进制文件的版本。

您也可以创建自定义二进制文件,就像fake_pumactl在项目内部一样,它将检查Gemfile.lock并自动代理您对库的调用,并自动为您指定版本。另一种方法是通过 shell 脚本解析 gem 版本,并将此脚本放入_<version>_您的调用中。

这是一个简短的例子

$ gem install puma
Fetching puma-4.3.3.gem

$ gem install puma -v 4.3.0
Fetching puma-4.3.0.gem

$ pumactl -v
4.3.3

$ pumactl _4.3.0_ -v
4.3.0

$ ruby -v
ruby 2.6.3p62

$ export puma_version=_4.3.0_
$ pumactl ${puma_version} -v
4.3.0
Run Code Online (Sandbox Code Playgroud)

puma_version变量可以从 bash 命令的结果定义,该命令将从中提取 gem 版本Gemfile.lock