Rails/postgres, "внешние ключи", хранящиеся в массиве, для создания ассоциации 1-много

Можно ли использовать массивы postgres для создания ассоциации "один ко многим /has _many" в рельсах (4)? Я знаю, что массив типа внешнего ключа невозможен.

Пример. Задача имеет несколько правопреемников. Традиционно я решал это, используя таблицу ассоциации: tasks- > assignees- > users. Используя массивы, это не было бы необходимо, так как можно было бы хранить несколько "внешних ключей".

Следующий запрос может быть использован для получения всех назначенных мне задач:

select * from tasks where ? IN tasks.assignees

Ответ 1

Вы не сможете сделать Rails осведомленным об этом массиве и использовать его для ассоциаций.

Но если вы хотите более быстрый поиск/фильтрацию заданий, назначенных пользователям, вы можете сохранить массив идентификаторов пользователей в объекте Task. В противном случае вам нужно будет сделать JOIN, чтобы найти все задачи, назначенные Алисе, в вашей стандартной таблице ассоциаций.

Таким образом, решение состоит в том, чтобы сохранить таблицу ассоциаций, а также дублировать идентификатор пользователя-получателя в объект Task и использовать этот список идентификаторов для более быстрого поиска/фильтрации.

Вам нужно подключиться к жизненному циклу after_create и after_destroy для объектов-получателей и вставить новые идентификаторы-получатели в массив записей задач. А затем, когда asignee удаляется из обновления задачи, массив удаляет идентификатор.

Смотрите документы Postgres для всех операторов массива:

Что-то вроде этого:

class Task < ActiveRecord::Base
    has_many :assignees, :dependent => :destroy
end

class Asignee < ActiveRecord::Base

    belongs_to :task
    after_create :insert_task_assignee
    after_destroy :remove_task_assignee

    # assumes that there is a column called assignee_id
    # that contains the User ID of the assigned person

    private

    def insert_task_assignee
        # TODO: check for duplicates here - before we naively push it on?
        task.assignee_list = task.assignee_list.push(assignee_id)
        task.assignee_list.save
    end

    def remove_task_assignee
        id_list = task.assignee_list
        id_list.reject! { |candidate_id| candidate_id == assignee_id }
        task.assignee_list = id_list
        task.assignee_list.save
    end

end

# find all tasks that have been assigned Alice & Bob
# this will require the `postgres_ext` gem for Ruby / Postgres array searching
tasks = Task.where.contains(:assignee_list => [alice.id, bob.id]).all