6

私は同時実行性の問題であるに違いないと思うものを抱えています。私は乗用車、レール 2.3.5、モンゴイド 1.9.2、モンゴ ルビー ドライバー 1.0.9 を使用しています。jQuery を使用して、MongoDB からプルされ、ブラウザーでレンダリングされるデータを要求しています。このようなリクエストを同時に 2 つ行うようになるまで、すべてがうまく機能しています。モデルでは、これらはリクエストによって実行されるメソッドです:

Class Visit
  include Mongoid::Document
  ...
  def self.cancellations_and_visits_by_therapist_graph(clinic_id) 
    visits = collection.group("function(x){ return { resource_id : x.resource_id } }",
            {:clinic_id => clinic_id, :visit_date => { "$gte" => Time.now - 6.months, "$lte" => Time.now}}, 
            {:visits => 0, :cancel_no_shows => 0},
            'function(obj, count) {
              if (obj.visit_status == "NO SHOW" || obj.visits_status == "CANCELLED") {
                count.cancel_no_shows += 1;
              } else {
                count.visits += 1;
              }

            }')

    visits = visits.group_by {|g| g['resource_id']}

    Resource.any_in(:mysql_id => visits.keys).order_by([:last_name, :asc]).order_by([:first_name, :asc]).inject({ 'visits' => [], 'cancel_no_shows' => [], 'xlabels' => []}) do |formatted_visits, resource|
      formatted_visits['visits'] << visits[resource.mysql_id.to_f].first['visits']
      formatted_visits['cancel_no_shows'] << visits[resource.mysql_id.to_f].first['cancel_no_shows']
      formatted_visits['xlabels'] << resource.last_name + ", " + resource.first_name
      formatted_visits
    end
  end



  def self.total_visits_for_graph(practice_id)
    visits = collection.group("function(x) { return { clinic_id : x.clinic_id } }", 
                          {:practice_id => practice_id, :visit_status => 'COMPLETE', :visit_date => { "$gte" => Time.now - 6.months, "$lte" => Time.now}},
                          {:visits => 0}, "function(obj, count) { count.visits += 1; }")

    visits = visits.group_by {|g| g['clinic_id']}
    Clinic.any_in(:mysql_id => visits.keys).order_by([:name, :asc]).inject({ 'data' => [], 'xlabels' => []}) do |formatted_visits, clinic|
      formatted_visits['data'] << visits[clinic.mysql_id.to_f].first['visits']
      formatted_visits['xlabels'] << clinic.name
      formatted_visits
    end
  end
end

問題を説明する最善の方法は、Mongo からの結果が間違ったオブジェクトに渡されているということです。私はこれの例を見ました:

これは、CLinic.any_in を呼び出したときに返されたものです (グループの 1 つからの結果です)。

{"group"=>{"$keyf"=>"function(x){ return { resource_id : x.resource_id } }", "cond"=>{:clinic_id=>101, :visit_date=>{"$gte"=>Tue Apr 20 15:34:37 +0800 2010, "$lte"=>Wed Oct 20 15:34:37 +0800 2010}}, "ns"=>"visits", "initial"=>{:visits=>0, :cancel_no_shows=>0}, "$reduce"=>"function(obj, count) {\n              if (obj.visit_status == \"NO SHOW\" || obj.visits_status == \"CANCELLED\") {\n                count.cancel_no_shows += 1;\n              } else {\n                count.visits += 1;\n              }\n\n            }"}}

これ (クリニック オブジェクト) は、collection.group 呼び出しによって返されました。

{"_id"=>BSON::ObjectId('4cb7d72b3bc5457800ce2e6f'), "name"=>"Corona", "practice_id"=>39, "mysql_id"=>101}

すべての優れた並行性の問題と同様に、結果はランダムであり、うまく機能することもあれば、失敗することもあります。私はmongoとmongoidが初めてなので、これがmongoidの問題なのかmongoドライバーの問題なのか実際にはわかりませんが、Mongoidに関連していると思います。Mongoid をレールにロードするために使用している初期化子を含めています。任意のアイデアまたは追加のデバッグのアイデアでさえも大歓迎です。

繋がり

mongoid_conf = YAML::load_file(Rails.root.join('config/mongoid.yml'))[Rails.env]

Mongoid.configure do |config|
  config.master = Mongo::Connection.new(mongoid_conf['host'], 27017, :pool_size => 5, :timeout => 5).db(mongoid_conf['database'])
end
4

1 に答える 1

6

私はこの問題の解決策を見つけました。それは実際にはMongoドライバーでもMongoidでもありませんでした、それは乗客でした。Passengerがレールプロセスにまたがる場合、現在のインスタンスをフォークして、ファイル記述子(TCP記述子を含む)がアプリインスタンス間で共有されるようにします。これは、mongoが同じソケットによって書き込まれ、読み取られていることを意味します。これが同時実行の問題につながります。解決策は、mongoがフォークするたびに再接続することです。これは私が見つけた解決策です:

# Handle the creation of new processes by Phusion Passenger
if defined?(PhusionPassenger)
  PhusionPassenger.on_event(:starting_worker_process) do |forked|
    if forked
      # We're in smart spawning mode.

      # Reset the connection to MongoDB
      Mongoid.config.master.connection.close
      load File.join(RAILS_ROOT, 'config/initializers/mongoid_init.rb')
    else
      # We're in conservative spawning mode. We don't need to do anything.    
    end
  end
end

元の回答の投稿とディスカッションについては、 http: //groups.google.com/group/mongodb-user/browse_thread/thread/f31e2d23de38136aを参照してください。

于 2010-10-22T01:55:41.360 に答える