Your goal is to build an advanced public library application in Ruby on Rails.
By the end of this lab you will have:
- Basic-Auth: Login, Signup, Logout
- Basic views for your users
- A Many-to-Many database relationship: Users, Libraries, Library Users
- "Skinny", Refactored controllers
Create a new rails app:
rails new lib_app -T -d postgresql
cd lib_app
Create the databases:
rake db:create
Let's start by generating a user
model.
rails g model user email:string first_name:string last_name:string password_digest:string
Then go ahead and verify that the migration looks correct:
db/migrate/*_create_users.rb
class CreateUsers < ActiveRecord::Migration
def change
create_table :users do |t|
t.string :email
t.string :first_name
t.string :last_name
t.string :password_digest
t.timestamps null: false
end
end
end
And it does! Whoot! We're ready to migrate!
rake db:migrate
Then all we have to do is create the routes for a user.
config/routes.rb
Rails.application.routes.draw do
root to: "users#index"
get "/users", to: "users#index", as: "users"
end
We can look at how these routes are interpreted by Rails.
rake routes
Which gives us the following routes:
Prefix Verb URI Pattern Controller#Action
root GET / users#index
users GET /users(.:format) users#index
Note the special Prefix
column this will be of great use later.
Question?? What the heck is
/users(.:format)
and what does that mean?
Well the even bigger question now is what to do next? The truth is we don't have a users#index
. We don't even have a UsersController
. Let's practice using our rails generate
skills.
rails g controller users
This does something like the following:
*** create app/controllers/users_controller.rb
invoke erb
*** create app/views/users
invoke helper
** create app/helpers/users_helper.rb
invoke assets
invoke coffee
** create app/assets/javascripts/users.coffee
invoke scss
** create app/assets/stylesheets/users.scss
Note the special create
statements here. The ***
ones are the most important. It creates the users_controller.rb
file and the views/users
directory.
Now that we have a users_controller.rb
we should add our users#index
method.
class UsersController < ApplicationController
# grab the users
def index
@users = User.all
render :index
end
end
Then we need to actually create an index.html.erb
:
touch app/views/users/index.html.erb
Then we can go ahead and add something special to our index
:
<h1>Welcome to Users Index.</h1>
<div>
There are currently <%= @users.length %> signed_up
</div>
Now we should see 0
users signed_up. We should change that!
Rails.application.routes.draw do
root to: "users#index"
get "/users", to: "users#index", as: "users"
get "/users/new", to: "users#new", as: "new_user"
end
With the following output after we rake routes
:
Prefix Verb URI Pattern Controller#Action
root GET / users#index
users GET /users(.:format) users#index
new_user GET /users/new(.:format) users#new
We don't have a users#new
so let's create one.
class UsersController < ApplicationController
def new
# we need to make
# a new user
# to pass to the
# form later
@user = User.new
render :new
end
end
Then we can continue on to creating a new.html.erb
Sign Up
<%= form_for @user do |f| %>
<div>
<%= f.text_field :first_name, placeholder: "First Name" %>
</div>
<div>
<%= f.text_field :last_name, placeholder: "Last Name" %>
</div>
<div>
<%= f.text_field :email, placeholder: "Email" %>
</div>
<div>
<%= f.password_field :password, placeholder: "Password" %>
</div>
<%= f.submit "Sign Up" %>
<% end %>
Which renders a form like the following (note the authenticity token):
<!-- DO NOT COPY THIS CODE -->
Sign Up
<form class="new_user" id="new_user" action="/users" accept-charset="UTF-8" method="post"><input name="utf8" type="hidden" value="✓" /><input type="hidden" name="authenticity_token" value="5989PH35p43aagbgiuA/C02p8uD6bLmZR+GCLd01lYPmBOSGLNoHMnEGuZXyzHjnTsMvW6h5860tN6CswMsU5A==" />
<div>
<input placeholder="First Name" type="text" name="user[first_name]" id="user_first_name" />
</div>
<div>
<input placeholder="Last Name" type="text" name="user[last_name]" id="user_last_name" />
</div>
<div>
<input placeholder="Email" type="text" name="user[email]" id="user_email" />
</div>
<div>
<input placeholder="Password" type="password" name="user[password]" id="user_password" />
</div>
<input type="submit" name="commit" value="Sign Up" />
</form>
Note here the correlation between the key we put into f.text_field
and name="..."
.
Also note where this form is going
<form class="new_user" id="new_user" action="/users" accept-charset="UTF-8" method="post">
It looks like this form is sending POST /USERS
, but we don't have that route so we have to create it.
Rails.application.routes.draw do
root to: "users#index"
get "/users", to: "users#index", as: "users"
get "/users/new", to: "users#new", as: "new_user"
post "/users", to: "users#create"
end
Then we need to add that method.
class UsersController < ApplicationController
...
def create
user_params = params.require(:user).permit(:first_name, :last_name, :email, :password)
@user = User.create(user_params)
redirect_to "/users"
end
end
Now when you submit the form you get the following error:
ActiveRecord::Unknown
AttributeError in UsersController#create
unknown attribute 'password' for User.
This is because we only have a password_digest
. We also haven't setup our application to help users sign up at all. This is a good time to start adding our authentication logic.
Uncomment your bcrypt
in your Gemfile
Gemfile
...
# Use ActiveModel has_secure_password
gem 'bcrypt', '~> 3.1.7'
...
Then we can add has_secure_password
to our user model application.
class User < ActiveRecord::Base
has_secure_password
end
Now when we post the form for the user you'll see the user being created. The difference now is the password_digest
is being properly hashed.
Now we want to add a route to GET /users/:id
.
Rails.application.routes.draw do
root to: "users#index"
get "/users", to: "users#index", as: "users"
get "/users/new", to: "users#new", as: "new_user"
post "/users", to: "users#create"
get "/users/:id", to: "users#show"
end
We want to add a users#show
page.
class UsersController < ApplicationController
def show
@user = User.find(params[:id])
render :show
end
end
Then we need a view to display the users information.
<div>
Welcome, <%= @user.email %>
</div>
Now that we can create a user we need to be able to sign a user in.
Signing and signing out is a concern of a new controller, the sessions controller.
rails g controller sessions
Note this will create both sessions_controller.rb
and sessions_helper.rb
.
Now we should use the session_helper
by adding our own logic to it.
module SessionsHelper
def login(user)
session[:user_id] = user.id
@current_user = user
end
def current_user
@current_user ||= User.find(session[:user_id])
end
def logged_in?
if current_user == nil
redirect_to "/sign_in"
end
end
def logout
@current_user = session[:user_id] = nil
end
end
These methods will help avoid code bloat when signing in and out. Before we can use the methods though we have to add these methods to the ApplicationController
.
class ApplicationController < ActionController::Base
# Prevent CSRF attacks by raising an exception.
# For APIs, you may want to use :null_session instead.
protect_from_forgery with: :exception
include SessionsHelper
end
Now, we are ready to continue. Let's add some routes to sign_in
.
Rails.application.routes.draw do
...
get "/sign_in", to: "sessions#new"
end
Now we need to add the sessions#new
.
class SessionsController < ApplicationController
def new
@user = User.new
render :new
end
end
Then we need to add a view for the sessions/new.html.erb
.
touch app/views/sessions/new.html.erb
Then very similarly to what did before for sign up we create a form for sign in.
Sign In
<%= form_for @user, url: "/sessions", method: "post" do |f| %>
<div>
<%= f.text_field :email, placeholder: "Email" %>
</div>
<div>
<%= f.password_field :password, placeholder: "Password" %>
</div>
<%= f.submit "Sign In" %>
<% end %>
Before we go forward let's go ahead and drop in a very key piece of confirmation logic into our user
model.
class User < ActiveRecord::Base
has_secure_password
def self.confirm(params)
@user = User.find_by({email: params[:email]})
@user.try(:authenticate, params[:password])
end
end
Note that the form is getting submited to POST /sessions
. We don't have a sessions#create
however or a route to handle the post.
Rails.application.routes.draw do
get "/sign_in", to: "sessions#new"
post "/sessions", to: "sessions#create"
end
Now let's add the sessions#create
class SessionsController < ApplicationController
def create
user_params = params.require(:user).permit(:email, :password)
@user = User.confirm(user_params)
if @user
login(@user)
redirect_to "/users/#{@user.id}"
else
redirect_to "/sign_in"
end
end
end
Then when we try to login let's see what happens. Do you see a welcome? If so you're ready to continue otherwise you should start the long work of debugging.
After a user is signed up they should be logged in.
class UsersController < ApplicationController
def create
user_params = params.require(:user).permit(:first_name, :last_name, :email, :password)
@user = User.create(user_params)
login(@user) # <-- login the user
redirect_to "/users/#{@user.id}" # <-- go to show
end
end
Let's add our second model a Library
model that will later have books.
rails g model library name:string floor_count:integer floor_area:integer
We want a user
to be able to join a library, but this means a m:n
relationship. A user will have many libraries and library will have many users.
Thus we need a library_user
model.
rails g model library_user user:references library:references
In the future we can store other things on the library_user
model that a relevant to someone's memembership to a library.
We will also need two different controllers for each of these models. Let's start by being able to do CRUD with Libraries.
rails g controller libraries
Let's add a route to be able to view all the libraries.
Rails.application.routes.draw do
...
get "/libraries", to: "libraries#index"
end
Then we need to add a libraries#index
method to our libraries controller.
class LibrariesController < ApplicationController
def index
@libraries = Library.all
render :index
end
end
Finally we can add a basic view for all libraries.
<% @libraries.each do |library| %>
<div>
<h3><%= library.name %></h3>
</div>
<br>
<% end %>
To be able to add a new library we need a libraries#new
.
Rails.application.routes.draw do
...
get "/libraries/new", to: "libraries#new", as: "new_library"
end
Then we add a libraries#new
method.
class LibrariesController < ApplicationController
...
def new
@library = Library.new
render :new
end
end
Finally, we can add a view for new
library.
<%= form_for @library do |f| %>
<div>
<%= f.text_field :name, placeholder: "Name" %>
</div>
<div>
<%= f.number_field :floor_count, placeholder: "Floor Count" %>
</div>
<div>
<%= f.number_field :floor_area, placeholder: "Floor Area" %>
</div>
<%= f.submit %>
<% end %>
This form has nowhere to go if we try to submit it we get an error because there is no POST /libraries
route.
Let's add one.
Rails.application.routes.draw do
...
post "/libraries", to: "libraries#create"
end
Then we need a corresponding libraries#create
.
class LibrariesController < ApplicationController
def create
library_params = params.require(:library).permit(:name, :floor_count, :floor_area)
@library = Library.create(library_params)
redirect_to "/libraries"
end
end
We now have the ability to view all libraries, and it's up to you to create methods to edit
, update
, show
, and delete
a library
.
Before we get started joining a library
and a user
we need to wire together our Library
and our User
via associations.
class User < ActiveRecord::Base
has_many :library_users
has_many :libraries, through: :library_users
...
end
And We do something similar for a Library.
class Library < ActiveRecord::Base
has_many :library_users
has_many :users, through: :library_users
end
But notice here that both models are connected through as library_users
model. Hence we need to let that model know it belongs to both of those.
class LibraryUser < ActiveRecord::Base
belongs_to :user
belongs_to :library
end
You should now test this out in the console.
> user = User.first
> user.libraries
#=> []
> sfpl = Library.create({name: "SFPL"}) # San Francisco Public Library
> sfpl.users
#=> []
> sfpl.users.push(user)
> sfpl.users
#=> [ <#User ... @id=1> ]
> LibraryUser.count
#=> 1
> user.libraries
#=> [ <#Library ... @name="SFPL" @id=1> ]
Joining a library requires creating library_users
controller
rails g controller library_users
We want to be able to view all user memberships to a library. We can specify this as a url like /users/:user_id/libraries
.
Rails.application.routes.draw do
...
get "/users/:user_id/libraries", to: "library_users#index", as: "user_libraries"
end
We also neeed the corresponding index
method in the library_users
controller
class LibraryUsersController < ApplicationController
def index
@user = User.find(params[:user_id])
@libraries = @user.libraries
render :index
end
end
Then we can have the libraries index render the user and the libraries:
<div><%= @user.first_name %> is a member of the following libraries</div>
<ul>
<% @libraries.each do |lib| %>
<li><%= lib.name %></li>
<% end %>
</ul>
We can test this by going to localhost:/users/1/libraries
.
So now that we can view, which libraries a user
has joined we can go ahead and make a button that allows a user to join
a library.
Let's go back to libraries#index
and add a button to do just that.
<% @libraries.each do |library| %>
<div>
<h3><%= library.name %></h3>
<% if @current_user %>
<%= button_to "Join", library_users_path(library) %>
<% end %>
</div>
<br>
<% end %>
We will have to define library_user_path
to POST /libraries/:library_id/users
later. But first we need to update the library#index
method.
class LibrariesController < ApplicationController
def index
@libraries = Library.all
current_user # sets @current_user
render :index
end
...
end
Of course we now realize we don't have a POST /libraries/:library_id/users
path, so we need to add one.
Rails.application.routes.draw do
...
get "/users/:user_id/libraries", to: "library_users#index", as: "user_libraries"
post "/libraries/:library_id/users", to: "library_users#create", as: "library_users"
end
Then we need to add the create
method to the library_users
controller.
class LibraryUsersController < ApplicationController
...
def create
@user = current_user
@library = Library.find(params[:library_id])
@user.libraries.push(@library)
redirect_to user_libraries(@user)
end
end
Let's say that in order to visit a users#show
page you have to be logged in. Then we can add a special before_action
to check this.
class UsersController < ApplicationController
before_action :logged_in?, only: [:show]
...
def show
@user = User.find(params[:id])
render :show
end
end
-
Make it so a user has to be
logged_in?
before viewing anything of theLibrariesController
actions or theLibraryUsers
actions. -
Modify exercise one such anyone can view
libraries#index
, but cannotcreate
or viewnew
without being logged in.
Every time we take in a lot of params in a controller it's tedious to write out.
class UsersController < ApplicationController
...
def create
user_params = params.require(:user).permit(:first_name, :last_name, :email, :password)
@user = User.create(user_params)
login(@user)
redirect_to "/users/#{@user.id}"
end
...
end
You can utilize a private method for doing this. Let's refactor.
class UsersController < ApplicationController
...
def create
@user = User.create(user_params) # calls user_params method
login(@user)
redirect_to "/users/#{@user.id}"
end
...
private
def user_params
params.require(:user).permit(:first_name, :last_name, :email, :password)
end
end
- Private methods like
user_params
are simple to implement and give us cleaner looking code. Rewritelibraries#create
using this idea.
- Can you add books to the application?
- For starters, just create a
Book
model and the associated views.
- For starters, just create a
- Can you add books to the library?
- What kind of a relationship is that? Where would foreign keys like
book_id
andlibrary_id
live in your database tables?
- What kind of a relationship is that? Where would foreign keys like