質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.62%

  • Ruby

    7299questions

    Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

  • Ruby on Rails

    6963questions

    Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Sidekiqで動的に生成したworkerを実行すると、NameError:uninitialize constant〜が発生。

受付中

回答 0

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 291

ys3128

score 14

前提・実現したいこと

railsでsidekiqを用いた非同期処理を実装しようと思っています。
クラス・キューをユーザーごとに動的生成を行い、非同期処理をユーザーごとに任意で実行・停止できるようにしたいのですが、下記エラーでキューが実行されません。
(Foo::Baz_User*というクラス、:work-*というキューを生成していきたいです。)
今回のような場合、関数で生成するためスコープの兼ね合いなどで参照できないということでしょうか?
もし対処方法があるようでしたら教えていただきたいと思います。よろしくお願いします。

発生している問題・エラーメッセージ

キューの登録までは実行できるのですが、いざsidekiqがキューを実行する際に、下記のエラーが発生します。

2017-09-12T03:38:47.079Z 60023 TID-ow8lyypd4 WARN: NameError: uninitialized constant Foo::Baz_User1
2017-09-12T03:38:47.080Z 60023 TID-ow8lyypd4 WARN: ~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/activesupport-5.0.6/lib/active_support/inflector/methods.rb:270:in `const_get'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/activesupport-5.0.6/lib/active_support/inflector/methods.rb:270:in `block in constantize'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/activesupport-5.0.6/lib/active_support/inflector/methods.rb:266:in `each'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/activesupport-5.0.6/lib/active_support/inflector/methods.rb:266:in `inject'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/activesupport-5.0.6/lib/active_support/inflector/methods.rb:266:in `constantize'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/activesupport-5.0.6/lib/active_support/core_ext/string/inflections.rb:66:in `constantize'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/sidekiq-3.4.1/lib/sidekiq/processor.rb:46:in `process'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/calls.rb:26:in `public_send'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/calls.rb:26:in `dispatch'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/calls.rb:122:in `dispatch'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/cell.rb:60:in `block in invoke'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/cell.rb:71:in `block in task'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/actor.rb:357:in `block in task'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/tasks.rb:57:in `block in initialize'
~/.rbenv/versions/2.2.7/lib/ruby/gems/2.2.0/gems/celluloid-0.16.0/lib/celluloid/tasks/task_fiber.rb:15:in `block in create'

該当のソースコード

workerファイル

module Foo
    class Bar
    include Sidekiq::Worker

        def self.id_init(user_id)
            klass_name = "#{self.name.split('::').last}_User#{user_id}"
            unless Foo.const_defined?(klass_name)
              Foo.const_set(klass_name.to_sym, self.clone)
              klass = Foo.const_get(klass_name.to_sym)
              queue = klass.instance_variable_get(:@queue).to_s + "-#{user_id}"
              klass.instance_variable_set(:@queue, queue)
            end
            Foo.const_get(klass_name.to_sym)
        end
    end

    class Baz < Bar
        @queue = :work

        def perform(user_id)
          #ループ処理
        end
    end
end

controllerからキューの登録

def start
user = current_user
Sidekiq::Client.enqueue(Foo::Baz.id_init(user.id),user.id)
end

最終的に呼び出されるconstanizeメソッド

def constantize(camel_cased_word)
      names = camel_cased_word.split('::'.freeze)

      # Trigger a built-in NameError exception including the ill-formed constant in the message.
      Object.const_get(camel_cased_word) if names.empty?

      # Remove the first blank element in case of '::ClassName' notation.
      names.shift if names.size > 1 && names.first.empty?

      names.inject(Object) do |constant, name|
        if constant == Object
          constant.const_get(name)
        else
          candidate = constant.const_get(name)
          next candidate if constant.const_defined?(name, false)
          next candidate unless Object.const_defined?(name)

          # Go down the ancestors to check if it is owned directly. The check
          # stops when we reach Object or the end of ancestors tree.
          constant = constant.ancestors.inject(constant) do |const, ancestor|
            break const    if ancestor == Object
            break ancestor if ancestor.const_defined?(name, false)
            const
          end

          # owner is in Object, so raise
          constant.const_get(name, false)
        end
      end
    end

試したこと

http://tech.vasily.jp/entry/resque_to_sidekiq
上記サイトを参考にさせていただきました。

補足情報(言語/FW/ツール等のバージョンなど)

Rails 5.0.6
ruby 2.2.7p470
sidekiq 3.4.1

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

まだ回答がついていません

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.62%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る

  • Ruby

    7299questions

    Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

  • Ruby on Rails

    6963questions

    Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。