Rails 5升级打破了延迟任务

时间:2019-02-13 16:54:04

标签: ruby-on-rails ruby ruby-on-rails-5 delayed-job ruby-on-rails-4.2

我将Rails应用程序从Rails 4.2升级到Rails 5.0,现在我在升级前创建的许多delayed jobs都坏了。我收到以下错误:

undefined method 'game_completion_feedback' for ConfirmationMailer:Class

即使我在ConfirmationMailer类中定义了该方法,并且在该类中或升级时从何处调用该方法,都没有改变。

在执行YAML.load_dj时出现以下错误:

ArgumentError: undefined class/module ActiveRecord::ConnectionAdapters::PostgreSQL::OID::Integer
from <path>/.rvm/rubies/ruby-2.2.5/lib/ruby/2.2.0/psych/class_loader.rb:53:in `path2class'
Caused by NameError: uninitialized constant 
ActiveRecord::ConnectionAdapters::PostgreSQL::OID::Integer
from <path>/.rvm/gems/ruby-2.2.5/gems/activesupport-5.0.7.1/lib/active_support/inflector/methods.rb:283:in `const_get' 

它似乎坏了,因为在将Rails 4.2升级到Rails 5.0期间发生了一些变化。

我在线发现运行Rails.cache.clear可以帮助解决此问题,但是我在tmp环境中的production文件夹为空,因此运行Rails.cache.clear只会引发错误,如下所示:< / p>

No such file or directory @ dir_initialize - /var/app/current/tmp/cache/

有什么办法可以使这些旧的延迟工作仍在Rails 5.0中工作,还是我只需要单独重新创建所有这些工作?

我的ConfirmationMailer班:

class ConfirmationMailer < ApplicationMailer
  def game_completion_feedback(user, date, feedback)
    @user     = user
    @date     = format_time(date.to_time)
    @feedback = feedback

    mail(to: user.email, subject: 'Game Completed')
  end
end

我将该函数称为:

def send_feedback_to_client
  ConfirmationMailer.delay.game_completion_feedback(user, date, feedback)
end

在其他情况下,例如我在呼叫Struct时也会出现这种情况:

class RemindersForGame < Struct.new(:gamer_email, :leader_email, :start)
  def perform
    ConfirmationMailer.game_reminder_email_gamer(gamer_email, leader_email, start).deliver_now
    ConfirmationMailer.game_reminder_email_leader(gamer_email, leader_email, start).deliver_now
  end
end

我将此struct称为:

def create_reminder_email(start)
  reminders = Delayed::Job.enqueue RemindersForGame.new(client.user, coach, start),
                                 run_at: start - 2.day,
                                 queue: 'game_reminder'

self.reminders_job_id = reminders.id

结束

game_reminder_email_gamergame_reminder_email_leader的定义与该类中的其他方法完全相同,并且我没有更改与调用方式有关的任何内容。

2 个答案:

答案 0 :(得分:0)

对于版本ConfirmationMailer.delay.game_completion_feedback(user, date, feedback)

对于版本> Rails 4.2:ConfirmationMailer.game_completion_feedback(user, date, feedback).deliver_later

请尝试使用它,让我们知道它是否可以解决问题。

此外,当将变量传递到Mailer类时,使用with()方法将创建实例变量供您在mailer实例中使用。例如:

ConfirmationMailer.with(u: user, d: date, f: feedback).game_completion_feedback.deliver_later

然后将创建@u, @d, @f作为实例变量以在Mailer实例中使用。

我不建议您将变量命名为单个字符:)但是请表明您不需要位置参数。

答案 1 :(得分:0)

ArgumentError: undefined class/module ActiveRecord::ConnectionAdapters::PostgreSQL::OID::Integer是由Rails 5的更改引起的,以致ActiveRecord::ConnectionAdapters::PostgreSQL::OID::IntegerActiveRecord::ConnectionAdapters::PostgreSQL::OID::Float不再存在。您可以使用以下迁移来解决该问题:

class UpdatePostgresDataTypeInDj < ActiveRecord::Migration[5.1]
  REPLACEMENTS = {
    'ruby/object:ActiveRecord::ConnectionAdapters::PostgreSQL::OID::Integer': 'ruby/object:ActiveModel::Type::Integer',
    'ruby/object:ActiveRecord::ConnectionAdapters::PostgreSQL::OID::Float': 'ruby/object:ActiveModel::Type::Float',
  }

  def up
    REPLACEMENTS.each do |old, new|
      Delayed::Job.where("handler LIKE ?", "%#{old}%")
        .update_all("handler = replace(handler, '#{old}', '#{new}')")
    end
  end

  def down
    REPLACEMENTS.each do |old, new|
      Delayed::Job.where("handler LIKE ?", "%#{new}%")
        .update_all("handler = replace(handler, '#{new}', '#{old}')")
    end
  end
end