Associating with Ruby on Rails
What is Association
all about?
Accessing info from Database B via Database A.
And vice versa.
A.B.attribute
and/or
B.A.attribute
Types of Association
One-to-One:
The simplest form of Association, where one instance of a Resource "has one" of the other, and that instance of the other resource "belongs to" one of the first
One-to-Many:
Where one instance of a Resource "has many" of the other, and those instances of the other resource only "belong(s) to" that instance of the first.
Many-to-Many:
The hard one! Where instances of a Resource can "have (has) many" instances of the other Resource, and many instances of that Resource can "have (has) many" right back. But this type of Association will require a 3rd DB: the Join Table!
One-to-one Association
Let's create a one-to-one association, between two databases: Driver & Vehicle
$ rails g scaffold Driver name:string
$ rails g scaffold Vehicle make:string model:string driver_id:integer
Notice the driver_id attribute of the Vehicle!
This is called the "Foreign Key" (FK). It relates to the ID number of a Driver in its own database.
Warning! If you're using any other name for your first DB than "Driver", make sure the _id attribute in Vehicle matches!
(e.g., pilot_id, owner_id, etc.)
Association Set-up, steps 2 & 3
The next two steps to setting up an association are defining the relationship in the respective models.
driver.rb
vehicle.rb
class Vehicle < ActiveRecord::Base
belongs_to :driver
end
class Driver < ActiveRecord::Base
has_one :vehicle
end
Drivers have one Vehicle
Vehicles belong to one Driver
And now we're associated!
That's really all it takes to do a one-to-one association!
The trick now is to make the calls between the databases...
who's driving this thing anyway?
In our Vehicle _form.html.erb, we want to pull in a list of our Drivers, to assign their IDs to the new (or updated) Vehicle.
First, we need to pull in the Drivers data in the vehicle_controller.rb:
def new
@vehicle = Vehicle.new
@drivers = Driver.all
end
def edit
@drivers = Driver.all
end
who's driving this thing anyway?
Now we can create a dropdown menu to select Drivers from, by name!
In vehicles/_form.html.erb:
<div class="field">
<%= f.label :make %><br>
<%= f.text_field :make %>
</div>
<div class="field">
<%= f.label :model %><br>
<%= f.text_field :model %>
</div>
<div class="field">
<%= f.label :driver_id %><br>
<!-- The line below is our new bit of code: -->
<%= f.select(:driver_id, @drivers.collect {|d| [ d.name, d.id ] }, {:include_blank =>
'Please select a item'}) %>
</div>
<div class="actions">
<%= f.submit %>
</div>
choose your driver
Now we have a nice dropdown selection tool:
We see the name, but it's the ID that's passed!
show your associations!
Let's edit the display pages (index & show) of the two resources to visualize the association.
Start with the show pages:
vehicles/show.html.erb
drivers/show.html.erb
<p id="notice"><%= notice %></p>
<p>
<strong>Make:</strong>
<%= @vehicle.make %>
</p>
<p>
<strong>Model:</strong>
<%= @vehicle.model %>
</p>
<!-- We'll display the Driver's name instead of the ID-->
<!-- First, set up a safeguard against nil -->
<% if @vehicle.driver != nil %>
<p>
<strong>Driver:</strong>
<!-- display the Driver's name thru assoc! -->
<%= @vehicle.driver.name %>
</p>
<% end %>
<%= link_to 'Edit', edit_vehicle_path(@vehicle) %> |
<%= link_to 'Back', vehicles_path %>
<p id="notice"><%= notice %></p>
<p>
<strong>Name:</strong>
<%= @driver.name %>
</p>
<p>
<strong>City:</strong>
<%= @driver.city %>
</p>
<!-- We'll display the Driver's Vehicle -->
<!-- First, set up a safeguard against nil -->
<% if @driver.vehicle != nil %>
<p>
<strong>Vehicle:</strong>
<!-- display the Vehicle's make & model
thru assoc! -->
<%= "#{@driver.vehicle.make}
#{@driver.vehicle.model}" %>
</p>
<% end %>
<%= link_to 'Edit', edit_driver_path(@driver) %> |
<%= link_to 'Back', drivers_path %>
show your associations!
Now we can edit the index pages:
(I'm showing just the tables from these pages)
vehicles/index.html.erb
drivers/index.html.erb
<table>
<thead>
<tr>
<th>Make</th>
<th>Model</th>
<!-- We'll display the Driver's name instead of the ID-->
<th>Driver</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @vehicles.each do |vehicle| %>
<tr>
<td><%= vehicle.make %></td>
<td><%= vehicle.model %></td>
<td>
<!-- set up a safeguard against nil -->
<% if vehicle.driver != nil %>
<!-- display the Driver's name thru association! -->
<%= vehicle.driver.name %>
<% end %>
</td>
<td><%= link_to 'Show', vehicle %></td>
<td><%= link_to 'Edit', edit_vehicle_path(vehicle) %></td>
<td><%= link_to 'Destroy', vehicle, method: :delete,
data: { confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
<table>
<thead>
<tr>
<th>Name</th>
<th>City</th>
<!-- We'll display the Driver's Vehicle -->
<th>Vehicle</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @drivers.each do |driver| %>
<tr>
<td><%= driver.name %></td>
<td><%= driver.city %></td>
<td>
<!-- set up a safeguard against nil -->
<% if driver.vehicle != nil %>
<!-- display the Vehicle's make & model thru assoc! -->
<%= "#{driver.vehicle.make} #{driver.vehicle.model}" %>
<% end %>
</td>
<td><%= link_to 'Show', driver %></td>
<td><%= link_to 'Edit', edit_driver_path(driver) %></td>
<td><%= link_to 'Destroy', driver, method: :delete,
data: { confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
</table>
I can see your associations!
One-to-many association
A One-to-Many Association is very similar to the One-to-One.
There's a just a slight change of verbiage.
But we'll start off the same.
Let's associate States with Cities...
We still have one Resource taking the Foreign Key (ID) of the other:
state_id
$ rails g scaffold State name:string
$ rails g scaffold City name:string state_id:integer
"One-to-many" set-up, Steps 2 & 3
Once again we will define the relationship
between the resources in the respective models.
state.rb
city.rb
class City < ActiveRecord::Base
belongs_to :state
end
class State < ActiveRecord::Base
has_many :cities
end
State has many Cities
(instead of has_one)
A City belongs to one State
(same as 1-to-1 assoc.)
Selecting States
Let's set up the State selection in the City
form just as we did with Driver/Vehicle.
cities/_form.html.erb
cities_controller.rb
def new
@city = City.new
@states = State.all
end
def edit
@states = State.all
end
<div class="field">
<%= f.label :name %><br>
<%= f.text_field :name %>
</div>
<div class="field">
<%= f.label :state_id %><br>
<!-- Copy the select tag from the
Vehicles form and adjust it to
fit our needs here... -->
<%= f.select(:state_id,
@states.collect {|s| [ s.name, s.id ] },
{:include_blank => 'Please select
an item'}) %>
</div>
<div class="actions">
<%= f.submit %>
</div>
Choose Your state
Voila! A nice dropdown menu to choose our city's state!
Display your States!
Showing the City, State association will be about the same as the Vehicle/Driver association.
cities/index.html.erb
cities/show.html.erb
<table>
<thead>
<tr>
<th>Name</th>
<th>State</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @cities.each do |city| %>
<tr>
<td><%= city.name %></td>
<!-- we only need to switch this line
from city.name_id to: -->
<td><%= city.state.name %></td>
<td><%= link_to 'Show', city %></td>
<td><%= link_to 'Edit', edit_city_path(city) %></td>
<td><%= link_to 'Destroy', city, method: :delete, data:
{ confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
<p id="notice"><%= notice %></p>
<!-- On this page, all we have to do is
display the city name and the state name,
preferably side-by-side, with a comma between
as is the preferred style -->
<p>
<strong>Welcome To...</strong>
<%= @city.name %>, <%= @city.state.name %>
</p>
<%= link_to 'Edit', edit_city_path(@city) %> |
<%= link_to 'Back', cities_path %>
I'm from the great state of...
localhost:3000/cities
localhost:3000/cities/1
What cities are in that state?
Showing the State/Cities association will be a bit trickier.
The cities associated with a State are collect in an Array.
states/show.html.erb
<p id="notice"><%= notice %></p>
<p>
<strong>The Great State of... </strong>
<%= @state.name %>
</p>
<p>
<strong>Major Cities:</strong>
<ul>
<!-- We'll loop through the array of cities
and present them as an unordered list -->
<% @state.cities.each do |city| %>
<li><%= city.name %></li>
<% end %>
</ul>
</p>
<%= link_to 'Edit', edit_state_path(@state) %> |
<%= link_to 'Back', states_path %>
Cities in our state
localhost:3000/states/1
Displaying cities in the state INdex
Once again, we'll have to loop through an array of cities, but this time let's present them in a one-line list...
states/index.html.erb
<table>
<thead>
<tr>
<th>Name</th>
<!-- We add a header for the cities column -->
<th>Cities</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @states.each do |state| %>
<tr>
<td><%= state.name %></td>
<!-- We add a table cell that will display the cities within the state-->
<td>
<!-- In here we'll loop through the array of cities, printing them out with a comma between
(the last city in the array will not have a comma after it) -->
<% state.cities.each_with_index do |city, index| %>
<% if index < (state.cities.length - 1) %>
<%= city.name %>,
<% else %>
<%= city.name %>
<% end %>
<% end %>
</td>
<td><%= link_to 'Show', state %></td>
<td><%= link_to 'Edit', edit_state_path(state) %></td>
<td><%= link_to 'Destroy', state, method: :delete, data: { confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
Cities in our State, Index-style
localhost:3000/states
Many-to-many Association
The Many-to-Many association is the trickiest of the bunch.
Neither of the two main scaffolds will include an FK attribute.
A third scaffold will be needed, and it will only have FK attributes.
This is the Join Table.
We'll explore Many-to-Many using a Doctor/Patient relationship, with Appointment being the JoinTable.
$ rails g scaffold Doctor name:string
$ rails g scaffold Patient name:string
$ rails g scaffold Appointment doctor_id:integer patient_id:integer
Setting up the Models
In the Many-to-Many case, both Doctors and Patients have many appointments.
But they also have many of each other, but only through Appointments.
Finally, Appointments belong to both Doctors & Patients.
We'll start with...
appointment.rb
class Appointment < ActiveRecord::Base
belongs_to :doctor
belongs_to :patient
end
Setting up the models
Now let's see the other two models, and how exactly this through relationship takes shape.
doctor.rb
patient.rb
class Patient < ActiveRecord::Base
has_many :appointments
has_many :doctors, :through => :appointments
end
class Doctor < ActiveRecord::Base
has_many :appointments
has_many :patients, :through => :appointments
end
Wait...that's it?
As before, setting up the Association is kinda easy. Include a FK here, write some code in a model, and PRESTO!, you got yourself some Associations.
The tricky park comes when you want to display the results of those Associations.
Get yourself some data
First, let create several Doctors and Patients, so we have some data to work with.
localhost:3000/doctors
localhost:3000/patients
Making Appointments
Before we start creating appointments, to associate our Doctors & Patients, let's set it up so that we choose from Doctor & Patient names, rather than having to supply ID numbers.
def new
@appointment = Appointment.new
@doctors = Doctor.all
@patients = Patient.all
end
def edit
@doctors = Doctor.all
@patients = Patient.all
end
<div class="field">
<%= f.label :doctor_id %><br>
<!-- Remember the select tags we used in the
association examples? Let's use that again,
twice this time! - adjusting the wording to
fit our purposes here. -->
<%= f.select(:doctor_id, @doctors.collect
{|d| [ d.name, d.id ] }, {:include_blank =>
'Please select a item'}) %>
</div>
<div class="field">
<%= f.label :patient_id %><br>
<%= f.select(:patient_id, @patients.collect
{|p| [ p.name, p.id ] }, {:include_blank =>
'Please select a item'}) %>
</div>
<div class="actions">
<%= f.submit %>
</div>
appointments_controller.rb
appointments/_form.html.erb
making appointment is fun!
Displaying Appointments
Configuring the Appointments pages to show Doctor and Patients names is the easiest of the three.
appointments/index.html.erb
<table>
<thead>
<tr>
<th>Doctor</th>
<th>Patient</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @appointments.each do |appointment| %>
<tr>
<td>
<%= appointment.doctor.name %>
</td>
<td><%= appointment.patient.name %></td>
<td><%= link_to 'Show', appointment %></td>
<td><%= link_to 'Edit', edit_appointment_path(appointment) %></td>
<td><%= link_to 'Destroy', appointment, method: :delete, data: { confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
All the appointments!
It might be nice to add appointment date and time attribute(s) to this Resource. Just sayin'.
Doctors & their Patients: The code
As taken separately, the Doctor/Patients and Patient/Doctors relationships are essentially One-to-Many relationships. So we'll be needing to loop through an array.
<table>
<thead>
<tr>
<th>Name</th>
<th>Patients</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @doctors.each do |doctor| %>
<tr>
<td><%= doctor.name %></td>
<td>
<% doctor.patients.each_with_index do |pat, index| %>
<% if index < (doctor.patients.length - 1) %>
<%= pat.name %>,
<% else %>
<%= " #{pat.name}" %>
<% end %>
<% end %>
</td>
<td><%= link_to 'Show', doctor %></td>
<td><%= link_to 'Edit', edit_doctor_path(doctor) %></td>
<td><%= link_to 'Destroy', doctor, method: :delete, data:
{ confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
<p id="notice"><%= notice %></p>
<p>
<strong>Name:</strong>
<%= @doctor.name %>
</p>
<p>
<strong>Patients:</strong>
<ul>
<!-- We'll loop through the array of patients
and present them as an unordered list -->
<% @doctor.patients.each do |patient| %>
<li><%= patient.name %></li>
<% end %>
</ul>
</p>
<%= link_to 'Edit', edit_doctor_path(@doctor) %> |
<%= link_to 'Back', doctors_path %>
doctors/index.html.erb
doctors/show.html.erb
Doctors & their Patients: In the Browser
localhost:3000/doctors
localhost:3000/doctors/1
Patients & their Doctors: The code
<table>
<thead>
<tr>
<th>Name</th>
<th>Doctors</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @patients.each do |patient| %>
<tr>
<td><%= patient.name %></td>
<td>
<% patient.doctors.each_with_index do |doc, index| %>
<% if index < (patient.doctors.length - 1) %>
<%= doc.name %>,
<% else %>
<%= " #{doc.name}" %>
<% end %>
<% end %>
</td>
<td><%= link_to 'Show', patient %></td>
<td><%= link_to 'Edit', edit_patient_path(patient) %></td>
<td><%= link_to 'Destroy', patient, method: :delete, data:
{ confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
<p id="notice"><%= notice %></p>
<p>
<strong>Name:</strong>
<%= @patient.name %>
</p>
<p>
<strong>Doctors:</strong>
<ul>
<!-- We'll loop through the array of doctors
and present them as an unordered list -->
<% @patient.doctors.each do |doctor| %>
<li><%= doctor.name %></li>
<% end %>
</ul>
</p>
<%= link_to 'Edit', edit_patient_path(@patient) %> |
<%= link_to 'Back', patients_path %>
patients/index.html.erb
patients/show.html.erb
Patients & their Doctors: in the browser
localhost:3000/patients
localhost:3000/patients/1
You did it!
You created a many-to-many association!
Rails Associations
By argroch
Rails Associations
Creating links between your Resource databases.
- 1,395