PostgreSQL
 sql >> Base de données >  >> RDS >> PostgreSQL

Rails/postgres, 'clés étrangères' stockées dans un tableau pour créer une association 1-plusieurs

Vous ne pourrez pas informer Rails de ce tableau et l'utiliser pour les associations.

Mais si vous voulez une recherche/filtrage plus rapide des tâches attribuées aux utilisateurs, vous pouvez conserver un tableau d'ID utilisateur dans l'objet Tâche. Sinon, vous devrez faire un JOIN pour trouver toutes les tâches assignées à Alice, dans votre table d'association standard.

La solution consiste donc à conserver la table d'association, mais également à dupliquer l'ID utilisateur du destinataire dans l'objet Tâche et à utiliser cette liste d'ID pour une recherche/filtrage plus rapide.

Vous devrez vous connecter au after_create et after_destroy cycle de vie des objets de destinataire et insérez de nouveaux ID de destinataire dans le tableau d'enregistrements de tâche. Et puis, lorsqu'un destinataire est supprimé d'une tâche, mettez à jour le tableau pour supprimer l'ID.

Voir les docs Postgres pour tous les opérateurs Array :

Quelque chose comme ça :

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