PostgreSQL
 sql >> Database >  >> RDS >> PostgreSQL

Rails/postgres, 'chiavi straniere' memorizzate nell'array per creare un'associazione 1-molti

Non sarai in grado di rendere Rails consapevole di questo array e usarlo per le associazioni.

Ma se desideri una ricerca/filtraggio più rapida delle attività assegnate agli utenti, puoi mantenere una matrice di ID utente nell'oggetto Attività. Altrimenti, dovresti fare un JOIN per trovare tutti i compiti assegnati ad Alice, nella tua tabella di associazione standard.

Quindi la soluzione è mantenere la tabella di associazione ma anche duplicare l'ID utente assegnatario nell'oggetto Attività e utilizzare quell'elenco di ID per una ricerca/filtraggio più veloce.

Dovrai collegarti a after_create e after_destroy ciclo di vita per gli oggetti assegnatario e inserire nuovi ID assegnatario nell'array record attività. E poi, quando un assegnatario viene rimosso da un'attività, aggiorna l'array per rimuovere l'ID.

Vedi i documenti di Postgres per tutti gli operatori Array:

Qualcosa del genere:

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