Rails Update Action在rails4, mongoid上失败.创建好了



我有一个非常简单的rails应用程序w/rails4和mongoid。我可以像魔术一样创建新的数据集。但是我就是不能更新现有的数据集。

有人有这个问题吗?这是怎么回事,我做错了什么?

刚刚用rails 4, ruby 2和mongoid从头创建,全部来自他们的git仓库:

rails new mongotest --skip-active-record

生成一个scaffold:

rails g scaffold things name description

我的模型现在看起来像这样:

class Thing
  include Mongoid::Document
  field :name, type: String
  field :description, type: String
end

Controller like that:

class ThingsController < ApplicationController
  before_action :set_thing, only: [:show, :edit, :update, :destroy]
  # GET /things
  # GET /things.json
  def index
    @things = Thing.all
  end
  # GET /things/1
  # GET /things/1.json
  def show
  end
  # GET /things/new
  def new
    @thing = Thing.new
  end
  # GET /things/1/edit
  def edit
  end
  # POST /things
  # POST /things.json
  def create
    @thing = Thing.new(thing_params)
    respond_to do |format|
      if @thing.save
        format.html { redirect_to @thing, notice: 'Thing was successfully created.' }
        format.json { render action: 'show', status: :created, location: @thing }
      else
        format.html { render action: 'new' }
        format.json { render json: @thing.errors, status: :unprocessable_entity }
      end
    end
  end
  # PATCH/PUT /things/1
  # PATCH/PUT /things/1.json
  def update
    respond_to do |format|
      if @thing.update(thing_params)
        format.html { redirect_to @thing, notice: 'Thing was successfully updated.' }
        format.json { head :no_content }
      else
        format.html { render action: 'edit' }
        format.json { render json: @thing.errors, status: :unprocessable_entity }
      end
    end
  end
  # DELETE /things/1
  # DELETE /things/1.json
  def destroy
    @thing.destroy
    respond_to do |format|
      format.html { redirect_to things_url }
      format.json { head :no_content }
    end
  end
  private
    # Use callbacks to share common setup or constraints between actions.
    def set_thing
      @thing = Thing.find(params[:id])
    end
    # Never trust parameters from the scary internet, only allow the white list through.
    def thing_params
      params.require(:thing).permit(:name, :description)
    end
end
我Gemfile

:

source 'https://rubygems.org'
# Bundle edge Rails instead: gem 'rails', github: 'rails/rails'
gem 'rails',     git: 'https://github.com/rails/rails.git'
gem 'arel',      git: 'https://github.com/rails/arel.git'
gem 'activerecord-deprecated_finders', git: 'https://github.com/rails/activerecord-deprecated_finders.git'
gem 'thin'
gem 'mongoid', git: 'https://github.com/mongoid/mongoid.git'

# Gems used only for assets and not required
# in production environments by default.
group :assets do
  gem 'sass-rails',   '~> 4.0.0.beta1'
  gem 'coffee-rails', '~> 4.0.0.beta1'
  # See https://github.com/sstephenson/execjs#readme for more supported runtimes
  # gem 'therubyracer', platforms: :ruby
  gem 'uglifier', '>= 1.0.3'
end
gem 'jquery-rails'
# Turbolinks makes following links in your web application faster. Read more: https://github.com/rails/turbolinks
gem 'turbolinks'
# Build JSON APIs with ease. Read more: https://github.com/rails/jbuilder
gem 'jbuilder', '~> 1.0.1'

这是来自尝试更新数据集时的日志:

Started PATCH "/things/51669078e05658cf22000001" for 127.0.0.1 at 2013-04-11 12:29:25 +0200
Processing by ThingsController#update as HTML
  Parameters: {"utf8"=>"✓", "authenticity_token"=>"lfELkffFSf9gk04TnHnzG9cCrOe3XrsjK4fEZ7Rt7qQ=", "thing"=>{"name"=>"qqqq", "description"=>"qqqqq"}, "commit"=>"Update Thing", "id"=>"51669078e05658cf22000001"}
  MOPED: 127.0.0.1:27017 QUERY        database=mongotest_development collection=things selector={"_id"=>"51669078e05658cf22000001"} flags=[:slave_ok] limit=0 skip=0 batch_size=nil fields=nil (0.5772ms)
Redirected to http://localhost:3000/things/51669078e05658cf22000001
Completed 302 Found in 6ms

如果有人有任何提示或链接…非常感谢你的帮助。

更新1:我确实在rails 3环境中尝试过,它有效…和预期的完全一样

更新2:记录器输出"Mongoid.logger. log ."level = Logger::DEBUG"one_answers"Moped.logger. DEBUG"。level = Logger::DEBUG":

Started PATCH "/things/5166c5ece05658f08a000001" for 127.0.0.1 at 2013-04-11 16:20:56 +0200
Processing by ThingsController#update as HTML
  Parameters: {"utf8"=>"✓", "authenticity_token"=>"lfELkffFSf9gk04TnHnzG9cCrOe3XrsjK4fEZ7Rt7qQ=", "thing"=>{"name"=>"EditTest", "description"=>"123"}, "commit"=>"Update Thing", "id"=>"5166c5ece05658f08a000001"}
D, [2013-04-11T16:20:56.450464 #37961] DEBUG -- :   MOPED: 127.0.0.1:27017 QUERY        database=mongotest_development collection=things selector={"_id"=>"5166c5ece05658f08a000001"} flags=[:slave_ok] limit=0 skip=0 batch_size=nil fields=nil (0.7041ms)
Redirected to http://localhost:3000/things/5166c5ece05658f08a000001
Completed 302 Found in 6ms

Update 3: mongoid。我用于最后一次日志输出的yml:

development:
  # Configure available database sessions. (required)
  sessions:
    # Defines the default session. (required)
    default:
      # Defines the name of the default database that Mongoid can connect to.
      # (required).
      database: mongotest_development
      # Provides the hosts the default session can connect to. Must be an array
      # of host:port pairs. (required)
      hosts:
        - localhost:27017
      options:
        # Change whether the session persists in safe mode by default.
        # (default: false)
        safe: true

我的例子应用程序在https://github.com/jlxq0/mongotest

更新4:更多的研究表明,这可能是一个类似Rails 4.0.0的问题。beta 1和Mongoid

如果没有人有答案,我也会很高兴,如果有人分享一个链接到一个工作的rail4/Mongo的例子,这样我就可以找出自己的差异。

更新5:在Rails控制台中,更新工作得很好

这是因为update不接受属性,它只接受选项,就像@thing.update(validate: false)

为了使你的代码工作,你可以这样做:

if @thing.update_attributes(thing_params)
  #...
end

或:

@thing.attributes = thing_params
if @thing.save
  #...
end

或:

@thing.attributes = thing_params
if @thing.update
  #...
end

MongoDB有一种叫做安全模式的东西,默认情况下它是关闭的,这意味着无论你告诉它执行什么,它都会为它返回OK。如果你将安全模式设置为on,如果你执行了一个根据mongo标准是非法的查询,它将引发一个异常。

在开发中,你通常会这样做,

development:
  sessions:
    default:
      hosts:
        - localhost:27017
      database: mongoid
      username: user
      password: password
      options:
        safe: true
在那之后,你可以把你的异常发给我们,我们可以尝试解决它!

如果您正在使用嵌入属性(无论是动态创建的哈希,还是显式定义的关系),使用upsert而不是update可能会有所帮助。

Upsert将导致如果条目存在则重写字段,如果条目不存在则创建字段。

person = Person.new(
  first_name: "Heinrich",
  last_name: "Heine"
)
person.upsert

示例来源:http://mongoid.org/en/mongoid/docs/persistence.html

相关内容

  • 没有找到相关文章

最新更新