PostgreSQL
 sql >> Base de Dados >  >> RDS >> PostgreSQL

Rails/postgres, 'chaves estrangeiras' armazenadas em array para criar associação 1-many


Você não poderá tornar o Rails ciente desse array e usá-lo para associações.

Mas se você quiser uma pesquisa/filtragem mais rápida de Tarefas atribuídas a usuários, você pode manter uma matriz de IDs de Usuário no objeto Tarefa. Caso contrário, você teria que fazer um JOIN para encontrar todas as tarefas atribuídas a Alice, em sua tabela de associação padrão.

Portanto, a solução é manter a tabela de associação, mas também duplicar o ID do usuário atribuído no objeto Tarefa e usar essa lista de IDs para pesquisa/filtragem mais rápida.

Você precisará se conectar ao after_create e after_destroy ciclo de vida para os objetos atribuídos e insira novos IDs de atribuídos na matriz de registro de tarefas. E então, quando um cessionário é removido de uma tarefa, atualize a matriz para remover o ID.

Veja a documentação do Postgres para todos os operadores de Array:

Algo assim:
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