【问题标题】:Rails error with rspec: "expected valid? to return true, got false"rspec 的 Rails 错误:“预期有效?返回 true,得到 false”
【发布时间】:2012-10-11 04:09:02
【问题描述】:

我是 Rails 新手,在完成 Michael Hartl 的 Ruby on Rails 教程后,我正在尝试扩展一些功能。我首先更改了微博,以便它们收集更多数据。

运行 rspec 测试时,我不知道为什么会出现此错误:

$ bundle exec rspec spec/models/micropost_spec.rb
...............F.....

Failures:

  1) Micropost 
     Failure/Error: it { should be_valid }
       expected valid? to return true, got false
     # ./spec/models/micropost_spec.rb:46:in `block (2 levels) in <top (required)>'

Finished in 1.57 seconds
21 examples, 1 failure

Failed examples:

rspec ./spec/models/micropost_spec.rb:46 # Micropost 

micropost_spec.rb

require 'spec_helper'

describe Micropost do

  let(:user) { FactoryGirl.create(:user) }
  before { @micropost = user.microposts.build(content: "Lorem ipsum", 
    title: "This is a test title", 
    privacy: "1", 
    groups: "This is a test Group", 
    loc1T: "21 Bond St. Toronto, Ontario",
    loc1Lat: "43.654653",
    loc1Lon: "-79.377627",
    loc2T: "21 Bond St. Toronto, Ontario",
    loc2Lat: "43.654653",
    loc2Lon: "-79.377627",
    startTime: "Jan 1, 2000 12:01:01",
    endTime: "Jan 2, 2000 12:01:01",
    imageURL: "http://i.cdn.turner.com/cnn/.e/img/3.0/global/header/hdr-main.gif") 

    puts @micropost.errors.messages
   }

  #before { @micropost = user.microposts.build(title: "This is a test title") }
  #before { @micropost = user.microposts.build(privacy: "1") }
  #before { @micropost = user.microposts.build(groups: "This is a test Group") }
  #before { @micropost = user.microposts.build(loc1T: "21 Bond St. Toronto, Ontario") }
  #before { @micropost = user.microposts.build(loc1Lat: "43.654653") }
  #before { @micropost = user.microposts.build(loc1Lon: "-79.377627") }
  #before { @micropost = user.microposts.build(loc2T: "21 Bond St. Toronto, Ontario") }
  #before { @micropost = user.microposts.build(loc2Lat: "43.654653") }
  #before { @micropost = user.microposts.build(loc2Lon: "-79.377627") }
  #before { @micropost = user.microposts.build(startTime: "Jan 1, 2000 12:01:01") }
  #before { @micropost = user.microposts.build(endTime: "Jan 2, 2000 12:01:01") }
  #before { @micropost = user.microposts.build(imageURL: "http://i.cdn.turner.com/cnn/.e/img/3.0/global/header/hdr-main.gif") }



  subject { @micropost }

  it { should respond_to(:content) }
  it { should respond_to(:user_id) }
  it { should respond_to(:user) }

  it { should respond_to(:title) }
  it { should respond_to(:privacy) }
  it { should respond_to(:groups) }
  it { should respond_to(:loc1T) }
  it { should respond_to(:loc1Lat) }
  it { should respond_to(:loc1Lon) }
  it { should respond_to(:loc2T) }
  it { should respond_to(:loc2Lat) }
  it { should respond_to(:loc2Lon) }
  it { should respond_to(:startTime) }
  it { should respond_to(:endTime) }
  it { should respond_to(:imageURL) }



  its(:user) { should == user }

  it { should be_valid }

  describe "accessible attributes" do
    it "should not allow access to user_id" do
      expect do
        Micropost.new(user_id: user.id)
      end.to raise_error(ActiveModel::MassAssignmentSecurity::Error)
    end    
  end

  describe "when user_id is not present" do
    before { @micropost.user_id = nil }
    it { should_not be_valid }
  end

  describe "with blank content" do
    before { @micropost.content = " " }
    it { should_not be_valid }
  end

  describe "with content that is too long" do
    before { @micropost.content = "a" * 141 }
    it { should_not be_valid }
  end
end

micropost.rb

class Micropost < ActiveRecord::Base
  attr_accessible :content, :title,:privacy,:groups,:loc1T,:loc1Lat,:loc1Lon,:loc2T,:loc2Lat,:loc2Lon,:startTime,:endTime,:imageURL



  belongs_to :user

  validates :user_id, presence: true
  validates :title, presence: true
  validates :privacy, presence: true
  validates :groups, presence: true
  validates :loc1T, presence: true
  validates :loc1Lat, presence: true
  validates :loc1Lon, presence: true
  validates :loc2T, presence: true
  validates :loc2Lat, presence: true
  validates :loc2Lon, presence: true
  validates :startTime, presence: true
  validates :endTime, presence: true
  validates :imageURL, presence: true

  validates :content, presence: true, length: { maximum: 140 }

  default_scope order: 'microposts.created_at DESC'

  def self.from_users_followed_by(user)
    followed_user_ids = "SELECT followed_id FROM relationships
                         WHERE follower_id = :user_id"
    where("user_id IN (#{followed_user_ids}) OR user_id = :user_id", 
          user_id: user.id)
  end
end

factories.rb

    FactoryGirl.define do
  factory :user do
    #name     "Michael Hartl"
    #email    "michael@example.com"
    sequence(:name)  { |n| "Person #{n}" }
    sequence(:email) { |n| "person_#{n}@example.com"} 
    password "foobar"
    password_confirmation "foobar"

    factory :admin do
      admin true
    end
  end

   factory :micropost do
    content "Lorem ipsum"
    title "This is a test title"
    privacy "1"
    groups "This is a test Group"
    loc1T "21 Bond St. Toronto, Ontario"
    loc1Lat "43.654653"
    loc1Lon "-79.377627"
    loc2T "21 Bond St. Toronto, Ontario"
    loc2Lat "43.654653"
    loc2Lon "-79.377627"
    startTime "Jan 1, 2000 12:01:01"
    endTime "Jan 2, 2000 12:01:01"
    imageURL "http://i.cdn.turner.com/cnn/.e/img/3.0/global/header/hdr-main.gif"

    user
  end
end

我不确定您是否还有其他需要我发布的内容。

【问题讨论】:

  • 你能把微博的值放上来看看它的值吗?
  • 如何把微博的值放上去看看它的值?
  • 在一些测试用例中使用 puts 语句,然后运行 ​​bundle exec rspec spec,它会打印出你的微博,比如 ..........microspost: blah.F......
  • 感谢您的帮助,但我是 Rails 新手。我不知道要在哪个文件中添加 puts 语句。我应该在文件中添加什么。谷歌搜索“puts 声明”没有帮助。
  • 您的before 块系列每个都覆盖@micropost,因此它们基本上是相互撤消的。这是我要解决的第一件事。你的工厂有相同的数据,所以你应该使用它。

标签: ruby-on-rails-3 rspec-rails railstutorial.org


【解决方案1】:

这是无效的,因为您的模型对象没有通过您的验证。在您的 before 块中,创建 @micropost 后,添加此行以查看哪些验证失败

puts @micropost.errors.messages

会有一个失败的验证(字段和错误消息)哈希。修复这些,然后您的对象将有效。一些较早的评论者对从哪里开始帮助解决问题提出了建议。

【讨论】:

  • 我这样做了,它通过了。 before { @micropost = user.microposts.build(content: "Lorem ipsum", title: "This is a test title", privacy: "1", groups: "This is a test Group", loc1T: "21 Bond St. Toronto, Ontario", loc1Lat: "43.654653", loc1Lon: "-79.377627", loc2T: "21 Bond St. Toronto, Ontario", loc2Lat: "43.654653", loc2Lon: "-79.377627", startTime: "Jan 1, 2000 12:01:01", endTime: "Jan 2, 2000 12:01:01", imageURL: "http://i.cdn.turner.com/cnn/.e/img/3.0/global/header/hdr-main.gif") puts @micropost.errors.messages }
  • @livi1717 检查 test.log。日志中可能有更多线索说明对象无效的原因。
【解决方案2】:

这消除了错误。

micropost_spec.rb

describe Micropost do

  let(:user) { FactoryGirl.create(:user) }
  before { @micropost = user.microposts.build(content: "Lorem ipsum", 
    title: "This is a test title", 
    privacy: "1", 
    groups: "This is a test Group", 
    loc1T: "21 Bond St. Toronto, Ontario",
    loc1Lat: "43.654653",
    loc1Lon: "-79.377627",
    loc2T: "21 Bond St. Toronto, Ontario",
    loc2Lat: "43.654653",
    loc2Lon: "-79.377627",
    startTime: "Jan 1, 2000 12:01:01",
    endTime: "Jan 2, 2000 12:01:01",
    imageURL: "http://i.cdn.turner.com/cnn/.e/img/3.0/global/header/hdr-main.gif") 

puts @micropost.errors.messages
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2010-11-03
    相关资源
    最近更新 更多