项目作者: Roger-Takeshita

项目描述 :
Buffer Tweet Scheduler
高级语言: Ruby
项目地址: git://github.com/Roger-Takeshita/Ruby_On_Rails_Buffer.git
创建时间: 2021-02-28T03:19:43Z
项目社区:https://github.com/Roger-Takeshita/Ruby_On_Rails_Buffer

开源协议:MIT License

下载


TABLE OF CONTENTS

SCHEDULE TWEETS - BUFFER CLONE

Go Back to Contents

Start New Project

Go Back to Contents

  1. rails new scheduled_tweets

MVC Structure

GET /about HTTP/1.1
Host: 127.0.0.1

Routes

Go Back to Contents

Matches for the URL that is requested

GET for /about

I see you requested /about, we’ll give that to the AboutController to handle

Model

Go Back to Contents

Database wrapper

User

  • query for record
  • wrap individual records

Views

Go Back to Contents

Your response body content

  • HTML
  • CSV
  • PDF
  • XML

This is what gets sent back to the browser and displayed

Controllers

Go Back to Contents

Decide how to process a request and define a response

About Page

About Route

Go Back to Contents

First thing we need to do is to config our routes.rb

In config/routes.rb

  1. Rails.application.routes.draw do
  2. # GET /about
  3. get 'about', to: 'about#index'
  4. end
  1. get 'about', to: 'about#index'
  2. # | | └── action (function)
  3. # | └── about controller
  4. # └── /about route
  5. # this route will look for a controller about_controller (rails convention)

About Controller

Go Back to Contents

Create the about controller following rails convention

  1. touch app/controllers/about_controller.rb

In app/controllers/about_controller.rb

  • We need the define the about controller, but as a Ruby Class (Title case without underscores) and we need to inherit from ApplicationController.

    • We need to inherit from ApplicationController so we can have access to all functionality from rails.
    • Inside our class we can define a new action/function index. Right now this can be an empty action, because we are inheriting from the ApplicationController and if don’t have anything defined inside our action/function, the ApplicationController has a default action to handle this case which is to serve an HTML file.
    1. class AboutController < ApplicationController
    2. def index
    3. end
    4. end

About View

Go Back to Contents

Create the about html file, by convention we rails will look for a folder called about in our views folder. And inside this folder for a file called index.html.erb

  1. touch app/views/about/index.html.erb

In app/views/about/index.html.erb

  • Add a simple h1 tag

    1. <h1>About Us</h1>

We can defile only the part of the app that we want to create, later this code will be injected in our layout (app/views/layouts/application.html.erb).

The layout is a wrapper that will define all our StyleSheets, JavaScripts, favicon… All that are defined inside our layout.

The layout will yield and replace with our view content using erb/ruby tags.

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>ScheduledTweets</title>
  5. <meta name="viewport" content="width=device-width,initial-scale=1">
  6. <%= csrf_meta_tags %>
  7. <%= csp_meta_tag %>
  8. <%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
  9. <%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
  10. </head>
  11. <body>
  12. <%= yield %>
  13. </body>
  14. </html>

Main Page/Root

Main Route

Go Back to Contents

For all our Rails app, we need to define the root controller.

In development mode, rails serve us with the default view that only works for development. It will throw an error if we deploy the production version without defining the root route

In config/routes.rb

  • We can define the main root as:

    1. get "", to: 'main#index'
  • But because the root is special, we can define as:

    1. root to: 'main#index'
  1. Rails.application.routes.draw do
  2. # GEt /about
  3. get 'about', to: 'about#index'
  4. # get "", to: 'main#index'
  5. root to: 'main#index'
  6. end

Main Controller

Go Back to Contents

Create the main controller following rails convention

  1. touch app/controllers/main_controller.rb

In app/controllers/main_controller.rb

  1. class MainController < ApplicationController
  2. def index
  3. end
  4. end

Main View

Go Back to Contents

Create the index.html.erb

In app/views/main/index.html.erb

  • Add a simple message

    1. <h1>Welcome to Scheduled Tweets</h1>

Bootstrap

Go Back to Contents

  • Bootstrap

    1. <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.0.0-beta2/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-BmbxuPwQa2lc/FVzBcNJ7UAyJxM6wuqIj61tLrc4wSX0szH/Ev+nYRRuWlolflfl" crossorigin="anonymous">

In app/views/layouts/application.html.erb

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>ScheduledTweets</title>
  5. <meta name="viewport" content="width=device-width,initial-scale=1">
  6. <%= csrf_meta_tags %>
  7. <%= csp_meta_tag %>
  8. <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.0.0-beta2/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-BmbxuPwQa2lc/FVzBcNJ7UAyJxM6wuqIj61tLrc4wSX0szH/Ev+nYRRuWlolflfl" crossorigin="anonymous">
  9. <%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
  10. <%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
  11. </head>
  12. <body>
  13. <%= yield %>
  14. </body>
  15. </html>

In app/views/main/index.html.erb

  1. <div class="d-flex align-items-center justify-content-center">
  2. <h1>Welcome to Scheduled Tweets</h1>
  3. </div>

Rendering Partials

Go Back to Contents

Partial is parts of views that we can render inside our view. In this case the application.html.erb layout (base template)

In app/views/shared/_navbar.html.erb

  • Paste the navbar that we copied from bootstrap website

    1. <nav class="navbar navbar-expand-lg navbar-light bg-light">
    2. <div class="container-fluid">
    3. <a class="navbar-brand" href="/">Navbar</a>
    4. <button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
    5. <span class="navbar-toggler-icon"></span>
    6. </button>
    7. <div class="collapse navbar-collapse" id="navbarNav">
    8. <ul class="navbar-nav">
    9. <li class="nav-item">
    10. <a class="nav-link active" aria-current="page" href="/">Home</a>
    11. </li>
    12. <li class="nav-item">
    13. <a class="nav-link" href="/about">About</a>
    14. </li>
    15. </li>
    16. </ul>
    17. </div>
    18. </div>
    19. </nav>

In app/views/layouts/application.html.erb

  • Inside our body, we can render our partial navbar
  • And we can wrap our view content with a container class

    1. <%= render partial: 'shared/navbar' %>
    2. <div class="container">
    3. <%= yield %>
    4. </div>

Rails Helpers

Go Back to Contents

We can replace our hard coded /about with one that is more dynamic that will help us to change the route (if we want to) more easily in several pages

In app/views/shared/_navbar.html.erb

  1. <!-- From -->
  2. <a class="nav-link" href="/about">About</a>
  3. <!-- To -->
  4. <%= link_to "About", about_path, class: "nav-link" %>
  5. <!-- | | └── css class -->
  6. <!-- | └── href (/about) -->
  7. <!-- └── content (text) -->
  • the about_path will generate /about

  • Update our navbar to use the link_to helper and url helper

    1. <nav class="navbar navbar-expand-lg navbar-light bg-light">
    2. <div class="container-fluid">
    3. <%= link_to "Navbar", root_path, class: "navbar-brand" %>
    4. <button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
    5. <span class="navbar-toggler-icon"></span>
    6. </button>
    7. <div class="collapse navbar-collapse" id="navbarNav">
    8. <ul class="navbar-nav">
    9. <li class="nav-item">
    10. <%= link_to "Home", root_path, class: "nav-link active" %>
    11. </li>
    12. <li class="nav-item">
    13. <%= link_to "About", about_path, class: "nav-link" %>
    14. </li>
    15. </li>
    16. </ul>
    17. </div>
    18. </div>
    19. </nav>

Once we update our navbar partial, we can update our route to map to a custom route

In config/routes.rb

  1. # Old way (hard coded)
  2. get 'about', to: 'about#index'
  3. # custom route `about-us` using alias
  4. get 'about-us', to: 'about#index', as: :about
  5. # | | | └── alias
  6. # | | └── action (function)
  7. # | └── about controller
  8. # └── /about-us route (custom route)

Flash Messages

Go Back to Contents

The flash method is inherit from ApplicationController, it’s a feature of controllers and views.

We use flash as a hash (key/value pairs) in Ruby, and this is going to store the value into our flash object

The flash object can be assigned in the controllers but we need to print them out somewhere in our views

  1. flash[:notice] = "Logged in successfully"
  2. # | | └── value/message
  3. # | └── key/variable
  4. # └── method
  1. flash.now[:notice] = "Logged in successfully"
  2. # | | | └── value/message
  3. # | | └── key/variable
  4. # | └── only display the flash message on the current page
  5. # └── method
  • when we add the .now, this tells Rails to display only for the current page/view/controller, it won’t persist if we change to another page

Because the flash object is shared across our entire rails app, we are going to create a shared/partial for it

Flash Partial

Go Back to Contents

Create a new partial _flash.html.erb

  1. touch app/views/shared/_flash.html.erb

In app/views/shared/_flash.html.erb

  1. <%= flash[:notice] %>
  2. <%= flash[:alert] %>

In app/views/layouts/application.html.erb

  • We we can add a new partial in our <body> tag
  1. <body>
  2. <%= render partial: 'shared/navbar' %>
  3. <div class="container">
  4. <%= render partial: 'shared/flash' %>
  5. <%= yield %>
  6. </div>
  7. </body>

User Authentication

Generate User Model

Go Back to Contents

We are going to create a user model to store email and password (digest - hashed)

  • rails g = rails generate
  • model User = User (Title case)
  • email = type string
  • password = hashed password
  1. rails g model User email:string password_digest:string
  2. # invoke active_record
  3. # create db/migrate/20210228170124_create_users.rb
  4. # create app/models/user.rb
  5. # invoke test_unit
  6. # create test/models/user_test.rb
  7. # create test/fixtures/users.yml
  8. rails db:migrate
  9. # == 20210228170124 CreateUsers: migrating ======================================
  10. # -- create_table(:users)
  11. # -> 0.0021s
  12. # == 20210228170124 CreateUsers: migrated (0.0021s) =============================

User Model

Go Back to Contents

In app/models/user.rb

  • We need to add has_secure_password, the has_secure_password will use our password_digest to add a virtual password and password confirmation attributes to our user. It uses bcrypt to authenticate/save the use

    | attribute | type | description |
    | ——————————- | ——— | ———————— |
    | email | string | column in the db |
    | password_digest | string | column in the db |
    | | | |
    | password | string | virtual |
    | password_confirmation | string | virtual |

    • we are never going to interact directly with password_digest, we’ll always use the password and password_confirmation

    • We need to install bcrypt gem for that

  1. class User < ApplicationRecord
  2. has_secure_password
  3. end

Add User Using Rails Console

After installing bcrypt, we can create our first user using the rails console

  1. rails c
  2. # Running via Spring preloader in process 8223
  3. # Loading development environment (Rails 6.1.3)
  4. User
  5. # => User (call 'User.connection' to establish a connection)
  6. User.all
  7. # (0.4ms) SELECT sqlite_version(*)
  8. # User Load (0.1ms) SELECT "users".* FROM "users" /* loading for inspect */ LIMIT ? [["LIMIT", 11]]
  9. # => #<ActiveRecord::Relation []>
  10. User.create({email: "roger@email.com", password: "123", password_confirmation: "123"})
  11. # (0.5ms) SELECT sqlite_version(*)
  12. # TRANSACTION (0.1ms) begin transaction
  13. # User Create (0.3ms) INSERT INTO "users" ("email", "password_digest", "created_at", "updated_at") VALUES (?, ?, ?, ?) [["email", "roger@email.com"], ["password_digest", "$2a$12$W0lFDMCW.VXzvTAigQpBIuOv2MhomwPRcnjWGzcStNOKQqIwexB7e"], ["created_at", "2021-02-28 17:27:33.442553"], ["updated_at", "2021-02-28 17:27:33.442553"]]
  14. # TRANSACTION (1.6ms) commit transaction
  15. # => #<User id: 1, email: "roger@email.com", password_digest: [FILTERED], created_at: "2021-02-28 17:27:33.442553000 +0000", updated_at: "2021-02-28 17:27:33.442553000 +0000">
  16. User.first
  17. # User Load (0.2ms) SELECT "users".* FROM "users" ORDER BY "users"."id" ASC LIMIT ? [["LIMIT", 1]]
  18. # => #<User id: 1, email: "roger@email.com", password_digest: [FILTERED], created_at: "2021-02-28 17:27:33.442553000 +0000", updated_at: "2021-02-28 17:27:33.442553000 +0000">

Validation

Go Back to Contents

In app/models/user.rb

  • We can use rails validation to check if a field is presence

    1. class User < ApplicationRecord
    2. has_secure_password
    3. validates :email, presence: true
    4. end
    • The validates :email, presence: true will check if an email is present before saving into the database.

Another way to validate the email, is by updating our migration file to require an email

In db/migrate/20210228170124_create_users.rb

  • Add , null: false to our email field

    1. class CreateUsers < ActiveRecord::Migration[6.1]
    2. def change
    3. create_table :users do |t|
    4. t.string :email, null: false
    5. t.string :password_digest
    6. t.timestamps
    7. end
    8. end
    9. end
    • Because we changed our migration file, we need to undo our modification and then migrate again to apply the modifications to our database.
    1. rails db:rollback
    2. # == 20210228170124 CreateUsers: reverting ======================================
    3. # -- drop_table(:users)
    4. # -> 0.0018s
    5. # == 20210228170124
    6. rails db:migrate
    7. # == 20210228170124 CreateUsers: migrating ======================================
    8. # -- create_table(:users)
    9. # -> 0.0016s
    10. # == 20210228170124 CreateUsers: migrated (0.0018s) ============================= CreateUsers: reverted (0.0121s) =============================
    11. # an alternative we could use a single command to do that
    12. rails db:migrate:redo
    13. # == 20210228170124 CreateUsers: reverting ======================================
    14. # -- drop_table(:users)
    15. # -> 0.0016s
    16. # == 20210228170124 CreateUsers: reverted (0.0031s) =============================
    17. # == 20210228170124 CreateUsers: migrating ======================================
    18. # -- create_table(:users)
    19. # -> 0.0015s
    20. # == 20210228170124 CreateUsers: migrated (0.0016s) =============================

On Rails Console, if we try to save an user without email we will get an User of id: nil, this means that this user wasn’t saved in the database.

  1. rails c
  2. User.create({password: "123", password_confirmation: "123"})
  3. # (0.4ms) SELECT sqlite_version(*)
  4. # => #<User id: nil, email: nil, password_digest: [FILTERED], created_at: nil, updated_at: nil>

Validate Format And Message

Go Back to Contents

In app/models/user.rb

  • We can add a regex to validate the format

    1. class User < ApplicationRecord
    2. has_secure_password
    3. validates :email, presence: true, format: { with: /\A[^@\s]+@[^@\s]+\z/, message: "must be a valid email address" }
    4. end
    1. rails c
    2. User.create({email: "A", password: "123", password_confirmation: "123"})
    3. # (0.4ms) SELECT sqlite_version(*)
    4. # => #<User id: nil, email: "A", password_digest: [FILTERED], created_at: nil, updated_at: nil>

Sign Up Page

Registration Route

Go Back to Contents

In config/routes.rb

  • Add the sign_up route and map to registrations controller > new action/function

    1. Rails.application.routes.draw do
    2. # GEt /about
    3. # get 'about', to: 'about#index'
    4. get 'about-us', to: 'about#index', as: :about
    5. get 'sign_up', to: 'registrations#new'
    6. root to: 'main#index'
    7. end

Registration Controller - New Action

Go Back to Contents

Create a the registration_controller.rb

  1. touch app/controllers/registrations_controller.rb

In app/controllers/registrations_controller.rb

  • We can create a new instance variable @user, when we create an instance variable this variable is available in our views
  1. class RegistrationsController < ApplicationController
  2. def new
  3. @user = User.new
  4. end
  5. end

Registration View

Go Back to Contents

Crate the registrations > new view

  1. touch app/views/registrations/new.html.erb

In app/views/registrations/new.html.erb

  • We can use the @user instance to print on our view

    1. <h1>Sign Up</h1>
    2. <%= @user %>
FORM

Go Back to Contents

To work with forms we need to update our routes.rb

In config/routes.rb

  • Add a POST route for sign_up

    1. Rails.application.routes.draw do
    2. get 'about-us', to: 'about#index', as: :about
    3. get 'sign_up', to: 'registrations#new'
    4. post 'sign_up', to: 'registrations#create'
    5. root to: 'main#index'
    6. end

In app/views/registrations/new.html.erb

  • In rails we can generate a form using an instance variable/object to create all the necessary fields

    1. <h1>Sign Up</h1>
    2. <%= form_with model: @user, url: sign_up_path do |form| %>
    3. <%= form.text_field :email %>
    4. <%= form.password_field :password %>
    5. <%= form.password_field :password_confirmation %>
    6. <% end %>
    1. form_with model: @user, url: sign_up_path do |form|
    2. # | | | └── /sign_up (url)
    3. # | | └── instance variable (User model)
    4. # | └── model
    5. # └── create a form with

      1. url
      1. type of request
      1. token (to validate the form, so our serve knows that is coming from our app)
  • Display errors

    • We can add an if statement to check for errors

      1. <% if @user.errors.any? %>
      2. <div class="alert alert-danger">
      3. <% @user.errors.full_messages.each do |message| %>
      4. <div>
      5. <%= message %>
      6. </div>
      7. <% end %>
      8. </div>
      9. <% end %>

  1. <h1>Sign Up</h1>
  2. <%= form_with model: @user, url: sign_up_path do |form| %>
  3. <% if @user.errors.any? %>
  4. <div class="alert alert-danger">
  5. <% @user.errors.full_messages.each do |message| %>
  6. <div>
  7. <%= message %>
  8. </div>
  9. <% end %>
  10. </div>
  11. <% end %>
  12. <div class="mb-3">
  13. <%= form.label :email%>
  14. <%= form.text_field :email, class: "form-control", placeholder: "your_email@email.com"%>
  15. </div>
  16. <div class="mb-3">
  17. <%= form.label :password%>
  18. <%= form.password_field :password, class: "form-control", placeholder: "password" %>
  19. </div>
  20. <div class="mb-3">
  21. <%= form.label :password_confirmation%>
  22. <%= form.password_field :password_confirmation, class: "form-control", placeholder: "password" %>
  23. </div>
  24. <div class="mb-3">
  25. <%= form.submit "Sign Up", class: "btn btn-primary"%>
  26. </div>
  27. <% end %>

Registration Controller - Create Action

Go Back to Contents

We can get all form information that we submitted using the params, this could be from the query params or from our form that has been added to the params

In app/controllers/registrations_controller.rb

  • As an example we could send back to the user the form that he just submitted

    • In this case we are pulling out all the data that we added to :user param
    1. class RegistrationsController < ApplicationController
    2. def new
    3. @user = User.new
    4. end
    5. def create
    6. params
    7. # {"authenticity_token"=>"[FILTERED]", "user"=>{"email"=>"test", "password"=>"[FILTERED]", "password_confirmation"=>"[FILTERED]"}, "commit"=>"Sign Up"}
    8. params[:user]
    9. # "user"=>{"email"=>"test", "password"=>"[FILTERED]", "password_confirmation"=>"[FILTERED]"}
    10. render plain: params[:user]
    11. end
    12. end

  • Now we can create our create action using params

    • We could simply define as:

      1. def create
      2. @user = User.new(params[:user])
      3. end
      • but this is bad, because the user can inject other things such as an admin field as true (if you had one) and rail would save this new user as an admin
    • To fix that, we can create a private method to allow only certain fields

      1. private
      2. def user_params
      3. params.require(:user).permit(:email, :password, :password_confirmation)
      4. # | └── permit only these fields
      5. # └── require a user (if not rails will throw an error)
      6. end
    • Then we could check if the user was successfully save in our database @user.save

      • If yes, redirect to the root_path and display the flash[:notice] msg
      • If no, redirect to the :new (/sign_up page)
  1. class RegistrationsController < ApplicationController
  2. def new
  3. @user = User.new
  4. end
  5. def create
  6. # @user = User.new(params[:user])
  7. @user = User.new(user_params)
  8. if @user.save
  9. redirect_to root_path, notice: 'Successfully create account'
  10. else
  11. render :new
  12. end
  13. end
  14. private
  15. def user_params
  16. params.require(:user).permit(:email, :password, :password_confirmation)
  17. end
  18. end

Go Back to Contents

To sign in we are going to use session cookie is stored in the user’s browser. The session cookie is encrypted and no one can read it or modify.

In app/controllers/registrations_controller.rb

  • We are going to update the create action to store the @user.id in the session cookie

    1. def create
    2. @user = User.new(user_params)
    3. if @user.save
    4. session[:user_id] = @user.id
    5. redirect_to root_path, notice: 'Successfully create account'
    6. else
    7. render :new
    8. end
    9. end

In app/controllers/main_controller.rb

  • In our index function, we check if there is a cookie before querying the database.

    1. class MainController < ApplicationController
    2. def index
    3. if session[:user_id]
    4. @user = User.find(session[:user_id])
    5. end
    6. end
    7. end
    • If we use User.find(), and rails doesn’t find a user, then I will throw an error (break the app).
    • A work around to this problem we can use User.find_by(id: session[:user_id]) = SELECT * FROM users WHERE id = user_id;

      1. class MainController < ApplicationController
      2. def index
      3. if session[:user_id]
      4. @user = User.find_by(session[:user_id])
      5. end
      6. end
      7. end
      8. # Shorter version
      9. class MainController < ApplicationController
      10. def index
      11. @user = User.find_by(id: session[:user_id]) if session[:user_id]
      12. end
      13. end

In app/views/main/index.html.erb

  • Then if we have a user, we can go to our main index

    1. <div class="d-flex align-items-center justify-content-center">
    2. <h1>Welcome to Scheduled Tweets</h1>
    3. <% if @user %>
    4. Logged in as: <%= @user.email %>
    5. <% end %>
    6. </div>

We can check our cookie under Application Tab

  • As we can see it’s marked as HttpOnly this means that JavaScript cannot access it

Logout - Session Controllers

Go Back to Contents

To logout from our app, we are going to create a new controller called session_controller.rb

Session Route

Go Back to Contents

Add a new route that maps a DELETE request to sessions > destroy

  1. Rails.application.routes.draw do
  2. get 'about-us', to: 'about#index', as: :about
  3. get 'sign_up', to: 'registrations#new'
  4. post 'sign_up', to: 'registrations#create'
  5. delete 'logout', to: 'sessions#destroy'
  6. root to: 'main#index'
  7. end

Session Controller

Go Back to Contents

Create a new controller:

  1. touch app/controllers/sessions_controller.rb

In app/controllers/sessions_controller.rb

  • Basically our destroy action will set the current session[:user_id] to nil then it will redirect to the home page.

    1. class SessionsController < ApplicationController
    2. def destroy
    3. session[:user_id] = nil
    4. redirect_to root_path, notice: 'Logged out'
    5. end
    6. end

In app/views/main/index.html.erb

  • We can logout using two ways:

    • link_to
    • button_to
  • Both will work, but using button_to is the more appropriate way to logout, because button_to creates a hidden form that sends a DELETE method

    1. <div class="d-flex align-items-center justify-content-center">
    2. <h1>Welcome to Scheduled Tweets</h1>
    3. </div>
    4. <% if @user %>
    5. Logged in as: <%= @user.email %>
    6. <%= link_to "Logout", logout_path, method: :delete %>
    7. <%= button_to "Logout", logout_path, method: :delete %>
    8. <% end %>

Sign In

Sign In Route

Go Back to Contents

Create two new routes:

  • One route to display the sign in page/view
  • The other one to submit the sign in form

In config/routes.rb

  1. Rails.application.routes.draw do
  2. get 'about-us', to: 'about#index', as: :about
  3. get 'sign_up', to: 'registrations#new'
  4. post 'sign_up', to: 'registrations#create'
  5. get 'sign_in', to: 'sessions#new'
  6. post 'sign_in', to: 'sessions#create'
  7. delete 'logout', to: 'sessions#destroy'
  8. root to: 'main#index'
  9. end

Sign In Controller

Go Back to Contents

In app/controllers/sessions_controller.rb

  • Create the new and create actions

    • Because new action is only used to render the new.html.erb we don’t need to pass anything to the view. We just need to create the form in our new.html.erb
    • The create action is used to validate the user
      • First we check if there is any user with the current email
      • Then we use the authenticate method to validate the user password (compare the hashed password with the hashed password in our database)
        • Just like the password and password confirmation, password_digest gives us another method to authenticate the user
        • If the passwords match, then we assign the user.id to our session cookie and redirect to the main page
        • if the passwords don’t match, then we display a flash message and redirect again to the new.html.erb page
    1. class SessionsController < ApplicationController
    2. def new
    3. end
    4. def create
    5. user = User.find_by(email: params[:email])
    6. if user.present? && user.authenticate(params[:password])
    7. session[:user_id]=user.id
    8. redirect_to root_path, notice: 'Logged in successfully'
    9. else
    10. flash[:alert] = 'Invalid email or password'
    11. render :new
    12. end
    13. end
    14. def destroy
    15. session[:user_id] = nil
    16. redirect_to root_path, notice: 'Logged out'
    17. end
    18. end

Sign In View

Go Back to Contents

Create a new folder and file

  1. touch app/views/sessions/new.html.erb

In app/views/sessions/new.html.erb

  1. <h1>Sign In</h1>
  2. <%= form_with url: sign_in_path do |form| %>
  3. <div class="mb-3">
  4. <%= form.label :email%>
  5. <%= form.text_field :email, class: "form-control", placeholder: "your_email@email.com"%>
  6. </div>
  7. <div class="mb-3">
  8. <%= form.label :password%>
  9. <%= form.password_field :password, class: "form-control", placeholder: "password" %>
  10. </div>
  11. <div class="mb-3">
  12. <%= form.submit "Sign In", class: "btn btn-primary"%>
  13. </div>
  14. <% end%>

Get Current User Anywhere - ApplicationController

Go Back to Contents

To use the current authenticated user in any controller/view, we just need to remove from the main_controller and add to our application_controller, because all the controllers inherit from the application_controller

In app/controllers/main_controller.rb

  1. class MainController < ApplicationController
  2. end

In app/controllers/application_controller.rb

  • We can create a new method called set_current_user and set the Current.user (that we are going to create) to our query, instead of our instance variable @user
  • Before any action we are going to run set_current_user

    1. class ApplicationController < ActionController::Base
    2. before_action :set_current_user
    3. def set_current_user
    4. Current.user = User.find_by(id: session[:user_id]) if session[:user_id]
    5. end
    6. end

Current Model

Go Back to Contents

Create a new model called current

  1. touch app/models/current.rb

Our Current class inherits from ActiveSupport::CurrentAttributes, and it will have only one attribute (:user)

The ActiveSupport helps us manage the Current.user from other sessions, this way we can have multiple Current.user without clashing/overriding other session user.

  1. class Current < ActiveSupport::CurrentAttributes
  2. attribute :user
  3. end

Update Views

Go Back to Contents

Now that we have configured the Current.user, we can refactor our views to use the Current.user instead of the instance @user

In app/views/main/index.html.erb

  1. <div class="d-flex align-items-center justify-content-center">
  2. <h1>Welcome to Scheduled Tweets</h1>
  3. </div>
  4. <% if Current.user %>
  5. Logged in as: <%= Current.user.email %>
  6. <%= button_to "Logout", logout_path, method: :delete %>
  7. <% end %>

Update Navbar

Go Back to Contents

Update our navbar to display the current user and add sign up and login buttons

In app/views/shared/_navbar.html.erb

  1. <nav class="navbar navbar-expand-lg navbar-light bg-light">
  2. <div class="container-fluid">
  3. <%= link_to "Navbar", root_path, class: "navbar-brand" %>
  4. <button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
  5. <span class="navbar-toggler-icon"></span>
  6. </button>
  7. <div class="collapse navbar-collapse" id="navbarNav">
  8. <ul class="navbar-nav">
  9. <li class="nav-item">
  10. <%= link_to "Home", root_path, class: "nav-link active" %>
  11. </li>
  12. <li class="nav-item">
  13. <%= link_to "About", about_path, class: "nav-link" %>
  14. </li>
  15. </li>
  16. </ul>
  17. <ul class="navbar-nav ms-auto mb-2 mb-lg-0">
  18. <% if Current.user %>
  19. <li class="nav-item">
  20. <span class="navbar-text">
  21. <%= Current.user.email %>
  22. </span>
  23. </li>
  24. <li class="nav-item">
  25. <%= button_to "Logout", logout_path, method: :delete, class: "btn btn-outline-secondary" %>
  26. </li>
  27. <% else %>
  28. <li class="nav-item">
  29. <%= link_to "Sign Up", sign_up_path, class: "nav-link" %>
  30. </li>
  31. <li class="nav-item">
  32. <%= link_to "Login", sign_in_path, class: "nav-link"%>
  33. </li>
  34. <% end %>
  35. </ul>
  36. </div>
  37. </div>
  38. </nav>

Update Password

Password Route

Go Back to Contents

Add two new routes to our routes.rb

In config/routes.rb

  • give a new name to our path

    1. get 'password', to: 'passwords#edit', as: :edit_password
    2. patch 'password', to: 'passwords#update'

Application Controller

Go Back to Contents

Before we started coding the password controller, we need to define another “global” method require_user_logged_in so we can use this method before each private route.

In app/controllers/application_controller.rb

  • Add the require_user_logged_in to check if the Current.user is nil. If true, redirect to sign_in page with an alert message

    1. class ApplicationController < ActionController::Base
    2. before_action :set_current_user
    3. def set_current_user
    4. Current.user = User.find_by(id: session[:user_id]) if session[:user_id]
    5. end
    6. def require_user_logged_in
    7. return unless Current.user.nil?
    8. redirect_to sign_in_path,
    9. alert: 'You must be signed in to do that.'
    10. end
    11. end

Password Controller

Go Back to Contents

Create a new controller

  1. touch app/controllers/passwords_controller.rb

In app/controllers/passwords_controller.rb

  • Create a private helper to require a :user andn only allow for :password and :password_confirmation to be edited

    1. class PasswordsController < ApplicationController
    2. before_action :require_user_logged_in
    3. def edit
    4. end
    5. def update
    6. if Current.user.update(password_params)
    7. redirect_to root_path, notice: 'Password updated successfully!'
    8. else
    9. render :edit
    10. end
    11. end
    12. private
    13. def password_params
    14. params.require(:user).permit(:password, :password_confirmation)
    15. end
    16. end

Password Edit View

Go Back to Contents

Create a new view

  1. touch app/views/passwords/edit.html.erb

In app/views/passwords/edit.html.erb

  • Create a new form similar to the sign up form

    1. <h1>Edit Password</h1>
    2. <%= form_with model: @user, url: edit_password_path do |form| %>
    3. <% if @user.errors.any? %>
    4. <div class="alert alert-danger">
    5. <% @user.errors.full_messages.each do |message| %>
    6. <div>
    7. <%= message %>
    8. </div>
    9. <% end %>
    10. </div>
    11. <% end %>
    12. <div class="mb-3">
    13. <%= form.label :password%>
    14. <%= form.password_field :password, class: "form-control", placeholder: "password" %>
    15. </div>
    16. <div class="mb-3">
    17. <%= form.label :password_confirmation%>
    18. <%= form.password_field :password_confirmation, class: "form-control", placeholder: "password" %>
    19. </div>
    20. <div class="mb-3">
    21. <%= form.submit "Udpate Password", class: "btn btn-primary"%>
    22. </div>
    23. <% end %>

Update Navbar

Go Back to Contents

In app/views/shared/_navbar.html.erb

  • Update our navbar to add a link to edit_password path

    1. <li class="nav-item">
    2. <%= link_to Current.user.email, edit_password_path, class: "nav-link" %>
    3. </li>

Reset Password

Password Reset Route

Go Back to Contents

Create two new routes to display (GET) the form, and another route to submit (POST) the form.

In config/routes.rb

  1. get 'password/reset', to: 'password_resets#new'
  2. post 'password/reset', to: 'password_resets#create'

Password Reset Controller

Go Back to Contents

Create a new controller

  1. touch app/controllers/password_resets_controller.rb

In app/controllers/password_resets_controller.rb

  • Create new action to render our form (new.html.erb)
  • Create create action to receive the form

    • Call our PasswordMailer, set some params (user:) and call the reset method to send a reset email. Chain a deliver_later method to run in the background (async)
      • Another option would be deliver_now, but this will make our app a little slower
  1. class PasswordResetsController < ApplicationController
  2. def new
  3. end
  4. def create
  5. @user = User.find_by(email: params[:email])
  6. return unless @user.present?
  7. PasswordMailer.with(user: @user).reset.deliver_later
  8. redirect_to root_path,
  9. notice: 'If an account with that email was found, we have sent a link to reset your password.'
  10. end
  11. end
ACTIONMAILER

Go Back to Contents

We are going to use the CLI to create a mailer (a builtin rails mailer)

  1. rails g mailer Password reset
  2. # | | | └── email
  3. # | | └── Password Mailer
  4. # | └── mailer
  5. # └── generator
  6. # Running via Spring preloader in process 12749
  7. # create app/mailers/password_mailer.rb
  8. # invoke erb
  9. # create app/views/password_mailer
  10. # create app/views/password_mailer/reset.text.erb
  11. # create app/views/password_mailer/reset.html.erb
  12. # invoke test_unit
  13. # create test/mailers/password_mailer_test.rb
  14. # create test/mailers/previews/password_mailer_preview.rb
  • As we can see rails created app/mailers/password_mailer.rb for us
  • And also created two formats for us
    • app/views/password_mailer/reset.text.erb (text)
    • app/views/password_mailer/reset.html.erb (html)

Password Reset View

Go Back to Contents

Crate a new view

  1. touch app/views/password_resets/new.html.erb

In app/views/password_resets/new.html.erb

  1. <h1>Forgot your password?</h1>
  2. <%= form_with url: password_reset_path do |form| %>
  3. <div class="mb-3">
  4. <%= form.label :email%>
  5. <%= form.text_field :email, class: "form-control", placeholder: "your_email@email.com"%>
  6. </div>
  7. <div class="mb-3">
  8. <%= form.submit "Reset Password", class: "btn btn-primary"%>
  9. </div>
  10. <% end %>

Update Sign In Form

Go Back to Contents

In app/views/sessions/new.html.erb

  • Add a new link_to

    1. <div class="mb-3">
    2. <%= form.label :password%>
    3. <%= form.password_field :password, class: "form-control", placeholder: "password" %>
    4. <%= link_to "Forgot your password?", password_reset_path %>
    5. </div>

Password Mailer

Go Back to Contents

The Password Mailer is similar to any other controller. We can access the params[:user] that we passing through our controller (PasswordMailer.with(user: @user))

In app/mailers/password_mailer.rb

  1. class PasswordMailer < ApplicationMailer
  2. def reset
  3. @token = params[:user].signed_id(
  4. purpose: 'password_reset',
  5. expires_in: 15.minutes
  6. )
  7. mail to: params[:user].email
  8. end
  9. end

RESET TOKEN

Go Back to Contents

In rails we have a builtin function that can generate a random token for us, we just need to reference a user from our database.

We can also have expiration data in our tokens

  1. rails c
  2. # Running via Spring preloader in process 68865
  3. # Loading development environment (Rails 6.1.3)
  4. user = User.last
  5. # (0.5ms) SELECT sqlite_version(*)
  6. # User Load (0.1ms) SELECT "users".* FROM "users" ORDER BY "users"."id" DESC LIMIT ? [["LIMIT", 1]]
  7. # => #<User id: 5, email: "your_email@gmail.com", password_digest: [FILTERED], created_at: "2021-03-03 00:4...
  8. user.signed_id
  9. # => "eyJfcmFpbHMiOnsibWVzc2FnZSI6Ik5RPT0iLCJleHAiOm51bGwsInB1ciI6InVzZXIifX0=--61bd3be1c2303e3825bf7824712d781b310906370fb5e04af921ae3759f35313"
  10. user.to_global_id
  11. #<GlobalID:0x00007fb158ad15b8 @uri=#<URI::GID gid://scheduled-tweets/User/5>>
  12. user.to_global_id.to_s
  13. # => "gid://scheduled-tweets/User/5"
  • The signed_id generates an encrypted token with our users information, in this case the user.id ("gid://scheduled-tweets/User/5")
  • We can also pass an expiration to our signed_id
  1. user.signed_id(expires_in: 15.minutes)
  2. # => "eyJfcmFpbHMiOnsibWVzc2FnZSI6Ik5RPT0iLCJleHAiOiIyMDIxLTAzLTA0VDAwOjE4OjU4LjI4MVoiLCJwdXIiOiJ1c2VyIn19--b35f3bbb8935db0396d1bcc72d500363ba823cd64cf7cfadcded30a38ec73abf"
  • We can also specify a purpose for this token, so in our server we can specify to only accept a token that has the same purpose of the request
  1. user.signed_id(expires_in: 15.minutes, purpose: "password_reset")
  2. # => "eyJfcmFpbHMiOnsibWVzc2FnZSI6Ik5RPT0iLCJleHAiOiIyMDIxLTAzLTA0VDAwOjIwOjQxLjA3OFoiLCJwdXIiOiJ1c2VyL3Bhc3N3b3JkX3Jlc2V0In19--4b0b719794f414f8d73cff9e6ed66fccdb18982555a88c904fce836ac2ee4083"

ROUTES

Go Back to Contents

To reset our password, we need to create two new routes to handle the requests.

We are going to create a GET page to render the form, and a PATCH route to receive the form to update the password

  1. get 'password/reset/edit', to: 'password_resets#edit'
  2. patch 'password/reset/edit', to: 'password_resets#update'

HTML CONTENT

Go Back to Contents

In app/views/password_mailer/reset.html.erb

  • We are going to create a normal erb file adding the path to to our edit page
  • Instead of using password_reset_edit_path we are going to use password_reset_edit_url

    • The password_reset_edit_path generates a relative path (/something)
    • The password_reset_edit_url generates the complete path (www.yourwebsite.com/something)
      • The password_reset_edit_url accepts a parameter token to attach to the url that we are sending in our email
    1. Hi <%= params[:user].email%>,
    2. Someone request a reset of your password.
    3. If this was you, click on the link to reset your password. The link will expire automatically in 15 minutes.
    4. <%= link_to "Click Here To Reset Password Your Password", password_reset_edit_url(token: @token) %>

TEXT CONTENT

Go Back to Contents

In app/views/password_mailer/reset.text.erb

  1. Hi <%= params[:user].email%>,
  2. Someone request a reset of your password.
  3. If this was you, click on the link to reset your password. The link will expire automatically in 15 minutes.
  4. <%= password_reset_edit_url(token: @token) %>

DEVELOPMENT ENVIRONMENT

Go Back to Contents

In development mode, we need to config our environment so rails can get the correct host

In config/environments/development.rb

  • Add the following config

    1. config.action_mailer.default_url_options = { host: 'localhost:3000' }

LOGS

Go Back to Contents

In our rails logs, we can see that our email was successfully delivered

  1. [ActiveJob] [ActionMailer::MailDeliveryJob] [fd9e3930-3bfd-4639-ae1f-5f538740765c] Delivered mail 604029fad2409_110b24164-3f7@Rogers-MBP.phub.net.cable.rogers.com.mail (53.0ms)
  2. [ActiveJob] [ActionMailer::MailDeliveryJob] [fd9e3930-3bfd-4639-ae1f-5f538740765c] Date: Wed, 03 Mar 2021 19:29:46 -0500
  3. From: from@example.com
  4. To: your_email@gmail.com
  5. Message-ID: <604029fad2409_110b24164-3f7@Rogers-MBP.phub.net.cable.rogers.com.mail>
  6. Subject: Reset
  7. Mime-Version: 1.0
  8. Content-Type: multipart/alternative;
  9. boundary="--==_mimepart_604029facbb82_110b24164-4b3";
  10. charset=UTF-8
  11. Content-Transfer-Encoding: 7bit
  12. ----==_mimepart_604029facbb82_110b24164-4b3
  13. Content-Type: text/plain;
  14. charset=UTF-8
  15. Content-Transfer-Encoding: 7bit
  16. Hi your_email@gmail.com,
  17. Someone request a reset of your password.
  18. If this was you, click on the link to reset your password. The link will expire automatically in 15 minutes.
  19. http://localhost:3000/password/reset/edit?token=eyJfcmFpbHMiOnsibWVzc2FnZSI6Ik5RPT0iLCJleHAiOiIyMDIxLTAzLTA0VDAwOjQ0OjQ2LjgyMFoiLCJwdXIiOiJ1c2VyL3Bhc3N3b3JkX3Jlc2V0In19--817cd5bba1257268efcc810b37077a608a66604ec0d058fd324a171543d7f726
  20. ----==_mimepart_604029facbb82_110b24164-4b3
  21. Content-Type: text/html;
  22. charset=UTF-8
  23. Content-Transfer-Encoding: 7bit
  24. <!DOCTYPE html>
  25. <html>
  26. <head>
  27. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  28. <style>
  29. /* Email styles need to be inline */
  30. </style>
  31. </head>
  32. <body>
  33. Hi your_email@gmail.com,
  34. Someone request a reset of your password.
  35. If this was you, click on the link to reset your password. The link will expire automatically in 15 minutes.
  36. <a href="http://localhost:3000/password/reset/edit?token=eyJfcmFpbHMiOnsibWVzc2FnZSI6Ik5RPT0iLCJleHAiOiIyMDIxLTAzLTA0VDAwOjQ0OjQ2LjgyMFoiLCJwdXIiOiJ1c2VyL3Bhc3N3b3JkX3Jlc2V0In19--817cd5bba1257268efcc810b37077a608a66604ec0d058fd324a171543d7f726">Click Here To Reset Password Your Password</a>
  37. </body>
  38. </html>
  39. ----==_mimepart_604029facbb82_110b24164-4b3--
  40. Rendered main/index.html.erb within layouts/application (Duration: 0.5ms | Allocations: 176)
  41. [ActiveJob] [ActionMailer::MailDeliveryJob] [fd9e3930-3bfd-4639-ae1f-5f538740765c] Performed ActionMailer::MailDeliveryJob (Job ID: fd9e3930-3bfd-4639-ae1f-5f538740765c) from Async(default) in 81.79ms
  42. [Webpacker] Everything's up-to-date. Nothing to do
  43. Rendered shared/_navbar.html.erb (Duration: 0.4ms | Allocations: 162)
  44. Rendered shared/_flash.html.erb (Duration: 0.1ms | Allocations: 59)
  45. Rendered layout layouts/application.html.erb (Duration: 32.5ms | Allocations: 4025)
  46. Completed 200 OK in 43ms (Views: 36.3ms | ActiveRecord: 0.0ms | Allocations: 6980)
  • As we can see we have our reset_url in the middle of the logs

    1. http://localhost:3000/password/reset/edit?token=eyJfcmFpbHMiOnsibWVzc2FnZSI6Ik5RPT0iLCJleHAiOiIyMDIxLTAzLTA0VDAwOjQ0OjQ2LjgyMFoiLCJwdXIiOiJ1c2VyL3Bhc3N3b3JkX3Jlc2V0In19--817cd5bba1257268efcc810b37077a608a66604ec0d058fd324a171543d7f726

RESET PASSWORD CONTROLLER

Go Back to Contents

We now need to build the action to edit and update the password

In app/controllers/password_resets_controller.rb

  • Edit Password

    • We are going to use find_signed function instead of normal find_by
    • The find_signed checks if our token is still valid, and also gets the user.id from the token, and checks if the purpose is correct too

      1. def edit
      2. @user = User.find_signed(params[:token], purpose: 'password_reset')
      3. end
      • The find_signed method has another version that with an ! mark (find_signed!) and this version throws an error if the token is expired
      • Lets refactor our action to handle the error

        1. def edit
        2. @user = User.find_signed!(params[:token], purpose: 'password_reset')
        3. rescue ActiveSupport::MessageVerifier::InvalidSignature
        4. redirect_to sign_in_path, alert: 'Your token has expired. Please try again'
        5. end
  • Update Password

    • To update the password, we need to send the @user instance that we are sending to our view when we loaded the edit page
    • We need to do that, because we are going to create a private helper that requires a user so we can update the password

      1. def update
      2. @user = User.find_signed!(params[:token], purpose: 'password_reset')
      3. if @user.update(password_params)
      4. redirect_to sign_in_path,
      5. notice: 'Your password has been reseted successfully. Please sign in again.'
      6. else
      7. render :edit
      8. end
      9. end
      10. private
      11. def password_params
      12. params.require(:user).permit(:password, :password_confirmation)
      13. end

RESET PASSWORD VIEW

Go Back to Contents

Create the edit.html.erb

  1. touch app/views/password_resets/edit.html.erb

In app/views/password_resets/edit.html.erb

  • Without sending the @user instance

    1. <h1>Reset your password</h1>
    2. <%= form_with url: password_reset_edit_path(token: params[:token]) do |form| %>
    3. <% if form.object.errors.any? %>
    4. <div class="alert alert-danger">
    5. <% form.object.errors.full_messages.each do |message| %>
    6. <div>
    7. <%= message %>
    8. </div>
    9. <% end %>
    10. </div>
    11. <% end %>
    12. <div class="mb-3">
    13. <%= form.label :password%>
    14. <%= form.password_field :password, class: "form-control", placeholder: "password" %>
    15. </div>
    16. <div class="mb-3">
    17. <%= form.label :password_confirmation%>
    18. <%= form.password_field :password_confirmation, class: "form-control", placeholder: "password" %>
    19. </div>
    20. <div class="mb-3">
    21. <%= form.submit "Reset Password", class: "btn btn-primary"%>
    22. </div>
    23. <% end %>
  • Sending the @user instance nested in our form (model: @user)

    1. <h1>Reset your password</h1>
    2. <%= form_with model: @user, url: password_reset_edit_path(token: params[:token]) do |form| %>
    3. <% if form.object.errors.any? %>
    4. <div class="alert alert-danger">
    5. <% form.object.errors.full_messages.each do |message| %>
    6. <div>
    7. <%= message %>
    8. </div>
    9. <% end %>
    10. </div>
    11. <% end %>
    12. <div class="mb-3">
    13. <%= form.label :password%>
    14. <%= form.password_field :password, class: "form-control", placeholder: "password" %>
    15. </div>
    16. <div class="mb-3">
    17. <%= form.label :password_confirmation%>
    18. <%= form.password_field :password_confirmation, class: "form-control", placeholder: "password" %>
    19. </div>
    20. <div class="mb-3">
    21. <%= form.submit "Reset Password", class: "btn btn-primary"%>
    22. </div>
    23. <% end %>

Generate Environment Variables

Go Back to Contents

Rails has a builtin environment system that can store and encrypt our environment variables

  1. rails credentials:edit --environment=development
  2. # Adding config/credentials/development.key to store the encryption key: fasdfasdfasdfasdfasdfasdfasdfasdfa
  3. # Save this in a password manager your team can access.
  4. # If you lose the key, no one, including you, can access anything encrypted with it.
  5. # create config/credentials/development.key
  6. # Ignoring config/credentials/development.key so it won't end up in Git history:
  7. # append .gitignore
  • This will open a file so we can add our environment variables

In 2312312.development.yml

  • Add your twitter keys

    1. twitter:
    2. api_key: faljsdkfjaljsdfjlsoiifjo
    3. api_secret: fasdfasdg123123klkhkjhlafkhdflaksjdhflk

On Terminal we can get our api keys

  1. rails c
  2. Rails.application.credentials.twitter
  3. # => {:api_key=>"faljsdkfjaljsdfjlsoiifjo", :api_secret=>"fasdfasdg123123klkhkjhlafkhdflaksjdhflk"}
  • We can get an individual keys from the hash using dig

    1. Rails.application.credentials.dig(:twitter, :api_key)
    2. # => "faljsdkfjaljsdfjlsoiifjo"

Omniauth-Twitter

Go Back to Contents

Add the following gems

  1. bundle add omniauth-twitter omniauth-rails_csrf_protection
  • We need to add omniauth-rails_csrf_protection to be more secure, previously we had to make a GET request, now we do a POST request and send some information with the request

Initializers - Middleware

Go Back to Contents

Then in our initializes folder, we have to create a new file called omniauth.rb

  1. touch config/initializers/omniauth.rb

In config/initializers/omniauth.rb

  • Here we are going to setup our OmniAuth, to do so we need o call Rails.application.config.middleware.use.
  • Then we pass the name of our application OmniAuth::Builder
  • Then we specify the provider as :twitter (this will make rails look up for the twitter gem for us)

    1. Rails.application.config.middleware.use OmniAuth::Builder do
    2. provider :twitter,
    3. Rails.application.credentials.dig(:twitter, :api_key),
    4. Rails.application.credentials.dig(:twitter, :api_secret)
    5. end

OmniAuth Routes (Special Routes)

Go Back to Contents

If we check our rails routes [http://localhost:3000/rails/info/routes] we won’t see an OmniAuth route.

If we try to visit [http://localhost:3000/auth/twitter] we get the following error

No route matches [GET] “/auth/twitter”

This is because OmniAuth only accepts POST request

Index Page

Go Back to Contents

In app/views/main/index.html.erb

  • Let’s create a button so we can make a post request to our /auth/twitter

    1. <div class="d-flex align-items-center justify-content-center">
    2. <h1 class="mt-4">Welcome to Scheduled Tweets</h1>
    3. </div>
    4. <%= button_to "Connect Twitter", "/auth/twitter", method: :post, class: "btn btn-primary" %>

After the user successfully connected his twitter account to our app, twitter will redirect to our callback route, with the oauth_token and oauth_verifier params

  1. http://localhost:3000/auth/twitter/callback?oauth_token=QGoDzgAAAAABNU53AAABd_rvTH8&oauth_verifier=iw9nXvXqKhXnNg1QW9UWlKHQmUNp9qYZ

We now can create our new route auth/twitter/callback to get those params and save in our database to connect our twitter account

Twitter Callback Route

Go Back to Contents

In config/routes.rb

  • Add our callback route

    1. get 'auth/twitter/callback', to: 'omniauth_callbacks#twitter'

OmniAuth Controller

Go Back to Contents

Create a new controller omniauth_callbacks_controller.rb

In app/controllers/omniauth_callbacks_controller.rb

  1. class OmniauthCallbacksController < ApplicationController
  2. def twitter
  3. render plain: 'Success!'
  4. end
  5. end

Twitter Model

Go Back to Contents

Generate a Twitter model

  1. rails g model TwitterAccount user:belongs_to name username image token secret
  2. # Running via Spring preloader in process 76050
  3. # invoke active_record
  4. # create db/migrate/20210304034024_create_twitter_accounts.rb
  5. # create app/models/twitter_account.rb
  6. # invoke test_unit
  7. # create test/models/twitter_account_test.rb
  8. # create test/fixtures/twitter_accounts.yml
  9. rails db:migrate
  10. # == 20210304034024 CreateTwitterAccounts: migrating ============================
  11. # -- create_table(:twitter_accounts)
  12. # -> 0.0030s
  13. # == 20210304034024 CreateTwitterAccounts: migrated (0.0031s) ===================
  • user:belongs_to, each twitter account points to a user

In app/models/twitter_account.rb

  • Rails generated for us

    1. class TwitterAccount < ApplicationRecord
    2. belongs_to :user
    3. end

In app/models/user.rb

  • We can do the same thing to connect our User model with TwitterAccount model

    • ATTENTION the :twitter_accounts is plural
    • This connection allows us to CRUD our twitter account through the User model
    1. class User < ApplicationRecord
    2. has_many :twitter_accounts
    3. has_secure_password
    4. validates :email,
    5. presence: true,
    6. format: {
    7. with: /\A[^@\s]+@[^@\s]+\z/,
    8. message: 'must be a valid email address'
    9. }
    10. end

On Terminal

  1. rails c
  2. User.last.twitter_accounts
  3. # (0.7ms) SELECT sqlite_version(*)
  4. # User Load (0.4ms) SELECT "users".* FROM "users" ORDER BY "users"."id" DESC LIMIT ? [["LIMIT", 1]]
  5. # TwitterAccount Load (0.4ms) SELECT "twitter_accounts".* FROM "twitter_accounts" WHERE "twitter_accounts"."user_id" = ? /* loading for inspect */ LIMIT ? [["user_id", 5], ["LIMIT", 11]]
  6. # => #<ActiveRecord::Associations::CollectionProxy []>
  • This query will only get the twitter account for our current user

Update OmniAuth Controller

Go Back to Contents

In app/controllers/omniauth_callbacks_controller.rb

  • We are going to create a new record in our TwitterAccount table
  • We don’t need to define the user.id, Rails already know because of our association

    1. Current.user.twitter_accounts.create()
  • OmniAuth gives us a hash of all of the things that the API sends back

    • So we can create a method auth that can help us get all these values

      1. def auth
      2. request.env['omniauth.auth']
      3. end
  1. class OmniauthCallbacksController < ApplicationController
  2. def twitter
  3. # Prints all the values of the auth hash
  4. Rails.logger.info auth
  5. twitter_account = Current.user.twitter_accounts.where(username: auth.info.nickname).first_or_initialize
  6. twitter_account.update(
  7. name: auth.info.name,
  8. username: auth.info.nickname,
  9. image: auth.info.image,
  10. token: auth.credentials.token,
  11. secret: auth.credentials.secret
  12. )
  13. redirect_to root_path, notice: 'Successfully connected your account'
  14. end
  15. def auth
  16. request.env['omniauth.auth']
  17. end
  18. end
  • Instead of using .create() to add twitter info in the database. We can check if there is a username with this nickname, .first_or_initialize will use the first or create one if not found
  • Then we can update de document

In app/models/twitter_account.rb

  • We can update our model to validate the :username saving

    1. class TwitterAccount < ApplicationRecord
    2. belongs_to :user
    3. validates :username, uniqueness: true
    4. end

On Terminal

  1. rails c
  2. User.last.twitter_accounts.count
  3. # User Load (0.1ms) SELECT "users".* FROM "users" ORDER BY "users"."id" DESC LIMIT ? [["LIMIT", 1]]
  4. # (0.2ms) SELECT COUNT(*) FROM "twitter_accounts" WHERE "twitter_accounts"."user_id" = ? [["user_id", 5]]
  5. # => 1

Twitter Accounts Page

Twitter Accounts Routes - Resources

Go Back to Contents

In config/routes.rb

  • Add the :twitter_accounts resource
  • The resources will generate all the CRUD operations routes for us
  • And also adds automatically the to: to map to our twitter controller

    1. resources :twitter_accounts
    2. # get 'twitter_accounts/:id'
    3. # delete 'twitter_accounts/:id'
    4. # new, create, update

Twitter Accounts Controller

Go Back to Contents

Create a new controller

  1. touch app/controllers/twitter_accounts_controller.rb

In app/controllers/twitter_accounts_controller.rb

  1. class TwitterAccountsController < ApplicationController
  2. before_action :require_user_logged_in
  3. def index
  4. @twitter_accounts = Current.user.twitter_accounts
  5. end
  6. def destroy
  7. @twitter_account = Current.user.twitter_accounts.find(params[:id])
  8. @twitter_account.destroy
  9. redirect_to twitter_accounts_path
  10. end
  11. end
  • Add a before_action to require a user before executing anything

Twitter Accounts View

Go Back to Contents

Create the twitter account index.html.erb

  1. touch app/views/twitter_accounts/index.html.erb

In app/views/twitter_accounts/index.html.erb

  1. <div class="d-flex align-items-center justify-content-between">
  2. <h1>Twitter Accounts</h1>
  3. <%= link_to "Connect a Twitter Account", "/auth/twitter", method: :post, class: "btn btn-primary"%>
  4. </div>
  5. <% @twitter_accounts.each do |twitter_account| %>
  6. <div class="d-flex align-items-center mb-4">
  7. <div class="me-4">
  8. <%= image_tag twitter_account.image, class: "rounded-circle" %>
  9. <%= link_to "@#{twitter_account.username}", "https://twitter.com/#{twitter_account.username}", target: :_blank %>
  10. </div>
  11. <%= button_to "Disconnect", twitter_account, method: :delete, data: { confirm: "Are you sure?" } %>
  12. </div>
  13. <% end %>
  • In rails, we don’t need to provide the route to delete/disconnect our twitter account, we just need to pass the model twitter_account and rails will figure out to find the correct user.id to use ( delete: 'twitter_accounts/:id')
  • Then we can pass the data object, that will prompt the use to confirm

Update OmniAuth Controller

Go Back to Contents

Update our twitter callback action to redirect to our new route/page twitter_accounts

In app/controllers/omniauth_callbacks_controller.rb

  1. class OmniauthCallbacksController < ApplicationController
  2. def twitter
  3. # Prints all the values of the auth hash
  4. Rails.logger.info auth
  5. twitter_account = Current.user.twitter_accounts.where(username: auth.info.nickname).first_or_initialize
  6. twitter_account.update(
  7. name: auth.info.name,
  8. username: auth.info.nickname,
  9. image: auth.info.image,
  10. token: auth.credentials.token,
  11. secret: auth.credentials.secret
  12. )
  13. redirect_to twitter_accounts_path, notice: 'Successfully connected your account'
  14. end
  15. def auth
  16. request.env['omniauth.auth']
  17. end
  18. end

before_action To Set Common Variables

Go Back to Contents

We can set a before_action to set a common variable across actions

In app/controllers/twitter_accounts_controller.rb

  • Let’s update our destroy action
  • Create a private method set_twitter_account
  • Require before_action for only the destroy action

    1. class TwitterAccountsController < ApplicationController
    2. before_action :require_user_logged_in
    3. before_action :set_twitter_account, only: [:destroy]
    4. def index
    5. @twitter_accounts = Current.user.twitter_accounts
    6. end
    7. def destroy
    8. @twitter_account.destroy
    9. redirect_to twitter_accounts_path,
    10. notice: "Successfully disconnected @#{@twitter_account.username}"
    11. end
    12. private
    13. def set_twitter_account
    14. @twitter_account = Current.user.twitter_accounts.find(params[:id])
    15. end
    16. end

Tweets

Tweets Model

Go Back to Contents

Generate a new Tweet model to schedule our tweets

  1. rails g model Tweet user:belongs_to twitter_account:belongs_to body:text publish_at:datetime tweet_id:string
  2. # Running via Spring preloader in process 67929
  3. # invoke active_record
  4. # create db/migrate/20210306030431_create_tweets.rb
  5. # create app/models/tweet.rb
  6. # invoke test_unit
  7. # create test/models/tweet_test.rb
  8. # create test/fixtures/tweets.yml
  9. rails db:migrate
  10. # == 20210306030431 CreateTweets: migrating =====================================
  11. # -- create_table(:tweets)
  12. # -> 0.0041s
  13. # == 20210306030431 CreateTweets: migrated (0.0044s) ============================
  • belongs_to creates an association between Tweet and User and twitter_account
  • In other words, a tweet belongs to a user and twitter account
  • The tweet_id is going to be our confirmation that our tweet was sent
User Model

Go Back to Contents

In app/models/user.rb

  • We need to make the connection with our tweets model
  • To do so, add has_many :tweets
Twitter Account Model

Go Back to Contents

In app/models/twitter_account.rb

  • We need to make the connection with our tweets model
  • To do so, add has_many :tweets and we can also add dependent: :destroy this means that when the user disconnects his account, we also delete his scheduled tweets

    1. class TwitterAccount < ApplicationRecord
    2. belongs_to :user
    3. has_many :tweets, dependent: :destroy
    4. end

Tweets Routes

Go Back to Contents

In config/routes.rb

  • Add a the tweet resources

    1. resources :tweets

Tweets Controller

Go Back to Contents

Create a new controller

  1. touch app/controllers/tweets_controller.rb

In app/controllers/tweets_controller.rb

  1. class TweetsController < ApplicationController
  2. before_action :require_user_logged_in
  3. def index
  4. @tweets = Current.user.tweets
  5. end
  6. def new
  7. @tweet = Tweet.new
  8. end
  9. def create
  10. @tweet = Current.user.tweets.create(tweet_params)
  11. if @tweet.save
  12. redirect_to tweets_path, notice: 'Tweet was schedule successfully'
  13. else
  14. render :new
  15. end
  16. end
  17. private
  18. def tweet_params
  19. params.require(:tweet).permit(:twitter_account_id, :body, :publish_at)
  20. end
  21. end

Tweets Index

Go Back to Contents

Create a new view

  1. touch touch app/views/tweets/index.html.erb

In app/views/tweets/index.html.erb

  1. <div class="d-flex justify-content-between align-items-center">
  2. <h1>Tweets</h1>
  3. <% if Current.user.twitter_accounts.any? %>
  4. <%= link_to "Schedule a Tweet", new_tweet_path, class: "btn btn-primary" %>
  5. <% end %>
  6. </div>
  7. <% if Current.user.twitter_accounts.none? %>
  8. <%= link_to "Connect Your Twitter Account", "/auth/twitter", method: :post, class: "btn btn-primary"%>
  9. <% end %>

Update Navbar

Go Back to Contents

In app/views/shared/_navbar.html.erb

  • Change our Home link to Tweets

    1. <li class="nav-item">
    2. <%= link_to "Tweets", tweets_path, class: "nav-link active" %>
    3. </li>

Tweets Model

Go Back to Contents

In app/models/tweet.rb

  • Add a validation to our form
  • We can also add something after_initialize, in this case it will check if the publish_at has already been set, otherwise, set the datetime for 24hrs from now

    1. class Tweet < ApplicationRecord
    2. belongs_to :user
    3. belongs_to :twitter_account
    4. validates :body, length: { minimum: 1, maximum: 280 }
    5. validates :publish_at, presence: true
    6. after_initialize do
    7. self.publish_at ||= 24.hours.from_now
    8. end
    9. end

Tweets New

Go Back to Contents

Create the new page

  1. touch app/views/tweets/new.html.erb
  2. touch app/views/shared/_form_errors.html.erb

In app/views/shared/_form_errors.html.erb

  • Create our shared error msg

    1. <% if form.object.errors.any? %>
    2. <div class="alert alert-danger">
    3. <% form.object.errors.full_messages.each do |message| %>
    4. <div>
    5. <%= message %>
    6. </div>
    7. <% end %>
    8. </div>
    9. <% end %>

In app/views/tweets/new.html.erb

  1. <h1>Schedule a Tweet</h1>
  2. <%= form_with model: @tweet do |form| %>
  3. <div class="mb-3">
  4. <%= form.label :twitter_account_id %>
  5. <%= form.collection_select :twitter_account_id, Current.user.twitter_accounts, :id, :username, {}, { class: "form-control" } %>
  6. <%= link_to "Connect Your Twitter Account", "/auth/twitter" %>
  7. </div>
  8. <div class="mb-3">
  9. <%= form.label :body %>
  10. <%= form.text_area :body, class: "form-control" %>
  11. </div>
  12. <div class="mb-3">
  13. <%= form.label :publish_at %>
  14. <div class="form-control">
  15. <%= form.datetime_select :publish_at %>
  16. </div>
  17. </div>
  18. <%= form.button "Schedule", class: "btn btn-primary" %>
  19. <% end %>

Tweets Render Partial

Tweets Index

Go Back to Contents

In app/views/tweets/index.html.erb

  • If we render an object

    1. <%= render @tweets %>
    • Rails will render each item of the object (query result), we just need to create a partial called _tweet.html.erb
Tweets Partial

Go Back to Contents

Create a new partial to render our query object (@tweets)

  1. touch app/views/tweets/_tweet.html.erb

In app/views/tweets/_tweet.html.erb

  1. <div class="mb-3 card card-body">
  2. <%= tweet.body %>
  3. <div class="me-4">
  4. <%= image_tag tweet.twitter_account.image, class: "rounded-circle" %>
  5. <%= link_to "@#{tweet.twitter_account.username}", "https://twitter.com/#{tweet.twitter_account.username}", target: :_blank %>
  6. </div>
  7. </div>

Tweets Model

Go Back to Contents

In app/models/tweet.rb

  • Add a new method to our tweet controller to check if our tweet has been published or not

    1. def published?
    2. tweet_id?
    3. end
    • Because we added a question mark in the end of the method, rails will return true/false instead of the normal behavior value or nil
    • This method will check the tweet_id column, if has a value, then it’s published

Tweet Form Partial

Go Back to Contents

Create a new partial to clean our code

  1. touch app/views/tweets/_form.html.erb

In app/views/tweets/_form.html.erb

  • We just need to update tweet variable, instead of using the instance @tweet we are going to use a local variable tweet that we are going to pass it when we invoke the form
  1. <%= form_with model: tweet do |form| %>
  2. <%= render "shared/form_errors", form: form%>
  3. <div class="mb-3">
  4. <%= form.label :twitter_account_id %>
  5. <%= form.collection_select :twitter_account_id, Current.user.twitter_accounts, :id, :username, {}, { class: "form-control" } %>
  6. <%= link_to "Connect Your Twitter Account", "/auth/twitter" %>
  7. </div>
  8. <div class="mb-3">
  9. <%= form.label :body %>
  10. <%= form.text_area :body, class: "form-control" %>
  11. </div>
  12. <div class="mb-3">
  13. <%= form.label :publish_at %>
  14. <div class="form-control">
  15. <%= form.datetime_select :publish_at %>
  16. </div>
  17. </div>
  18. <%= form.button "Schedule", class: "btn btn-primary" %>
  19. <% if form.object.persisted? %>
  20. <%= link_to "Delete", form.object, method: :delete, data: { confirm: "Are you sure you want to delete this tweet?" }, class: "btn btn-outline-danger" %>
  21. <% end %>
  22. <% end %>

Update Tweet Index

Go Back to Contents

Update the tweet index to use our partial _form

In app/views/tweets/new.html.erb

  1. <h1>Schedule a Tweet</h1>
  2. <%= render "form", tweet: @tweet %>

Tweets Edit Schedule

Go Back to Contents

Create a new view to edit a tweet

  1. touch app/views/tweets/edit.html.erb

In app/views/tweets/edit.html.erb

  1. <h1>Edit Scheduled Tweet</h1>
  2. <%= render "form", tweet: @tweet %>

Update Tweets Controller

Go Back to Contents

In app/controllers/tweets_controller.rb

  • We can create a new private method to help use to set the current Tweet
  • And before each action we set the current tweet

    1. ...
    2. before_action :set_tweet, only: %i[show edit update destroy]
    3. ...
    4. def edit
    5. end
    6. def update
    7. if @tweet.update(tweet_params)
    8. redirect_to tweets_path, notice: 'Twee has been updated successfully'
    9. else
    10. render :edit
    11. end
    12. end
    13. def destroy
    14. @tweet.destroy
    15. redirect_to tweets_path, notice: 'Tweet has been successfully unscheduled'
    16. end
    17. private
    18. ...
    19. def set_tweet
    20. @tweet = Current.user.tweets.find(params[:id])
    21. end

Twitter Gem

Go Back to Contents

Instead of coding and making calls to Twitter’s API directly, we are going to use third party gem to help use to connect to Twitter’s API

  1. bundle add twitter

Twitter Controller

Go Back to Contents

In app/models/twitter_account.rb

  • Add a new function called client

    1. class TwitterAccount < ApplicationRecord
    2. belongs_to :user
    3. has_many :tweets, dependent: :destroy
    4. validates :username, uniqueness: true
    5. def client
    6. Twitter::REST::Client.new do |config|
    7. config.consumer_key = Rails.application.credentials.dig(
    8. :twitter,
    9. :api_key
    10. )
    11. config.consumer_secret = Rails.application.credentials.dig(
    12. :twitter,
    13. :api_secret
    14. )
    15. config.access_token = token
    16. config.access_token_secret = secret
    17. end
    18. end
    19. end

Tweet Model

Go Back to Contents

Create a new function called publish_to_twitter! to post a tweet using the twitter gem

We added the ! in the end, just to help us identify that this function makes an external API call

In app/models/tweet.rb

  1. def publish_to_twitter!
  2. tweet = twitter_account.client.update(body)
  3. update(tweet_id: tweet.id)
  4. end

Background Job

Go Back to Contents

Rails has a builtin background jobs app/jobs/application_job.rb

Let’s create a new one for Tweets

  1. rails g job Tweet
  2. # Running via Spring preloader in process 84498
  3. # invoke test_unit
  4. # create test/jobs/tweet_job_test.rb
  5. # create app/jobs/tweet_job.rb

In app/jobs/tweet_job.rb

  • Update the perform method
  • Check if the tweet has already been published, if yes return
  • Check if publish_at datetime greater than current datetime, if yes return
  • For last publish the tweet

    1. class TweetJob < ApplicationJob
    2. queue_as :default
    3. def perform(tweet)
    4. return if tweet.published?
    5. # Rescheduled a tweet to the future
    6. return if tweet.publish_at > Time.current
    7. tweet.publish_to_twitter!
    8. end
    9. end

Tweet Model

Go Back to Contents

Create a new method that will run every time some has successfully saved in our database (CREATE and UPDATE)

In app/models/tweet.rb

  • The TweetJob.set adds a new job to the queue, we need to pass wait_until as argument to set the datetime and chain perform_later to attach the object that we want to use later

    1. after_save_commit do
    2. if publish_at_previously_changed?
    3. TweetJob.set(wait_until: publish_at).perform_later(self)
    4. end
    5. end

Sidekiq

Go Back to Contents

Our current background job doesn’t persist the queue if the server restarts

Let’s add sidekiq to fix this problem for us

  1. bundle add sidekiq
  2. bundle exec sidekiq -e development

Config Rails + Sidekiq

Go Back to Contents

In config/environments/development.rb

  • Add the following line to the environment configuration

    1. config.active_job.queue_adapter = :sidekiq

In config/environments/production.rb

  • Add the same configuration to production

    1. config.active_job.queue_adapter = :sidekiq

Check If It’s Working

Go Back to Contents

On Terminal 1

  1. rails c
  2. TweetJob.perform_later(Tweet.last)
  3. # (0.4ms) SELECT sqlite_version(*)
  4. # Tweet Load (0.1ms) SELECT "tweets".* FROM "tweets" ORDER BY "tweets"."id" DESC LIMIT ? [["LIMIT", 1]]
  5. # Enqueued TweetJob (Job ID: fe63464e-bfca-4ce8-ad26-f4986e5690ca) to Sidekiq(default) with arguments: #<GlobalID:0x00007fec512b81b8 @uri=#<URI::GID gid://scheduled-tweets/Tweet/7>>
  6. # => #<TweetJob:0x00007fec512d9598 @arguments=[#<Tweet id: 7, user_id: 5, twitter_account_id: 4, body: "New Twitter API Job", publish_at: "2021-03-15 07:49:00.000000000 +0000", tweet_id: nil, created_at: "2021-03-09 00:59:31.911678000 +0000", updated_at: "2021-03-09 01:03:59.717806000 +0000">], @job_id="fe63464e-bfca-4ce8-ad26-f4986e5690ca", @queue_name="default", @priority=nil, @executions=0, @exception_executions={}, @timezone="UTC", @provider_job_id="42ab7c66ed679d503569e978">

On Terminal 2

  1. bundle exec sidekiq -e development
  2. # 2021-03-09T01:19:48.892Z pid=87485 tid=1wg1 INFO: Booted Rails 6.1.3 application in development environment
  3. # 2021-03-09T01:19:48.892Z pid=87485 tid=1wg1 INFO: Running in ruby 2.7.2p137 (2020-10-01 revision 5445e04352) [x86_64-darwin19]
  4. # 2021-03-09T01:19:48.892Z pid=87485 tid=1wg1 INFO: See LICENSE and the LGPL-3.0 for licensing details.
  5. # 2021-03-09T01:19:48.892Z pid=87485 tid=1wg1 INFO: Upgrade to Sidekiq Pro for more features and support: https://sidekiq.org
  6. # 2021-03-09T01:19:48.892Z pid=87485 tid=1wg1 INFO: Booting Sidekiq 6.1.3 with redis options {}
  7. # 2021-03-09T01:19:48.894Z pid=87485 tid=1wg1 INFO: Starting processing, hit Ctrl-C to stop
  8. # Added new job
  9. # 2021-03-09T01:19:50.739Z pid=87485 tid=20i9 class=TweetJob jid=d32d7b46438dc915228dbc1f INFO: start
  10. # 2021-03-09T01:19:50.865Z pid=87485 tid=20i9 class=TweetJob jid=d32d7b46438dc915228dbc1f elapsed=0.127 INFO: done