Получение "ActiveRecord:: UnknownAttributeError: неизвестный атрибут: email_confirmation" Ошибка с rspec

Я запускаю эту ошибку при выполнении моих тестов. Я проверил, чтобы все email_confirmation были написаны правильно и (если только я не сумасшедший). Я немного Rails noob, так что это может быть что-то простое.

Модель пользователя

class User < ActiveRecord::Base
  attr_accessible :email, :email_confirmation, :first_name, :last_name,
                  :password, :password_confirmation
  has_secure_password

  before_save { |user| user.email = email.downcase }

  validates :first_name, presence: true, length: { maximum: 25 }
  validates :last_name, presence: true, length: { maximum: 25 }
  VALID_EMAIL_REGEX = /\A[\w+\-.][email protected][a-z\d\-.]+\.[a-z]+\z/i
  validates :email, presence: true, format: { with: VALID_EMAIL_REGEX },
                    uniqueness: { case_sensitive: false }
  validates :email_confirmation, presence: true
  validates :password, presence: true, length: { maximum: 6 }
  validates :password_confirmation, presence: true
end

Тесты Rspec

require 'spec_helper'

describe User do
  before { @user = User.new(email: "[email protected]",
                            first_name: "John", last_name: "Smith",
                            password: "foobar", password_confirmation: "foobar",
                            email_confirmation: "[email protected]") }

  subject { @user }

  it { should respond_to(:first_name) }
  it { should respond_to(:last_name) }
  it { should respond_to(:email) }
  it { should respond_to(:email_confirmation) }
  it { should respond_to(:password_digest) }
  it { should respond_to(:password) }
  it { should respond_to(:password_confirmation) }
  it { should respond_to(:authenticate) }

  it { should be_valid }

  describe "when first name is not present" do
    before { @user.first_name = " " }
    it { should_not be_valid }
  end

  describe "when last name is not present" do
    before { @user.last_name = " " }
    it { should_not be_valid }
  end

  describe "when email is not present" do
    before { @user.email = @user.email_confirmation = " " }
    it { should_not be_valid }
  end

  describe "when password is not present" do
    before { @user.password = @user.password_confirmation = " " }
    it { should_not be_valid }
  end

  describe "when first_name is too long" do
    before { @user.first_name = "a" * 26 }
    it { should_not be_valid }
  end

  describe "when last_name is too long" do
    before { @user.last_name = "a" * 26 }
    it { should_not be_valid }
  end

  describe "when email format is invalid" do
    it "should be invalid" do
      addresses = %w[[email protected],com user_at_foo.org [email protected]
                             [email protected]_baz.com [email protected]+baz.com]
      addresses.each do |invalid_address|
        @user.email = invalid_address
        @user.should_not be_valid
     end      
    end
  end

  describe "when email format is valid" do
    it "should be valid" do
      addresses = %w[[email protected] [email protected] [email protected] [email protected]]
      addresses.each do |valid_address|
        @user.email = valid_address
        @user.should be_valid
      end      
    end
  end

  describe "when email address is already taken" do
    before do
      user_with_same_email = @user.dup
      user_with_same_email.email = @user.email.upcase
      user_with_same_email.save
    end

    it { should_not be_valid }
  end

  describe "when password doesn't match confirmation" do
    before { @user.password_confirmation = "mismatch" }
    it { should_not be_valid }
  end

  describe "when email doesn't match confirmation" do
    before { @user.email_confirmation = "[email protected]" }
    it { should_not be_valid }
  end

  describe "when password confirmation is nil" do
    before { @user.password_confirmation = nil }
    it { should_not be_valid }
  end

  describe "when email confirmation is nil" do
    before { @user.email_confirmation = nil }
    it { should_not be_valid }
  end

  describe "with a password that too short" do
    before { @user.password = @user.password_confirmation = "a" * 5 }
    it { should be_invalid }
  end

  describe "return value of authenticate method" do
    before { @user.save }
    let(:found_user) { User.find_by_email(@user.email) }

    describe "with valid password" do
      it { should == found_user.authenticate(@user.password) }
    end

    describe "with invalid password" do
      let(:user_for_invalid_password) { found_user.authenticate("invalid") }

      it { should_not == user_for_invalid_password }
      specify { user_for_invalid_password.should be_false }
    end
  end
end

schema.rb

ActiveRecord::Schema.define(:version => 20130417021135) do

  create_table "users", :force => true do |t|
    t.string   "first_name"
    t.string   "last_name"
    t.string   "email"
    t.datetime "created_at",      :null => false
    t.datetime "updated_at",      :null => false
    t.string   "password_digest"
  end

  add_index "users", ["email"], :name => "index_users_on_email", :unique => true

end

Ответ 1

Вы получаете UnknownAttributeError, потому что у вас нет столбца в таблице users с именем email_confirmation. По умолчанию ActiveRecord будет искать столбцы БД по имени, аналогичные атрибутам, которые вы используете для построения модели, но эта строка пытается создать пользователя с атрибутом, о котором база данных не знает:

  before { @user = User.new(email: "[email protected]",
                        first_name: "John", last_name: "Smith",
                        password: "foobar", password_confirmation: "foobar",
                        email_confirmation: "[email protected]") }

Действительно ли вы собираетесь сохранить подтверждение по электронной почте в базе данных или просто хотите проверить, соответствует ли он электронной почте, прежде чем сохранять его? Я предполагаю, что последний, и у Rails на самом деле есть встроенная поддержка для этого:

class User < ActiveRecord::Base
  validates :email, :confirmation => true
  validates :email_confirmation, :presence => true
end

Подробнее о Rails Guide to Validations или validates_confirmation_of API. (И вам, вероятно, нужно будет сделать то же самое для :password_confirmation.)

Ответ 2

Я понимаю, что ответ выше правильно обозначен и решает проблему ОП. Но есть еще одна причина для этой ошибки, которая не учитывается в нескольких сообщениях stackoverflow в этом разделе. Эта ошибка может произойти в полиморфном множестве многих, когда вы забудете использовать параметр as: has_many. Например:

class AProfile < ActiveRecord::Base
  has_many :profile_students
  has_many :students, through: :profile_students
end

class BProfile < ActiveRecord::Base
  has_many :profile_students
  has_many :students, through: :profile_students
end

class ProfileStudent < ActiveRecord::Base
  belongs_to :profile, polymorphic: :true
  belongs_to :student
end

class Student < ActiveRecord::Base
  has_many :profile_students
  has_many :aprofiles, through: :profile_students
  has_many :bprofiles, through: :profile_students
end

Это даст вам эту ошибку:

Getting "ActiveRecord::UnknownAttributeError: unknown attribute: profile_id

когда вы пытаетесь сделать следующее:

a = AProfile.new
a.students << Student.new

Решение состоит в том, чтобы добавить опцию: as в AProfile и BProfile:

class AProfile < ActiveRecord::Base
  has_many :profile_students, as: :profile
  has_many :students, through: :profile_students
end

class BProfile < ActiveRecord::Base
  has_many :profile_students, as: :profile
  has_many :students, through: :profile_students
end

Ответ 3

Просто потратив кучу времени на отладку моего собственного экземпляра, я подумал, что буду перекликать с третьей возможностью.

Я правильно выполнил миграцию и проверил ее, проверив мой ActiveRecord в консоли rails. Я много раз пытался воссоздать свой db из схемы, и я много раз пытался перезапустить миграцию, все безрезультатно.

В моем случае проблема заключается в том, что я видел проблему при выполнении моих модульных тестов, а не во время выполнения. Проблема в том, что моя тестовая база данных не синхронизировалась в моем тестировании миграции/отката. Решение было довольно простым. Все, что мне нужно было сделать, это reset тестовая база данных:

rake db:test:prepare

Ответ 4

У меня есть одна и та же ошибка сообщения, и я исправляю порядок параметров как один и тот же порядок определения столбцов в базе данных:

КОНТРОЛЛЕР

def create
    worktime = Worktime.create(name: params[:name], workhours: params[:workhours], organization: @organization, workdays: params[:workdays])

    render json: worktime
end

БАЗА ДАННЫХ

Table: worktimes
Columns:
id  int(11) AI PK
name    varchar(255)
workhours   text
organization_id int(11)
workdays    text