Rails - проверка ошибок приборов NoMethodError: undefined метод `type 'для nil: NilClass
У меня есть проблемы с запущенными тестами, которые используют приборы с ассоциациями между моделями.
Вот ошибка, которую я получаю, как только я запускаю rake test
:
ERROR["test_truth", SevenPortfolioTest, 0.005154775]
test_truth#SevenPortfolioTest (0.01s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_destroy_item_video", SevenPortfolio::ItemVideosControllerTest, 0.008887804]
test_should_destroy_item_video#SevenPortfolio::ItemVideosControllerTest (0.01s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_get_index", SevenPortfolio::ItemVideosControllerTest, 0.011364416]
test_should_get_index#SevenPortfolio::ItemVideosControllerTest (0.01s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_create_item_video", SevenPortfolio::ItemVideosControllerTest, 0.014584266]
test_should_create_item_video#SevenPortfolio::ItemVideosControllerTest (0.01s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_get_new", SevenPortfolio::ItemVideosControllerTest, 0.017282812]
test_should_get_new#SevenPortfolio::ItemVideosControllerTest (0.02s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_update_item_video", SevenPortfolio::ItemVideosControllerTest, 0.019729858]
test_should_update_item_video#SevenPortfolio::ItemVideosControllerTest (0.02s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_get_edit", SevenPortfolio::ItemVideosControllerTest, 0.022365633]
test_should_get_edit#SevenPortfolio::ItemVideosControllerTest (0.02s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_destroy_item", SevenPortfolio::ItemsControllerTest, 0.025860205]
test_should_destroy_item#SevenPortfolio::ItemsControllerTest (0.03s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_get_edit", SevenPortfolio::ItemsControllerTest, 0.030867796]
test_should_get_edit#SevenPortfolio::ItemsControllerTest (0.03s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_show_item", SevenPortfolio::ItemsControllerTest, 0.036687105]
test_should_show_item#SevenPortfolio::ItemsControllerTest (0.04s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_update_item", SevenPortfolio::ItemsControllerTest, 0.040130774]
test_should_update_item#SevenPortfolio::ItemsControllerTest (0.04s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_create_item_with_video", SevenPortfolio::ItemsControllerTest, 0.042776553]
test_should_create_item_with_video#SevenPortfolio::ItemsControllerTest (0.04s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_get_index", SevenPortfolio::ItemsControllerTest, 0.045301694]
test_should_get_index#SevenPortfolio::ItemsControllerTest (0.05s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_get_new", SevenPortfolio::ItemsControllerTest, 0.048094189]
test_should_get_new#SevenPortfolio::ItemsControllerTest (0.05s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
ERROR["test_should_create_item_with_gallery", SevenPortfolio::ItemsControllerTest, 0.051019403]
test_should_create_item_with_gallery#SevenPortfolio::ItemsControllerTest (0.05s)
NoMethodError: NoMethodError: undefined method `type' for nil:NilClass
Я тестирую двигатель рельсов. Здесь мои светильники:
# test/fixtures/seven_portfolio/items.yml
item_one:
description: item one description
finished_at: <%= 7.days.ago %>
is_featured: true
item_type: 0
item_two:
description: item two description
finished_at: <%= 6.days.ago %>
is_featured: false
item_type: 1
# test/fixtures/seven_gallery/galleries.yml
gallery_one:
title: seven_gallery_gallery_one_title
item: item_one
# test/fixtures/seven_gallery/photos.yml
photo_one:
caption: photo_one_caption
gallery: gallery_one
# test/fixtures/seven_portfolio/item_videos.yml
one:
item: item_one
Когда, например, я удаляю строки item: item_one
и подобные ассоциации. Тест хорошо работает.
-
ИЗМЕНИТЬ
Здесь backtrace
Command failed with status (1): [ruby -I"lib:lib:test" "/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/rake_test_loader.rb" "test/**/*_test.rb" ]
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/testtask.rb:108:in `block (3 levels) in define'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/file_utils.rb:57:in `call'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/file_utils.rb:57:in `sh'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/file_utils_ext.rb:37:in `sh'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/file_utils.rb:96:in `ruby'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/file_utils_ext.rb:37:in `ruby'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/testtask.rb:104:in `block (2 levels) in define'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/file_utils_ext.rb:58:in `verbose'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/testtask.rb:100:in `block in define'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:240:in `call'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:240:in `block in execute'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:235:in `each'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:235:in `execute'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:179:in `block in invoke_with_call_chain'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/monitor.rb:211:in `mon_synchronize'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:172:in `invoke_with_call_chain'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/task.rb:165:in `invoke'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:150:in `invoke_task'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:106:in `block (2 levels) in top_level'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:106:in `each'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:106:in `block in top_level'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:115:in `run_with_threads'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:100:in `top_level'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:78:in `block in run'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:176:in `standard_exception_handling'
/home/rafael/.rbenv/versions/2.2.0/lib/ruby/2.2.0/rake/application.rb:75:in `run'
/home/rafael/.rbenv/versions/2.2.0/bin/rake:33:in `<main>'
ИЗМЕНИТЬ
Здесь Item
class:
module SevenPortfolio
class Item < ActiveRecord::Base
has_one :item_video, class_name:'SevenPortfolio::ItemVideo', foreign_key: "seven_portfolio_item_id"
has_one :item_gallery, class_name: 'SevenGallery::Gallery', foreign_key: "seven_portfolio_item_id"
accepts_nested_attributes_for :item_video, :item_gallery
before_save :process_type
def process_type
self.build_item_video if self.item_type == 0
self.build_item_gallery if self.item_type == 1
end
def type_content
self.item_gallery if self.item_type == 0
self.item_video if self.item_type == 1
end
end
end
И Gallery
класс:
class SevenGallery::Gallery < ActiveRecord::Base
include SevenGallery::Concerns::Gallery
belongs_to :item, :class_name => 'SevenPortfolio::Item'
end
ИЗМЕНИТЬ
Здесь schema.rb
файл:
ActiveRecord::Schema.define(version: 20150414145951) do
create_table "seven_gallery_galleries", force: :cascade do |t|
t.string "title", limit: 255
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.integer "seven_portfolio_item_id", limit: 4
end
add_index "seven_gallery_galleries", ["seven_portfolio_item_id"], name: "index_seven_gallery_galleries_on_seven_portfolio_item_id", using: :btree
create_table "seven_gallery_photos", force: :cascade do |t|
t.string "caption", limit: 255
t.string "image", limit: 255
t.integer "seven_gallery_gallery_id", limit: 4
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.boolean "is_new", limit: 1, default: true
t.boolean "is_featured", limit: 1, default: false
t.text "description", limit: 65535
t.string "alt", limit: 255
t.integer "position", limit: 4, default: 0
end
add_index "seven_gallery_photos", ["seven_gallery_gallery_id"], name: "index_seven_gallery_photos_on_seven_gallery_gallery_id", using: :btree
create_table "seven_portfolio_item_videos", force: :cascade do |t|
t.string "url", limit: 255
t.text "description", limit: 65535
t.string "title", limit: 255
t.integer "seven_portfolio_item_id", limit: 4
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
end
add_index "seven_portfolio_item_videos", ["seven_portfolio_item_id"], name: "index_seven_portfolio_item_videos_on_seven_portfolio_item_id", using: :btree
create_table "seven_portfolio_items", force: :cascade do |t|
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.text "description", limit: 65535
t.date "finished_at"
t.boolean "is_featured", limit: 1, default: false
t.integer "item_type", limit: 4, default: 0
end
add_foreign_key "seven_gallery_galleries", "seven_portfolio_items", on_delete: :cascade
add_foreign_key "seven_gallery_photos", "seven_gallery_galleries", on_delete: :cascade
add_foreign_key "seven_portfolio_item_videos", "seven_portfolio_items", on_delete: :cascade
end
Ответы
Ответ 1
Я ударил те же самые ошибки после миграции и отката назад несколько раз в моей среде dev. Это как-то связано с столбцом "_type".
Единственное, что исправило это для меня, это создание/создание тестовой базы данных:
bundle exec rake db:reset RAILS_ENV=test
Будьте уверены, что ваш schema.rb
находится в хорошей форме, прежде чем делать это.
Ответ 2
Эта ошибка возникает, когда в вашей модели вы создаете связь, у которой нет поля в db. Пример:
class Contact < ActiveRecord::Base
belongs_to account
end
Но в вашей таблице контакты:
create_table "contacts", force: :cascade do |t|
t.string "fname"
t.string "lname"
t.string "title"
t.string "description"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.boolean "primary"
t.integer "contactable_id"
t.integer "contactable_type"
end
См. там нет поля account_id.
В моем случае я забыл обновить модель belongs_to
, чтобы быть полиморфной:
class Contact < ActiveRecord::Base
belongs_to :contactable, polymorphic: true
end
class Account < ActiveRecord::Base
has_many :contacts, as: :contactable
end
Ответ 3
Разве это не проблема с дубликатом/подобной проблемой?
Загрузка рельсов в конкретном порядке при тестировании
Что может случиться, так это то, что ваши светильники не загружаются в правильном порядке, поэтому, когда ссылка на объект указана, он появляется как ноль.
Из ответа другого ответа вы можете попробовать
Вот как вы исправляете свою проблему:
-
Войдите в Postgresql со своим суперпользователем по умолчанию (мой - postgres)
Выполните следующую команду:
-
ALTER ROLE yourtestdbuser WITH SUPERUSER;
-
Наслаждайтесь правильно работающими светильниками.
Если вы используете mysql, это, вероятно, аналогичная проблема.
Ответ 4
TL; DR: Люди обсуждают этот вопрос по разным причинам. Моя проблема/решение описано в разделе "Исправление ошибки". Я надеюсь, что это расширенное объяснение поможет вам узнать, почему вы испытываете эту загадочную ошибку, и сэкономит ваше время на ее устранение.
Как диагностировать эту проблему
У меня была почти та же ошибка при запуске моих тестов: NoMethodError: undefined method 'id' for nil:NilClass
или NoMethodError: undefined method 'each' for nil:NilClass
. Там не было номеров строк для ссылки, и ответ Лукаса о восстановлении иногда работал, а затем перестал работать; Позже я узнаю, что это из-за порядка кэширования и загрузки файлов фикстуры.
Нахождение где ошибка брошена
Моя ошибка была в том, как мои приборы были настроены. Я нашел это, запустив свои тесты с полной обратной трассировкой, описанной в fooobar.com/info/475164/...:
BACKTRACE=YES rake test test/controllers/document_submissions_controller_test.rb
Это указывало на файлы gem в ActiveSupport, которые выдавали ошибку. Вот мой полный журнал для одного теста:
![enter image description here]()
Так что, похоже, ошибка в загрузке моих приборов, так как ошибка происходит из active_record/fixtures.rb:758
(см. active_record/fixtures.rb:758
строку выше). Тем не менее, у меня есть большое количество приборов в моем приложении, мне нужно было сузить, что, в частности, вызывает проблему.
Установка точки останова
Я открыл active_record/fixtures.rb:758
в моем редакторе кода с помощью команды ниже:
code /Users/mattmcinnis/.rvm/gems/ruby-2.3.0/gems/activerecord-4.2.0/lib/active_record/fixtures.rb
Примечание: я использую VSCode с командой code
, но подойдет любой редактор кода.
В строке 758, где происходит ошибка, я увидел этот код:
FixtureSet::File.open(file) do |fh|
fh.each do |fixture_name, row| # line 758
fixtures[fixture_name] = ActiveRecord::Fixture.new(row, model_class)
end
end
... который я завернул в спасательный блок и изменил:
FixtureSet::File.open(file) do |fh|
begin
fh.each do |fixture_name, row|
fixtures[fixture_name] = ActiveRecord::Fixture.new(row, model_class)
end
rescue
debugger
end
end
ПРИМЕЧАНИЕ. Обязательно отмените все изменения в этом файле, когда закончите!
Нахождение файла проблемы
Меня интересует, какой файл, записанный в цикле выше в переменной fh
, вызывал ошибку. Запустив мой тестовый набор снова, я достиг своей точки останова и нашел имя файла, которое вызывало проблему:
> (byebug) fh
=> ... @file="permissions.yml"
Исправление ошибки
В моем файле permissions.yml
мой прибор ссылается на связанную модель путем жесткого кодирования 'id' вместо ссылки по имени:
Ex. permissions.yml
permission1:
...
user_id: <%= User.first.id %> # Bad!
Закрепление последней строки на что - то подобное, user: user1
обратиться к user1
арматуре в users.yml
решить эту проблему.
Надеюсь, это кому-нибудь поможет. Я потерял полдня на этом, и мне было очень трудно диагностировать.