如何使用Rails 6 / Zeitwerk在Rails初始化程序中预加载问题?

Gly*_*oko 6 ruby ruby-on-rails autoload ruby-on-rails-6

我正在使用一个初始化程序,该初始化程序通过将一些应用程序关注点包含到第三方库中来对应用程序进行一些猴子修补。基本上:

# config/initializers/my_initializer.rb

class SomeExternalLib
  include MyConcern1
  include MyConcern2
end
Run Code Online (Sandbox Code Playgroud)

在Rails 5.2.3中可以正常工作,但是在升级到Rails 6时收到以下弃用消息:

声明警告:初始化会自动加载常量MyConcern1和MyConcern2。

不建议这样做。在将来的Rails版本中,初始化期间的自动加载将成为错误情况。

重新加载不会重新启动应用程序,因此初始化期间执行的代码不会再次运行。因此,例如,如果重新加载ApplicationHelper,则预期的更改将不会反映在该过时的Module对象中。

这些自动加载的常量已被卸载。

请查看“自动加载和重新加载常量”指南以获取解决方案。(从/Users/myuser/code/myapp/config/environment.rb:7调用)

我的担心在应用程序/控制器/问题/中。经过一番调查,我发现该路径没有被自动加载,但是我无法弄清楚如何使Zeitwerk(Rails 6的新自动加载器)动态地加载该路径。我尝试按照此处描述的STI自动加载模式进行操作,但是没有运气。任何想法如何解决此弃用警告?

Jos*_*ers 20

正如@Glyoko 的回答所述,使用require依赖项可防止在初始化程序中自动加载。但是,正如@Puhlze 在他的评论中提到的那样,这样做会导致重新加载期间出现问题。

我偶然发现了Rails.configuration.to_prepare这篇文章中使用的另一种方法。

一个例子是:

# config/initializers/my_initializer.rb

Rails.configuration.to_prepare do
  class SomeExternalLib
    include MyConcern1
    include MyConcern2
  end
end
Run Code Online (Sandbox Code Playgroud)

请注意,这在开发中的每个请求之前运行,但在生产中预先加载之前仅运行一次。

编辑:它似乎也适用于重新加载。


Gly*_*oko 9

如果我更仔细地阅读错误消息会有所帮助:

在 Rails 的未来版本中,初始化期间的自动加载将成为错误条件。

更改的讨论在这里,指南在这里

简而言之,不应该在初始化程序中完成自动加载,这将被逐步淘汰。解决方案是 1) 不要使用需要在初始化程序中自动加载的东西(显然是首选),或者 2) 在初始化程序中明确需要依赖项。

所以我会这样做:

# config/initializers/my_initializer.rb

require 'my_concern1'
require 'my_concern2'

class SomeExternalLib
  include MyConcern1
  include MyConcern2
end
Run Code Online (Sandbox Code Playgroud)

  • 使用“require”的问题是它会破坏所需文件中的代码重新加载。不幸的是,似乎没有更好的选择来维持代码重新加载。 (3认同)