sessions_controller.rb 3.4 KB
Newer Older
1
class SessionsController < Devise::SessionsController
2
  include AuthenticatesWithTwoFactor
3

4
  prepend_before_action :authenticate_with_two_factor, only: [:create]
5
  prepend_before_action :store_redirect_path, only: [:new]
6
  before_action :auto_sign_in_with_provider, only: [:new]
7

8
  def new
9
    if Gitlab.config.ldap.enabled
10
      @ldap_servers = Gitlab::LDAP::Config.servers
V
Valery Sizov 已提交
11 12
    else
      @ldap_servers = []
13 14
    end

15 16 17 18
    super
  end

  def create
19
    super do |resource|
R
Robert Speicher 已提交
20
      # User has successfully signed in, so clear any unused reset token
21 22 23 24
      if resource.reset_password_token.present?
        resource.update_attributes(reset_password_token: nil,
                                   reset_password_sent_at: nil)
      end
V
Valery Sizov 已提交
25 26
      authenticated_with = user_params[:otp_attempt] ? "two-factor" : "standard"
      log_audit_event(current_user, with: authenticated_with)
27
    end
28
  end
29 30 31

  private

32 33 34 35
  def user_params
    params.require(:user).permit(:login, :password, :remember_me, :otp_attempt)
  end

R
Robert Speicher 已提交
36 37 38 39 40 41 42
  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
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
  
  def store_redirect_path
    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

    # Prevent a 'you are already signed in' message directly after signing:
    # we should never redirect to '/users/sign_in' after signing in successfully.
    unless redirect_path == new_user_session_path
      store_location_for(:redirect, redirect_path)
    end
  end
R
Robert Speicher 已提交
63

64
  def authenticate_with_two_factor
R
Robert Speicher 已提交
65 66
    user = self.resource = find_user

67
    return unless user && user.two_factor_enabled?
68

R
Robert Speicher 已提交
69 70 71 72 73
    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)

74
        sign_in(user) and return
75
      else
76
        flash.now[:alert] = 'Invalid two-factor code.'
77 78 79
        render :two_factor and return
      end
    else
R
Robert Speicher 已提交
80
      if user && user.valid_password?(user_params[:password])
81
        prompt_for_two_factor(user)
82 83 84
      end
    end
  end
85

86 87 88 89 90 91 92 93 94 95 96 97
  def auto_sign_in_with_provider
    provider = Gitlab.config.omniauth.auto_sign_in_with_provider
    return unless provider.present?

    # Auto sign in with an Omniauth provider only if the standard "you need to sign-in" alert is 
    # registered or no alert at all. In case of another alert (such as a blocked user), it is safer  
    # to do nothing to prevent redirection loops with certain Omniauth providers.
    return unless flash[:alert].blank? || flash[:alert] == I18n.t('devise.failure.unauthenticated')
    
    # Prevent alert from popping up on the first page shown after authentication.
    flash[:alert] = nil 
    
98
    redirect_to user_omniauth_authorize_path(provider.to_sym)
99 100
  end

R
Robert Speicher 已提交
101
  def valid_otp_attempt?(user)
102
    user.validate_and_consume_otp!(user_params[:otp_attempt]) ||
R
Robert Speicher 已提交
103
    user.invalidate_otp_backup_code!(user_params[:otp_attempt])
104
  end
V
Valery Sizov 已提交
105 106 107 108 109

  def log_audit_event(user, options = {})
    AuditEventService.new(user, user, options).
      for_authentication.security_event
  end
110
end