3

我试图在这里解决一些谜团......我们创建了一个通过 cron 调用的 gem。

谜团是:这适用于机器A但不适用于机器B,据我所知,环境是相同的。

两台机器都是 Mac OS X 10.6。

我也了解 cron 在简约的环境中运行。

crontab:

10 2 * * * /Users/michael/.rvm/gems/ruby-1.9.3-p194/bin/my_gem

机器上的错误B是:

/Users/michael/.rvm/rubies/ruby-1.9.3-p194/lib/ruby/site_ruby/1.9.1/rubygems/dependency.rb:247:in `to_specs': Could not find my_gem-toolchain (>= 0) amongst [bigdecimal-1.1.0, io-console-0.3, json-1.5.4, minitest-2.5.1, rake-0.9.2.2, rdoc-3.9.4] (Gem::LoadError)
    from /Users/michael/.rvm/rubies/ruby-1.9.3-p194/lib/ruby/site_ruby/1.9.1/rubygems/dependency.rb:256:in `to_spec'
    from /Users/michael/.rvm/rubies/ruby-1.9.3-p194/lib/ruby/site_ruby/1.9.1/rubygems.rb:1231:in `gem'
    from /Users/michael/.rvm/gems/ruby-1.9.3-p194/bin/my_gem:22:in `<main>'

这里是内容/Users/michael/.rvm/gems/ruby-1.9.3-p194/bin/my_gem

#!/Users/michael/.rvm/rubies/ruby-1.9.3-p194/bin/ruby
#
# This file was generated by RubyGems.
#
# The application 'my_gem-toolchain' is installed as part of a gem, and
# this file is here to facilitate running it.
#

require 'rubygems'

version = ">= 0"

if ARGV.first
  str = ARGV.first
  str = str.dup.force_encoding("BINARY") if str.respond_to? :force_encoding
  if str =~ /\A_(.*)_\z/
    version = $1
    ARGV.shift
  end
end

gem 'my_gem-toolchain', version
load Gem.bin_path('my_gem-toolchain', 'my_gem', version)

第 22 行是:gem 'my_gem-toolchain', version

这是env在两台机器上通过 cron 运行的输出:

SHELL=/bin/sh
USER=michael
PATH=/usr/bin:/bin
PWD=/Users/michael
SHLVL=1
HOME=/Users/michael
LOGNAME=michael
_=/usr/bin/env

我相信这与GEM_PATH未设置有关。但是,在机器AGEM_PATH似乎没有设置,但一切正常。

我想更好地了解红宝石的工作原理。显然我仍然缺少一些东西。

为什么这可以在机器上工作,A但不能在机器上工作B

4

1 回答 1

2

RVM 是开发人员维护其环境的绝佳工具,但在生产环境中却是一场噩梦。一旦你从 cron 执行任务,为了维护执行环境,你本质上是“在生产中”,即使它恰好是你的开发机器上的 cron。

我建议您在系统范围内安装 gem 并从脚本运行它,该脚本不假设它的安装位置。确保您遵循创建 gem 的所有最佳实践,以便正确安装 gem 的依赖项并正确处理二进制文件等。一旦您构建了 gem,您应该通过系统 Ruby 安装它,以便它可以用于操作系统的 Ruby 解释器的默认位置中的所有用户,以及它的所有依赖项都已安装。

查看 Rubygems 的Make your own gem 指南以获得最佳实践。

然后使用系统 ruby​​ 安装生成的 gem 文件,而不是安装到您的 RVM 目录中:

rvm use system
gem install my-gem-1.0.0.gem

这应该得到所有设置,以便它可以从 cron 和任何用户愉快地运行。您可能不想在您的主目录中运行依赖于 RVM 的 cron 作业。

为了具体解决您的问题,它可能适用于 A 但不适用于 B 的原因有很多。如果您在一台机器上安装 gem 系统范围并通过 RVM 在另一台机器上安装,这将影响安装依赖项的位置(系统-wide vs. 在您的主目录中);如果您甚至在另一台机器上都没有 RVM,那肯定是这种情况。可能是您的 Gem 没有正确声明其依赖项,因此当您将其安装在机器 B 上时,依赖项并未安装在那里,而是您在开发过程中手动将它们安装在机器 A 上。可能是您在两者上都安装了相同的 gem,并且主目录是共享的,并且其他所有内容似乎都是重复的,但是您的 shell 配置中的 RVM 魔法并没有在两者上执行。可能是不同的用户,因此不同的环境。无论根本原因是什么,只要将 RVM 排除在外,并遵循 ruby​​gems 指南,你就会好起来的。

于 2013-01-24T01:29:14.467 回答