如何从 rake 任务中提前返回?

Posted

技术标签:

【中文标题】如何从 rake 任务中提前返回?【英文标题】:How do I return early from a rake task? 【发布时间】:2011-01-19 23:07:10 【问题描述】:

我有一个 rake 任务,我在开始时会进行一些检查,如果其中一个检查失败,我想从 rake 任务中提前返回,我不想执行任何剩余的代码。

我认为解决方案是在我想从代码返回的地方放置一个返回,但我收到以下错误

unexpected return

【问题讨论】:

【参考方案1】:

Rake 任务基本上是一个块。一个块,除了 lambdas,不支持 return,但是你可以使用 next 跳到下一个语句,这在 rake 任务中具有与在方法中使用 return 相同的效果。

task :foo do
  puts "printed"
  next
  puts "never printed"
end

或者你可以在方法中移动代码并在方法中使用return。

task :foo do
  do_something
end

def do_something
  puts "startd"
  return
  puts "end"
end

我更喜欢第二种选择。

【讨论】:

我也最喜欢第二个。我使用 rake 的次数越多,我就越喜欢将重要的代码保留在任务定义之外。不是 100% 严格的规则,但似乎是一个很好的工作指南。 我已经尝试使用break,但我遇到了这个错误:rake aborted!从 proc-closure 中断(通过使用 --trace 运行任务查看完整跟踪) 我更喜欢使用下一个。为什么我们要声明一个新方法只是为了支持提前返回? 如果你嵌套在多个块​​中,你会怎么做? (next 仅在有“级别”的块需要突破时才有效。 警告:在 Rake 任务中声明方法是个坏主意,因为它们对所有加载的 Rake 任务都是全局的,与命名空间无关。使用 Next 而不是 break,因为块中的代码可能会被执行块的任何东西多次调用(想想 .each 方法)。【参考方案2】:

您可以在任务内部使用abort(message) 来通过消息中止该任务。

【讨论】:

@TylerRick 不,是Kernel#abort。 这种方式非常适合在不成功的情况下退出,因为它会自动设置退出状态。 这是赢家。也是为参数错误提供使用反馈的简单方法。 内联且比next 更具解释性。喜欢它。 如果您正在测试您的 rake 任务,这可能不是一个好主意,这很可能会导致测试失败。【参考方案3】:

返回错误❌

如果您返回 with 错误(即退出代码 1),您将需要使用 abort,它还需要一个可选参数退出时输出的字符串参数:

task :check do
  
  # If any of your checks fail, you can exit early like this.
  abort( "One of the checks has failed!" ) if check_failed?

end

在命令行上:

$ rake check && echo "All good"
#=> One of the checks has failed!

成功归来✅

如果您返回没有错误(即退出代码0),您需要使用exit确实如此不 采用字符串参数。

task :check do
  
  # If any of your checks fail, you can exit early like this.
  exit if check_failed?
  
end

在命令行上:

$ rake check && echo "All good"
#=> All good

如果您在 cron 作业中使用它,或者需要根据 rake 任务是否成功执行某些操作,这一点很重要。


奖励:从没有堆栈跟踪的 rescue 块返回错误。

默认情况下,如果您在 rescue 块内使用 abort,它将输出整个堆栈跟踪,即使您只使用 abort 而不会重新引发错误。

要解决这个问题,您可以向exit 命令提供一个非零退出代码,例如:


task :check do

  begin
    do_the_thing_that_raises_an_exception
  rescue => error
    puts error.message
 
    exit( 1 )
  end

end

【讨论】:

这个应该被接受答案@simone-carletti【参考方案4】:

我倾向于使用abort,在这种情况下这是一个更好的选择,例如:

task :foo do
  something = false
  abort 'Failed to proceed' unless something
end

【讨论】:

但是你如何abort 而不使用1 退出代码退出? Rake 任务通常在命令行中用于确定成功或失败。有没有“成功”的abort 回答了我自己的问题:看起来exit 是成功退出的好方法。【参考方案5】:

如果您需要突破多个块级别,可以使用fail。

例如

task :something do
  [1,2,3].each do |i|
    ...
    fail "some error" if ...
  end
end

(见https://***.com/a/3753955/11543。)

【讨论】:

【参考方案6】:

如果您的意思是退出 rake 任务而不导致“rake 中止!”要打印的消息,那么您可以使用“中止”或“退出”。但是,当在救援块中使用“中止”时,会终止任务并打印整个错误(即使不使用 --trace)。所以“退出”是我使用的。

【讨论】:

总的来说,我认为使用“exit”而不是 return/break 是一个坏主意,因为它不只是跳出 current proc/method/etc。 -- 它退出整个过程并跳过调用者方法可能打算在之后运行的任何代码(可能包括一些清理)。但对于 rake 任务,我想这可能不是问题...... "abort",当在救援块中使用时,会终止任务并打印整个错误(即使不使用 --trace)。 你说对了这!在其他任何地方都找不到这个。我也更新了我的答案以表明这一点。谢谢!【参考方案7】:

我使用了 Simone Carletti 建议的 next 方法,因为在测试 rake 任务时,abort(实际上只是 exit 的包装器)不是所需的行为。

例子:

task auto_invoice: :environment do
  if Application.feature_disabled?(:auto_invoice)
    $stderr.puts 'Feature is disabled, aborting.'
  next
end

【讨论】:

以上是关于如何从 rake 任务中提前返回?的主要内容,如果未能解决你的问题,请参考以下文章

如何从 Capistrano 运行 rake 任务?

如何从控制台运行 rake 任务?

如何在 Ruby 脚本中运行 Rake 任务?

如何编写 Rake 任务以将数据导入 Rails 应用程序?

rake 不返回任何东西

Sinatra 应用程序中的“Rake 无法加载此类文件”错误