If you’ve spent any time learning/using Rails, then you know it’s a very magical language where lots of things that you would otherwise have to code manually, just seem to work. Sometimes that makes it hard to truly grasp why something is happening and where it’s coming from. Rails routing fell into this category for me.
For each of your application’s routes, Rails automatically creates corresponding path and url helper methods that can be used rather than hard coding. Below is a list of the named path helpers that are generated by Rails using a Song model as an example. Each of these helpers has a corresponding _url helper (such as songs_url).
Named Helpers
Verb | Path | Action | Path Helper | URL Helper |
---|---|---|---|---|
GET | /songs | index | songs_path | songs_url |
GET | /songs/new | new | new_song_path | new_song_url |
POST | /songs | create | songs_path | songs_url |
GET | /songs/:id | show | song_path(:id) | song_url(:id) |
GET | /songs/:id/edit | edit | edit_song_path(:id) | edit_song_url(:id) |
PUT/PATCH | /songs/:id | update | song_path(:id) | song_url(:id) |
DELETE | /songs/:id | destroy | song_path(:id) | song_url(:id) |
Path Example
Let’s say we have a new Rails app that creates and stores a list of songs with the following attributes: name, artist, genre.
We’ll want to have an index page that lists all of the songs, a new page where a new song can be created, and a show page where individual song data can be found, so our config/routes.rb
file will look like the below:
We’ll want to create methods for each of our routes, so our app/controllers/songs_controller.rb
file will have the following:
And in app/views/songs/index.html.erb
, we’ll want to iterate through all of our songs for the song name and link to it.
We can see that we’re referencing path helpers in theindex.html.erb
file as well as the in the create method in songs_controller.rb
.
The Index Page
Let’s focus on how this path helper method is being used on the index page first.
def index
@songs = Song.all
end
In the songs_controller.rb, we’re creating an instance variable @songs
that contains all songs in the database.
<% @songs.each do |song| %>
<div><%= link_to song.name, song_path(song) %> by <%= song.artist %></div>
<% end %>
Then in our index.html.erb file
, we’re iterating through @songs
to get to each song object. Since we want to display each song name and link to it’s show page, for each iteration we’ll use the link_to method (another Rails helper) to say we want the link text to be song.name and we want to call the built in path helper method - song_path(song)
for the show page on that song id.
Side note: although we’re passing along the whole song instance in song_path(song)
, Rails is just using the song ID. We could just as easily have used song_path(song.id)
.
The Create Page
The path helper for the show page is also being used in the create method in the songs_controller.rb
file.
def create
@song = Song.create(song_params)
redirect_to song_path(@song)
end
Here we’ve created a new song instance and set the @song
instance variable to it. Once the song has been created, we want to redirect the user to that song’s show page, so we’ve called song_path(@song)
, passing in the instance of song that we’ve just created.
URL vs. Path Helpers
URL helpers are similar to path, except their return values are prefixed with the current host, port and path prefix. URL helpers are used when referencing a link outside of your application such as an RSS feed.
Still fuzzy?
Finding the methods available for your application
- In command line, from inside of your app directory:
rails console
- If you’re using input the following, depending on your version of Rails:
- Rails 5:
helpers = Rails.application.routes.named_routes.helper_names
. - Rails 4:
helpers = Rails.application.routes.named_routes.helpers
- Rails 5:
- You’ll see that thenew variable
helpers
returns a large array of all of the helpers available for your routes. If you want to see only the path and url helpers for the app:helpers.select {|h| h.to_s.include?("song") }
. For the example in this post, this returns:["songs_path", "new_song_path", "song_path", "songs_url", "new_song_url", "song_url"]