sessions_controller.rb 3.7 KB
Newer Older
1
class SessionsController < Devise::SessionsController
2
  include AuthenticatesWithTwoFactor
3
  include Devise::Controllers::Rememberable
4
  include Recaptcha::ClientHelper
5

6 7
  skip_before_action :check_2fa_requirement, only: [:destroy]

8
  prepend_before_action :check_initial_setup, only: [:new]
9 10
  prepend_before_action :authenticate_with_two_factor,
    if: :two_factor_enabled?, only: [:create]
11
  prepend_before_action :store_redirect_path, only: [:new]
12

13
  before_action :auto_sign_in_with_provider, only: [:new]
14
  before_action :load_recaptcha
15

16
  def new
17
    if Gitlab.config.ldap.enabled
18
      @ldap_servers = Gitlab::LDAP::Config.servers
V
Valery Sizov 已提交
19 20
    else
      @ldap_servers = []
21 22
    end

23 24 25 26
    super
  end

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

  private

39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
  # Handle an "initial setup" state, where there's only one user, it's an admin,
  # and they require a password change.
  def check_initial_setup
    return unless User.count == 1

    user = User.admins.last

    return unless user && user.require_password?

    token = user.generate_reset_token
    user.save

    redirect_to edit_user_password_path(reset_password_token: token),
      notice: "Please create a password for your new account."
  end

55
  def user_params
56
    params.require(:user).permit(:login, :password, :remember_me, :otp_attempt, :device_response)
57 58
  end

R
Robert Speicher 已提交
59
  def find_user
60
    if session[:otp_user_id]
R
Robert Speicher 已提交
61
      User.find(session[:otp_user_id])
62 63
    elsif user_params[:login]
      User.by_login(user_params[:login])
R
Robert Speicher 已提交
64 65
    end
  end
66

67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
  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 已提交
86

87 88 89 90
  def two_factor_enabled?
    find_user.try(:two_factor_enabled?)
  end

91 92 93 94
  def auto_sign_in_with_provider
    provider = Gitlab.config.omniauth.auto_sign_in_with_provider
    return unless provider.present?

95 96
    # 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
97 98
    # to do nothing to prevent redirection loops with certain Omniauth providers.
    return unless flash[:alert].blank? || flash[:alert] == I18n.t('devise.failure.unauthenticated')
99

100
    # Prevent alert from popping up on the first page shown after authentication.
101 102
    flash[:alert] = nil

103
    redirect_to user_omniauth_authorize_path(provider.to_sym)
104 105
  end

R
Robert Speicher 已提交
106
  def valid_otp_attempt?(user)
107
    user.validate_and_consume_otp!(user_params[:otp_attempt]) ||
R
Robert Speicher 已提交
108
    user.invalidate_otp_backup_code!(user_params[:otp_attempt])
109
  end
V
Valery Sizov 已提交
110 111 112 113 114

  def log_audit_event(user, options = {})
    AuditEventService.new(user, user, options).
      for_authentication.security_event
  end
115 116 117 118

  def load_recaptcha
    Gitlab::Recaptcha.load_configurations!
  end
119 120 121 122 123 124 125 126 127 128

  def authentication_method
    if user_params[:otp_attempt]
      "two-factor"
    elsif user_params[:device_response]
      "two-factor-via-u2f-device"
    else
      "standard"
    end
  end
129
end