Railscasts:从头开始的身份验证 - 缺少:Rails 4中的会话参数问题

时间:2016-04-04 04:46:19

标签: ruby-on-rails authentication railscasts

我在Rails 4中完成 Railscast:从头开始验证,尽管该教程是为Rails 3构建的。 因此,必须从原始教程说明中进行一些更改。

教程:http://railscasts.com/episodes/250-authentication-from-scratch?autoplay=true

当前遇到错误

  

param丢失或值为空:session

思想:

  • 我比较了下面的2个控制器。我从教程中的变化是将params转换为session_params方法。
  • 我理解错误是因为发布的哈希不包含像"session" => { }这样的哈希。但是我对Rails没有足够深入的了解才能解决。

推荐的控制器(导轨3)

def create
  user = User.authenticate(params[:email], params[:password])
  if user
    session[:user_id] = user.id
    redirect_to root_url, :notice => "Logged in!"
  else
    flash.now.alert = "Invalid email or password"
    render "new"
  end
end

我的控制器(转换为Rails 4)

  def create
    user = User.authenticate(session_params)
    if user
      session[:user_id] = user.id
      redirect_to root_url, :notice => "Logged in!"
    else
      flash.now.alert = "Invalid email or password"
      render "new"
    end
  end

  private
  def session_params
    params.require(:session).permit(:email, :password)
  end

到目前为止,我只使用Devise进行身份验证,所以如果您可以分享任何建议,请提前表示非常感谢。

更新

根据要求,表格发布会话

<h1>Log in</h1>
<%= form_tag sessions_path do %>
    <p>
        <%= label_tag :email %> <br/>
        <%= text_field_tag :email, params[:email] %>
    </p>
    <p>
        <%= label_tag :password %> <br/>
        <%= password_field_tag :password %>
    </p>
    <p class="button"> <%= submit_tag %> </p>
<% end %>

更新2

同时添加user.rb模型

class User < ActiveRecord::Base
    attr_accessor :password
    before_save :encrypt_password

    validates_confirmation_of :password
    validates_presence_of :password, :on => :create
    validates_presence_of :email
    validates_uniqueness_of :email

    def self.authenticate(email, password)
        user = find_by_email(email)
        if user && user.password_hash == BCrypt::Engine.hash_secret(password, password_salt)
            user
        end
    end

    def encrypt_password
        if password.present?
            self.password_salt = BCrypt::Engine.generate_salt
            self.password_hash = BCrypt::Engine.hash_secret(password, password_salt)
        end
    end
end

1 个答案:

答案 0 :(得分:1)

试试这个:

sessions_controller.rb

中的

def create
    @user = User.find_by(email: params[:session][:email].downcase)
    if @user && @user.authenticate(params[:session][:password])
        session[:user_id] = @user.id
        flash[:success] = "#{@user.email}, Successfully Logged In"
        redirect_to root_url, :notice => "Logged in!"
    else
        flash.now[:danger] = "Incorrect User/Password"
        render 'new'
    end
end

html表单

<%= form_for :session, url: :sessions do |f| %>

    <div class="field">
        <%= f.label :email %>
        <%= f.email_field :email %>
    </div>

    <div class="field">
        <%= f.label :password %>
        <%= f.password_field :password %>
    </div>

    <%= f.submit "Sign In", class: "button" %>
<% end %>

<强>的routes.rb

post 'sessions'  => 'sessions#create'

<强> User.rb

class User < ActiveRecord::Base
    has_secure_password

    validates_confirmation_of :password
    validates_presence_of :password, :on => :create
    validates_presence_of :email
    validates_uniqueness_of :email

end