ActiveRecord :: StatementInvalid для двух моделей пространства имен - PullRequest
0 голосов
/ 10 февраля 2019

У меня есть две модели в одном и том же пространстве имен, которые имеют отношение habtm.

class Resource::Item < ApplicationRecord
    has_and_belongs_to_many :resource_sets, foreign_key: 'resource_item_id', class_name: 'Resource::Set', table_name: 'resource_items_sets'
end

class Resource::Set < ApplicationRecord
    has_and_belongs_to_many :resource_items, foreign_key: 'resource_set_id', class_name: 'Resource::Item', table_name: 'resource_items_sets'
end

Миграция была сгенерирована с помощью rails g migration CreateJoinTableResourceItemsResourceSets resource_item resource_set

class CreateJoinTableResourceItemsResourceSets < ActiveRecord::Migration[5.2]
    def change
        create_join_table :resource_items, :resource_sets do |t|
            # t.index [:resource_item_id, :resource_set_id]
            # t.index [:resource_set_id, :resource_item_id]
        end
    end
end

Пока все выглядит отлично,Таблица resource_items_sets создается с двумя столбцами resource_item_id и resource_set_id.

Это схема

create_table "resource_items_sets", id: false, force: :cascade do |t|
    t.bigint "resource_item_id", null: false
    t.bigint "resource_set_id", null: false
end

После создания resource_item я получаю следующее:ожидается.

pry(main)> Resource::Item.first.resource_sets
=> #<Resource::Set::ActiveRecord_Associations_CollectionProxy:0x3fdd08748004>

Но выполнение следующих действий приводит к ошибке.Я ожидал 0.

pry(main)> Resource::Item.first.resource_sets.count
ActiveRecord::StatementInvalid: PG::UndefinedColumn: ERROR:  column resource_items_sets.set_id does not exist
LINE 1: ...N "resource_items_sets" ON "resource_sets"."id" = "resource_...
                                                         ^
: SELECT "resource_sets"."id" FROM "resource_sets" INNER JOIN "resource_items_sets" ON "resource_sets"."id" = "resource_items_sets"."set_id" WHERE "resource_items_sets"."resource_item_id" = $1 ORDER BY "resource_sets"."name" ASC
from /Users/username/.rvm/gems/ruby-2.6.0/gems/activerecord-5.2.2/lib/active_record/connection_adapters/postgresql_adapter.rb:677:in `async_prepare'
Caused by PG::UndefinedColumn: ERROR:  column resource_items_sets.set_id does not exist
LINE 1: ...N "resource_items_sets" ON "resource_sets"."id" = "resource_...

Откуда взялся set_id, когда resource_set_id было объявлено везде?Как я могу решить эту проблему? Я хочу сохранить пространства имен для обоих, так как я мог бы в конечном итоге создавать элементы и устанавливать для них больше пространств имен.

Большое спасибо, ребята!

1 Ответ

0 голосов
/ 11 февраля 2019

Вам необходимо установить внешние ключи с обеих сторон таблицы объединения, потому что они не могут быть выведены в вашем случае.

В этом случае правильные вызовы has_and_belongs_to_many должны выглядеть следующим образом:

  has_and_belongs_to_many :resource_items,
                          foreign_key: 'resource_set_id',
                          association_foreign_key: 'resource_item_id',
                          class_name: 'Resource::Item',
                          join_table: 'resource_items_sets'
end

и

class Resource::Item < ApplicationRecord
  has_and_belongs_to_many :resource_sets,
                          foreign_key: 'resource_item_id',
                          association_foreign_key: 'resource_set_id',
                          class_name: 'Resource::Set',
                          join_table: 'resource_items_sets'
end

Обратите внимание на добавленную опцию association_foreign_key, указывающую другой внешний ключ в таблице соединения.

...