sessions_controller.rb 2.3 KB
Newer Older
1
class SessionsController < Devise::SessionsController
2
  prepend_before_action :authenticate_with_two_factor, only: :create
3

4
  def new
5 6 7 8 9 10 11 12 13 14 15
    redirect_path =
      if request.referer.present? && (params['redirect_to_referer'] == 'yes')
        referer_uri = URI(request.referer)
        if referer_uri.host == Gitlab.config.gitlab.host
          referer_uri.path
        else
          request.fullpath
        end
      else
        request.fullpath
      end
M
Marin Jankovski 已提交
16

17 18
    # Prevent a 'you are already signed in' message directly after signing:
    # we should never redirect to '/users/sign_in' after signing in successfully.
19
    unless redirect_path == new_user_session_path
20 21
      store_location_for(:redirect, redirect_path)
    end
22

23
    if Gitlab.config.ldap.enabled
24
      @ldap_servers = Gitlab::LDAP::Config.servers
25 26
    end

27 28 29 30
    super
  end

  def create
31
    super do |resource|
R
Robert Speicher 已提交
32
      # User has successfully signed in, so clear any unused reset token
33 34 35 36 37
      if resource.reset_password_token.present?
        resource.update_attributes(reset_password_token: nil,
                                   reset_password_sent_at: nil)
      end
    end
38
  end
39 40 41

  private

42 43 44 45
  def user_params
    params.require(:user).permit(:login, :password, :remember_me, :otp_attempt)
  end

R
Robert Speicher 已提交
46 47 48 49 50 51 52 53
  def find_user
    if user_params[:login]
      User.by_login(user_params[:login])
    elsif user_params[:otp_attempt] && session[:otp_user_id]
      User.find(session[:otp_user_id])
    end
  end

54
  def authenticate_with_two_factor
R
Robert Speicher 已提交
55 56 57
    user = self.resource = find_user

    return unless user && user.otp_required_for_login
58

R
Robert Speicher 已提交
59 60 61 62 63 64
    if user_params[:otp_attempt].present? && session[:otp_user_id]
      if valid_otp_attempt?(user)
        # Remove any lingering user data from login
        session.delete(:otp_user_id)

        sign_in(user)
65
      else
66
        flash.now[:alert] = 'Invalid two-factor code.'
67 68 69
        render :two_factor and return
      end
    else
R
Robert Speicher 已提交
70 71 72 73
      if user && user.valid_password?(user_params[:password])
        # Save the user's ID to session so we can ask for a one-time password
        session[:otp_user_id] = user.id
        render :two_factor and return
74 75 76
      end
    end
  end
77

R
Robert Speicher 已提交
78 79 80
  def valid_otp_attempt?(user)
    user.valid_otp?(user_params[:otp_attempt]) ||
    user.invalidate_otp_backup_code!(user_params[:otp_attempt])
81
  end
82
end