質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.34%
Ruby on Rails 7

Ruby on Rails 7は、2021年12月に正式リリースされました。Ruby on Railsのバージョン7であり、フロントエンド開発環境を大幅に刷新。Node.jsを用いない構成がデフォルトになっています。

Q&A

解決済

2回答

286閲覧

Railsチュートリアル第7版の11章 11.3.1authenticated?メソッドの抽象化の突破。 エラーの原因が不明です

12a1070

総合スコア1

Ruby on Rails 7

Ruby on Rails 7は、2021年12月に正式リリースされました。Ruby on Railsのバージョン7であり、フロントエンド開発環境を大幅に刷新。Node.jsを用いない構成がデフォルトになっています。

0グッド

0クリップ

投稿2025/01/04 05:40

編集2025/01/04 10:40

実現したいこと

  • エラー原因の特定。
  • テストのパス

発生している問題・分からないこと

11.3.1authenticated?メソッドの抽象化
https://railstutorial.jp/chapters/account_activation?version=7.0#sec-generalizing_the_authenticated_method
の進行中でテストをしたところエラーが2件発生しました。

エラーメッセージ

error

1ERROR SessionsHelperTest#test_current_user_returns_nil_when_remember_digest_is_wrong (0.13s) 2Minitest::UnexpectedError: ArgumentError: wrong number of arguments (given 1, expected 2) 3 app/models/user.rb:39:in `authenticated?' 4 test/helpers/sessions_helper_test.rb:26:in `current_user' 5 test/helpers/sessions_helper_test.rb:17:in `block in <class:SessionsHelperTest>' 6 7ERROR SessionsHelperTest#test_current_user_returns_right_user_when_session_is_nil (0.14s) 8Minitest::UnexpectedError: ArgumentError: wrong number of arguments (given 1, expected 2) 9 app/models/user.rb:39:in `authenticated?' 10 test/helpers/sessions_helper_test.rb:26:in `current_user' 11 test/helpers/sessions_helper_test.rb:11:in `block in <class:SessionsHelperTest>'

該当のソースコード

Ruby

1authenticated?メソッドを含めたapp/models/user.rb 2 3class User < ApplicationRecord 4 attr_accessor :remember_token, :activation_token 5 before_save :downcase_email 6 before_create :create_activation_digest 7 validates :name, presence: true, length: { maximum: 50 } 8 VALID_EMAIL_REGEX = /\A[\w+\-.]+@[a-z\d\-.]+\.[a-z]+\z/i 9 validates :email, presence: true, length: { maximum: 255 }, 10 format: { with: VALID_EMAIL_REGEX }, 11 uniqueness: true 12 has_secure_password 13 validates :password, presence: true, length: { minimum: 6 }, allow_nil: true 14 15 # 渡された文字列のハッシュ値を返す 16 def User.digest(string) 17 cost = ActiveModel::SecurePassword.min_cost ? BCrypt::Engine::MIN_COST : 18 BCrypt::Engine.cost 19 BCrypt::Password.create(string, cost: cost) 20 end 21 22 # ランダムなトークンを返す 23 def User.new_token 24 SecureRandom.urlsafe_base64 25 end 26 27 # 永続的セッションのためにユーザーをデータベースに記憶する 28 def remember 29 self.remember_token = User.new_token 30 update_attribute(:remember_digest, User.digest(remember_token)) 31 remember_digest 32 end 33 34 # セッションハイジャック防止のためにセッショントークンを返す 35 # この記憶ダイジェストを再利用しているのは単に利便性のため 36 def session_token 37 remember_digest || remember 38 end 39 40 # 渡されたトークンがダイジェストと一致したらtrueを返す(追加した部分です) 41 def authenticated?(attribute, token) 42 digest = send("#{attribute}_digest") 43 return false if digest.nil? 44 BCrypt::Password.new(digest).is_password?(token) 45 end 46 47 # ユーザーのログイン情報を破棄する 48 def forget 49 update_attribute(:remember_digest, nil) 50 end 51 52 private 53 54 # メールアドレスをすべて小文字にする 55 def downcase_email 56 self.email = email.downcase 57 end 58 59 # 有効化トークンとダイジェストを作成および代入する 60 def create_activation_digest 61 self.activation_token = User.new_token 62 self.activation_digest = User.digest(activation_token) 63 end 64 65end 66 67

Ruby

1current_user内の抽象化したauthenticated?メソッド 2 3app/helpers/sessions_helper.rb 4 5module SessionsHelper 6 7 # 渡されたユーザーでログインする 8 def log_in(user) 9 session[:user_id] = user.id 10 # セッションリプレイ攻撃から保護する 11 # 詳しくは https://techracho.bpsinc.jp/hachi8833/2023_06_02/130443 を参照 12 session[:session_token] = user.session_token 13 end 14 15 # 永続的セッションのためにユーザーをデータベースに記憶する 16 def remember(user) 17 user.remember 18 cookies.permanent.encrypted[:user_id] = user.id 19 cookies.permanent[:remember_token] = user.remember_token 20 end 21 22 # 記憶トークンcookieに対応するユーザーを返す 23 def current_user 24 if (user_id = session[:user_id]) 25 user = User.find_by(id: user_id) 26 @current_user ||= user if session[:session_token] == user.session_token 27 elsif (user_id = cookies.encrypted[:user_id]) 28 user = User.find_by(id: user_id) 29 if user && user.authenticated?(:remember, cookies[:remember_token]) ⇦追加したコードです 30 log_in user 31 @current_user = user 32 end 33 end 34 end 35 36 37 # 渡されたユーザーがカレントユーザーであればtrueを返す 38 def current_user?(user) 39 user && user == current_user 40 end 41 42 # ユーザーがログインしていればtrue、その他ならfalseを返す 43 def logged_in? 44 !current_user.nil? 45 end 46 47 # 永続的セッションを破棄する 48 def forget(user) 49 user.forget 50 cookies.delete(:user_id) 51 cookies.delete(:remember_token) 52 end 53 54 # 現在のユーザーをログアウトする 55 def log_out 56 forget(current_user) 57 reset_session 58 @current_user = nil 59 end 60 61 # アクセスしようとしたURLを保存する 62 def store_location 63 session[:forwarding_url] = request.original_url if request.get? 64 end 65 66end 67

Ruby

1Userテスト内の抽象化したauthenticated?メソッド 2 3test/models/user_test.rb 4 5require "test_helper" 6 7class UserTest < ActiveSupport::TestCase 8 9 def setup 10 @user = User.new(name: "Example User", email: "user@example.com", 11 password: "foobar", password_confirmation: "foobar") 12 end 13 14 test "should be valid" do 15 assert @user.valid? 16 end 17 18 test "name should be present" do 19 @user.name = "" 20 assert_not @user.valid? 21 end 22 23 test "email should be present" do 24 @user.email = " " 25 assert_not @user.valid? 26 end 27 28 test "name should not be too long" do 29 @user.name = "a" * 51 30 assert_not @user.valid? 31 end 32 33 test "email should not be too long" do 34 @user.email = "a" * 244 + "@example.com" 35 assert_not @user.valid? 36 end 37 38 test "email validation should accept valid addresses" do 39 valid_addresses = %w[user@example.com USER@foo.COM A_US-ER@foo.bar.org 40 first.last@foo.jp alice+bob@baz.cn] 41 valid_addresses.each do |valid_address| 42 @user.email = valid_address 43 assert @user.valid?, "#{valid_address.inspect} should be valid" 44 end 45 end 46 47 test "email validation should reject invalid addresses" do 48 invalid_addresses = %w[user@example,com user_at_foo.org user.name@example. 49 foo@bar_baz.com foo@bar+baz.com] 50 invalid_addresses.each do |invalid_address| 51 @user.email = invalid_address 52 assert_not @user.valid?, "#{invalid_address.inspect} should be invalid" 53 end 54 end 55 56 57 test "email addresses should be unique" do 58 duplicate_user = @user.dup 59 @user.save 60 assert_not duplicate_user.valid? 61 end 62 63 test "password should be present (nonblank)" do 64 @user.password = @user.password_confirmation = " " * 6 65 assert_not @user.valid? 66 end 67 68 test "password should have a minimum length" do 69 @user.password = @user.password_confirmation = "a" * 5 70 assert_not @user.valid? 71 end 72 73 test "authenticated? should return false for a user with nil digest" do ⇦追加した部分です。 74 assert_not @user.authenticated?(:remember, '') 75 end 76end 77

※追加したsessions_helper_test.rb

Ruby

1 2require "test_helper" 3 4class SessionsHelperTest < ActionView::TestCase 5 6 def setup 7 @user = users(:michael) 8 remember(@user) 9 end 10 11 test "current_user returns right user when session is nil" do 12 assert_equal @user, current_user 13 assert is_logged_in? 14 end 15 16 test "current_user returns nil when remember digest is wrong" do 17 @user.update_attribute(:remember_digest, User.digest(User.new_token)) 18 assert_nil current_user 19 end 20 21# 記憶トークンcookieに対応するユーザーを返す 22 def current_user 23 if (user_id = session[:user_id]) 24 @current_user ||= User.find_by(id: user_id) 25 elsif (user_id = cookies.encrypted[:user_id]) 26 user = User.find_by(id: user_id) 27 if user && user.authenticated?(cookies[:remember_token]) 28 log_in user 29 @current_user = user 30 end 31 end 32 end 33 34end 35

試したこと・調べたこと

  • teratailやGoogle等で検索した
  • ソースコードを自分なりに変更した
  • 知人に聞いた
  • その他
上記の詳細・結果
  • チャットgptにて質問した結果を引用

エラーメッセージから判断すると、authenticated? メソッドに渡す引数が間違っている可能性があります。authenticated? メソッドは2つの引数を期待していますが、渡された引数が1つだけであることが原因でエラーが発生しているようです。

authenticated? メソッドに渡す引数を確認し、適切なトークン(cookies[:remember_token] など)を渡してみてください。

修正方法
authenticated? メソッドに渡す引数を確認し、適切なトークン(cookies[:remember_token] など)を渡してみてください。

以下のように修正してみてください:

修正例:
ruby
コードをコピーする

user.rb の authenticated? メソッド

def authenticated?(attribute, token)
digest = send("#{attribute}_digest")
return false if digest.nil?
BCrypt::Password.new(digest).is_password?(token)
end

sessions_helper.rb 内での呼び出し

if user && user.authenticated?(:remember, cookies[:remember_token]) # ここでトークンを渡す
log_in user
@current_user = user
end

修正箇所がないように見えます。
sessions_helper_test.rbを追記しました。リスト 11.30のrails testでgreenになるはずがエラーによってRedの状態です。

補足

Rails 7.0.8.4

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

maisumakun

2025/01/04 06:48

「環境を再起動して、それでもエラーになるかチェックしてみる」「あえてエラーの出るようなコードを仕込むことで、本当に書き換えたコードを実行していると確認する」ことなどをまず行ってみましょう。
winterboum

2025/01/04 08:13

test_current_user_returns_nil_when_remember_digest_is_wrong のcodeが載ってないですね
12a1070

2025/01/04 10:35

>> maisumakun 様 Rails sをしても同じでした。 >> winterboum 様 test_current_user_returns_nil_when_remember_digest_is_wrongのコード。sessions_helper_test.rbを追記しました。テストのエラー内容は同じです。
winterboum

2025/01/04 12:00

「テストのエラー内容は同じです。」<= これ本当? エラーが指摘してる行とcodeとずれてる気がする。
guest

回答2

0

自己解決

app/helpers/sessions_helper.rbの

ruby

1# 現在ログイン中のユーザーを返す(いる場合) 2 def current_user 3 if (user_id = session[:user_id]) 4 user = User.find_by(id: user_id) 5 @current_user ||= user if session[:session_token] == user.session_token 6 elsif (user_id = cookies.encrypted[:user_id]) 7 user = User.find_by(id: user_id) 8 if user && user.authenticated?(cookies[:remember_token]) 9 log_in user 10 @current_user = user 11 end 12 end 13 end

ruby

1 if user && user.authenticated?(cookies[:remember_token])

ruby

1if user && user.authenticated?(:remember, cookies[:remember_token])

に変えればパスできました。

投稿2025/01/05 12:27

12a1070

総合スコア1

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

エラーメッセージは 17行 → 26行 → user.rb:39行 ってなってるけど、
「追加したsessions_helper_test.rb」の 17行から26行には行かないと思われる。また user.rb39行は何もない行だからここでエラーが起きるはずばない。
エラーメッセージを採ったときと1行ずれてて今載ってる sessions_helper_test.rb の 18行→27行 → user.rb:41行 と思われるのでその線で回答すると
if user && user.authenticated?(cookies[:remember_token]) の user.authenticated? は引数が一つ、 user.rb:41行 の def authenticated?(attribute, token) は引数2つを要求してる。

ということで
ArgumentError: wrong number of arguments (given 1, expected 2)

投稿2025/01/04 12:09

winterboum

総合スコア23589

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

12a1070

2025/01/04 14:41 編集

>> winterboum 様 改めてテストを行なったところ ```Ruby ERROR SessionsHelperTest#test_current_user_returns_right_user_when_session_is_nil (1.59s) Minitest::UnexpectedError: ArgumentError: wrong number of arguments (given 1, expected 2) app/models/user.rb:39:in `authenticated?' test/helpers/sessions_helper_test.rb:26:in `current_user' test/helpers/sessions_helper_test.rb:11:in `block in <class:SessionsHelperTest>' ERROR SessionsHelperTest#test_current_user_returns_nil_when_remember_digest_is_wrong (1.60s) Minitest::UnexpectedError: ArgumentError: wrong number of arguments (given 1, expected 2) app/models/user.rb:39:in `authenticated?' test/helpers/sessions_helper_test.rb:26:in `current_user' test/helpers/sessions_helper_test.rb:17:in `block in <class:SessionsHelperTest>' 39/39: [========] 100% Time: 00:00:01, Time: 00:00:01 Finished in 1.72135s 39 tests, 162 assertions, 0 failures, 2 errors, 0 skips ``` 関連するコードは ```Ruby app/models/user.rb:39:in `authenticated?' # 渡されたトークンがダイジェストと一致したらtrueを返す def authenticated?(attribute, token) ⇦ app/models/user.rb:39:in `authenticated?' digest = send("#{attribute}_digest") return false if digest.nil? BCrypt::Password.new(digest).is_password?(token) end ``` ```Ruby test/helpers/sessions_helper_test.rb:26:in `current_user' # 記憶トークンcookieに対応するユーザーを返す def current_user if (user_id = session[:user_id]) @current_user ||= User.find_by(id: user_id) elsif (user_id = cookies.encrypted[:user_id]) user = User.find_by(id: user_id) if user && user.authenticated?(cookies[:remember_token]) ⇦26行目 log_in user @current_user = user end end end ``` ```Ruby test/helpers/sessions_helper_test.rb:11:in `block in <class:SessionsHelperTest>' test "current_user returns right user when session is nil" do assert_equal @user, current_user   ⇦11行目 assert is_logged_in? end ``` エラーが起きたのは分かったのですがテキストを噛み砕いてやっているのですが変数やインスタンスはどう変えれば良いのか分からず。
winterboum

2025/01/04 23:03

エラーが発覚したのは app/models/user.rb:39:in `authenticated?' ですが、その原因となったのがどのfileのどの行か、というのは理解できました?
12a1070

2025/01/05 08:26

エラー文が ```Ruby ERROR SessionsHelperTest#test_current_user_returns_nil_when_remember_digest_is_wrong (1.03s) Minitest::UnexpectedError: ArgumentError: wrong number of arguments (given 1, expected 2) app/models/user.rb:39:in `authenticated?' test/helpers/sessions_helper_test.rb:26:in `current_user' test/helpers/sessions_helper_test.rb:17:in `block in <class:SessionsHelperTest>' ERROR SessionsHelperTest#test_current_user_returns_right_user_when_session_is_nil (1.04s) Minitest::UnexpectedError: ArgumentError: wrong number of arguments (given 1, expected 2) app/models/user.rb:39:in `authenticated?' test/helpers/sessions_helper_test.rb:26:in `current_user' test/helpers/sessions_helper_test.rb:11:in `block in <class:SessionsHelperTest>' ``` 39行目に当たる ```Ruby def authenticated?(attribute, token) ⇦39行目でエラー発覚 digest = send("#{attribute}_digest") return false if digest.nil? BCrypt::Password.new(digest).is_password?(token) end ``` session_helper_test.rbの ```Ruby # 現在ログイン中のユーザーを返す(いる場合) def current_user if (user_id = session[:user_id]) @current_user ||= User.find_by(id: user_id) elsif (user_id = cookies.encrypted[:user_id]) user = User.find_by(id: user_id) if user && user.authenticated?(cookies[:remember_token]) ⇦ 26:in `current_user' log_in user @current_user = user end end end ``` おそらく26行目にcurrent_user'がないことが原因だと思います。その後のコードをどう書けば良いのか分からないです。 if user && user.authenticated?(cookies[:remember_token])の上にbinding.pryして [2] pry(#<SessionsHelperTest>)> @current_user したのですがnillでした。
winterboum

2025/01/05 10:05

エラーメッセージきちんと読んで。 どのfileの何行目でどんなエラーが起きているか そこへは どのfileの何行目から行ってるのか が書かれているから、そこに注目。 「26行目に」は正しいけど、「current_user'がないことが原因」ってなるのはエラーメッセージきちんと読んでいない。
12a1070

2025/01/05 11:47

回答ありがとうございます。 ベストアンサーは別の方を選ばせていただきましたが、こちらの回答も非常に参考になりました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.34%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問