こんにちは。
Rails8がいよいよ公式リリースされましたね。
私も早速色々試してみている状態ですが皆さんはいかがでしょうか?
個人的には、Rails8での注目ポイントはSolid QueueやSolid Cache、Solid Cableなどです。
Redis前提になっていたこれまでのAction CableやSidekiqなどが、これらの新機能に統合されることで、よりシンプルに開発ができるようになりそうです。
Redisもライセンス変更があったりしたので、今後どうなると様子を見ていかねばいけないなあと思っていた中、このような新機能が追加されることで、Redisに依存しない開発ができるようになるのはとてもありがたいです。
さて、今回はRails8で導入されたAuthenticationを試してみたが導入を見送った話をお伝えします。
Rails8で導入されたAuthentication
Rails8で導入されたAuthenticationとは、Railsがデフォルトで提供する認証機能の作成が簡単になる機能です。
bin/rails generate authentication
コマンドを実行することで、簡単に認証機能を作成することができます。
作成されるファイル
bin/rails generate authentication
コマンドを実行すると、以下のファイルが作成 or 更新されます。
module ApplicationCable
class Connection < ActionCable::Connection::Base
identified_by :current_user
def connect
set_current_user || reject_unauthorized_connection
end
private
def set_current_user
if session = Session.find_by(id: cookies.signed[:session_id])
self.current_user = session.user
end
end
end
end
class ApplicationController < ActionController::Base
include Authentication
# ...
end
module Authentication
extend ActiveSupport::Concern
included do
before_action :require_authentication
helper_method :authenticated?
end
class_methods do
def allow_unauthenticated_access(**options)
skip_before_action :require_authentication, **options
end
end
private
def authenticated?
resume_session
end
def require_authentication
resume_session || request_authentication
end
def resume_session
Current.session ||= find_session_by_cookie
end
def find_session_by_cookie
Session.find_by(id: cookies.signed[:session_id])
end
def request_authentication
session[:return_to_after_authenticating] = request.url
redirect_to new_session_path
end
def after_authentication_url
session.delete(:return_to_after_authenticating) || root_url
end
def start_new_session_for(user)
user.sessions.create!(user_agent: request.user_agent, ip_address: request.remote_ip).tap do |session|
Current.session = session
cookies.signed.permanent[:session_id] = { value: session.id, httponly: true, same_site: :lax }
end
end
def terminate_session
Current.session.destroy
cookies.delete(:session_id)
end
end
class PasswordsController < ApplicationController
allow_unauthenticated_access
before_action :set_user_by_token, only: %i[ edit update ]
def new
end
def create
if user = User.find_by(email_address: params[:email_address])
PasswordsMailer.reset(user).deliver_later
end
redirect_to new_session_path, notice: "Password reset instructions sent (if user with that email address exists)."
end
def edit
end
def update
if @user.update(params.permit(:password, :password_confirmation))
redirect_to new_session_path, notice: "Password has been reset."
else
redirect_to edit_password_path(params[:token]), alert: "Passwords did not match."
end
end
private
def set_user_by_token
@user = User.find_by_password_reset_token!(params[:token])
rescue ActiveSupport::MessageVerifier::InvalidSignature
redirect_to new_password_path, alert: "Password reset link is invalid or has expired."
end
end
class SessionsController < ApplicationController
allow_unauthenticated_access only: %i[ new create ]
rate_limit to: 10, within: 3.minutes, only: :create, with: -> { redirect_to new_session_url, alert: "Try again later." }
def new
end
def create
if user = User.authenticate_by(params.permit(:email_address, :password))
start_new_session_for user
redirect_to after_authentication_url
else
redirect_to new_session_path, alert: "Try another email address or password."
end
end
def destroy
terminate_session
redirect_to new_session_path
end
end
class PasswordsMailer < ApplicationMailer
def reset(user)
@user = user
mail subject: "Reset your password", to: user.email_address
end
end
class Current < ActiveSupport::CurrentAttributes
attribute :session
delegate :user, to: :session, allow_nil: true
end
class Session < ApplicationRecord
belongs_to :user
end
class User < ApplicationRecord
has_secure_password
has_many :sessions, dependent: :destroy
normalizes :email_address, with: ->(e) { e.strip.downcase }
end
<p>
You can reset your password within the next 15 minutes on
<%= link_to "this password reset page", edit_password_url(@user.password_reset_token) %>.
</p>
You can reset your password within the next 15 minutes on this password reset page:
<%= edit_password_url(@user.password_reset_token) %>.
Rails.application.routes.draw do
resource :session
resources :passwords, param: :token
# ...
end
class CreateUsers < ActiveRecord::Migration[8.0]
def change
create_table :users do |t|
t.string :email_address, null: false
t.string :password_digest, null: false
t.timestamps
end
add_index :users, :email_address, unique: true
end
end
class CreateSessions < ActiveRecord::Migration[8.0]
def change
create_table :sessions do |t|
t.references :user, null: false, foreign_key: true
t.string :ip_address
t.string :user_agent
t.timestamps
end
end
end
<% password_digest = BCrypt::Password.create("password") %>
one:
email_address: one@example.com
password_digest: <%= password_digest %>
two:
email_address: two@example.com
password_digest: <%= password_digest %>
# Preview all emails at http://localhost:3000/rails/mailers/passwords_mailer
class PasswordsMailerPreview < ActionMailer::Preview
# Preview this email at http://localhost:3000/rails/mailers/passwords_mailer/reset
def reset
PasswordsMailer.reset(User.take)
end
end
require "test_helper"
class UserTest < ActiveSupport::TestCase
# test "the truth" do
# assert true
# end
end
はい。こんな感じでファイルが作成されました。
導入を見送った理由
しかし、私はこのAuthenticationを導入することを見送りました。
Rails公式が準備した機能としては結構クセのある実装であるところが気になりました。
特に、やっぱりapp/models/current.rb
ですね。
ActiveSupport::CurrentAttributes
というクラス自体、僕個人的には初めて見たレベルでした。
調べてみると理解はできたのですが、「わざわざこれするメリットってなんだ…?」というのが正直なところでした。
こちらの記事にもあるように、ちゃんとそのタイミングで設定がされているかどうかを判定するのが相当むずいのかな?と思っています。
基本的な?(というかDeviseのような、昔からよくある)current_user
を使う方法の方が、わかりやすいし、他の人が見たときにもわかりやすいかなと思いました。
まとめ
というわけで、今回はRails8で導入されたAuthenticationを試してみたが導入を見送った話をお伝えしました。
Rails8で導入された新機能は、とても便利なものが多いですが、その中でもクセのある実装があることもあるので、導入する際はよく検討してみることをおすすめします。
今回はこのあたりで。