2018-09-14 01:42:05 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2014-07-11 06:24:11 -04:00
|
|
|
class SessionsController < Devise::SessionsController
|
2018-05-02 14:25:21 -04:00
|
|
|
include InternalRedirect
|
2015-05-14 14:24:05 -04:00
|
|
|
include AuthenticatesWithTwoFactor
|
2016-05-30 22:17:26 -04:00
|
|
|
include Devise::Controllers::Rememberable
|
2015-12-27 12:03:06 -05:00
|
|
|
include Recaptcha::ClientHelper
|
2018-06-21 14:13:08 -04:00
|
|
|
include Recaptcha::Verify
|
2020-04-24 11:09:37 -04:00
|
|
|
include RendersLdapServers
|
2020-05-12 23:08:26 -04:00
|
|
|
include KnownSignIn
|
2020-09-02 11:10:54 -04:00
|
|
|
include Gitlab::Utils::StrongMemoize
|
2015-05-09 17:04:02 -04:00
|
|
|
|
2017-03-07 13:48:57 -05:00
|
|
|
skip_before_action :check_two_factor_requirement, only: [:destroy]
|
2020-09-01 05:10:28 -04:00
|
|
|
skip_before_action :check_password_expiration, only: [:destroy]
|
|
|
|
|
2019-04-24 09:23:07 -04:00
|
|
|
# replaced with :require_no_authentication_without_flash
|
|
|
|
skip_before_action :require_no_authentication, only: [:new, :create]
|
2016-01-23 19:44:46 -05:00
|
|
|
|
2016-03-02 17:48:49 -05:00
|
|
|
prepend_before_action :check_initial_setup, only: [:new]
|
2016-04-07 05:19:29 -04:00
|
|
|
prepend_before_action :authenticate_with_two_factor,
|
2019-02-27 02:41:14 -05:00
|
|
|
if: -> { action_name == 'create' && two_factor_enabled? }
|
2018-06-21 14:13:08 -04:00
|
|
|
prepend_before_action :check_captcha, only: [:create]
|
2017-09-19 03:44:58 -04:00
|
|
|
prepend_before_action :store_redirect_uri, only: [:new]
|
2019-04-24 09:23:07 -04:00
|
|
|
prepend_before_action :require_no_authentication_without_flash, only: [:new, :create]
|
2019-02-27 02:41:14 -05:00
|
|
|
prepend_before_action :ensure_password_authentication_enabled!, if: -> { action_name == 'create' && password_based_login? }
|
2019-04-24 09:23:07 -04:00
|
|
|
|
2015-05-27 11:40:21 -04:00
|
|
|
before_action :auto_sign_in_with_provider, only: [:new]
|
2019-07-18 04:27:02 -04:00
|
|
|
before_action :store_unauthenticated_sessions, only: [:new]
|
|
|
|
before_action :save_failed_login, if: :action_new_and_failed_login?
|
2015-12-28 15:21:34 -05:00
|
|
|
before_action :load_recaptcha
|
2020-08-10 14:09:54 -04:00
|
|
|
before_action :set_invite_params, only: [:new]
|
2020-09-07 08:08:27 -04:00
|
|
|
before_action do
|
|
|
|
push_frontend_feature_flag(:webauthn)
|
|
|
|
end
|
2015-03-30 21:19:01 -04:00
|
|
|
|
2019-07-18 04:27:02 -04:00
|
|
|
after_action :log_failed_login, if: :action_new_and_failed_login?
|
2020-05-12 23:08:26 -04:00
|
|
|
after_action :verify_known_sign_in, only: [:create]
|
2019-07-18 04:27:02 -04:00
|
|
|
|
|
|
|
helper_method :captcha_enabled?, :captcha_on_login_required?
|
2018-06-21 14:13:08 -04:00
|
|
|
|
2019-07-26 03:05:50 -04:00
|
|
|
# protect_from_forgery is already prepended in ApplicationController but
|
|
|
|
# authenticate_with_two_factor which signs in the user is prepended before
|
|
|
|
# that here.
|
|
|
|
# We need to make sure CSRF token is verified before authenticating the user
|
|
|
|
# because Devise.clean_up_csrf_token_on_authentication is set to true by
|
|
|
|
# default to avoid CSRF token fixation attacks. Authenticating the user first
|
|
|
|
# would cause the CSRF token to be cleared and then
|
|
|
|
# RequestForgeryProtection#verify_authenticity_token would fail because of
|
|
|
|
# token mismatch.
|
2020-02-21 10:09:05 -05:00
|
|
|
protect_from_forgery with: :exception, prepend: true, except: :destroy
|
2019-07-26 03:05:50 -04:00
|
|
|
|
2019-08-31 15:21:10 -04:00
|
|
|
CAPTCHA_HEADER = 'X-GitLab-Show-Login-Captcha'
|
2019-07-18 04:27:02 -04:00
|
|
|
MAX_FAILED_LOGIN_ATTEMPTS = 5
|
2018-06-21 14:13:08 -04:00
|
|
|
|
2014-07-11 06:24:11 -04:00
|
|
|
def new
|
2016-06-03 06:48:11 -04:00
|
|
|
set_minimum_password_length
|
2014-10-13 07:39:54 -04:00
|
|
|
|
2014-07-11 06:24:11 -04:00
|
|
|
super
|
|
|
|
end
|
|
|
|
|
|
|
|
def create
|
2015-04-08 14:26:04 -04:00
|
|
|
super do |resource|
|
2015-05-08 20:41:53 -04:00
|
|
|
# User has successfully signed in, so clear any unused reset token
|
2015-04-08 14:26:04 -04:00
|
|
|
if resource.reset_password_token.present?
|
2018-07-02 06:43:06 -04:00
|
|
|
resource.update(reset_password_token: nil,
|
|
|
|
reset_password_sent_at: nil)
|
2015-04-08 14:26:04 -04:00
|
|
|
end
|
2018-01-11 11:34:01 -05:00
|
|
|
|
2019-10-09 20:06:44 -04:00
|
|
|
if resource.deactivated?
|
|
|
|
resource.activate
|
|
|
|
flash[:notice] = _('Welcome back! Your account had been deactivated due to inactivity but is now reactivated.')
|
|
|
|
else
|
|
|
|
# hide the default signed-in notification
|
|
|
|
flash[:notice] = nil
|
|
|
|
end
|
|
|
|
|
2017-08-23 00:40:16 -04:00
|
|
|
log_audit_event(current_user, resource, with: authentication_method)
|
2016-10-05 10:41:32 -04:00
|
|
|
log_user_activity(current_user)
|
2015-04-08 14:26:04 -04:00
|
|
|
end
|
2014-07-11 06:24:11 -04:00
|
|
|
end
|
2015-03-30 21:19:01 -04:00
|
|
|
|
2016-12-06 12:02:30 -05:00
|
|
|
def destroy
|
2017-08-23 00:40:16 -04:00
|
|
|
Gitlab::AppLogger.info("User Logout: username=#{current_user.username} ip=#{request.remote_ip}")
|
2016-12-06 12:02:30 -05:00
|
|
|
super
|
|
|
|
# hide the signed_out notice
|
|
|
|
flash[:notice] = nil
|
|
|
|
end
|
|
|
|
|
2015-03-30 21:19:01 -04:00
|
|
|
private
|
|
|
|
|
2019-04-24 09:23:07 -04:00
|
|
|
def require_no_authentication_without_flash
|
|
|
|
require_no_authentication
|
|
|
|
|
|
|
|
if flash[:alert] == I18n.t('devise.failure.already_authenticated')
|
|
|
|
flash[:alert] = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-06-21 14:13:08 -04:00
|
|
|
def captcha_enabled?
|
|
|
|
request.headers[CAPTCHA_HEADER] && Gitlab::Recaptcha.enabled?
|
|
|
|
end
|
|
|
|
|
2019-07-18 04:27:02 -04:00
|
|
|
def captcha_on_login_required?
|
|
|
|
Gitlab::Recaptcha.enabled_on_login? && unverified_anonymous_user?
|
|
|
|
end
|
|
|
|
|
2018-06-21 14:13:08 -04:00
|
|
|
# From https://github.com/plataformatec/devise/wiki/How-To:-Use-Recaptcha-with-Devise#devisepasswordscontroller
|
|
|
|
def check_captcha
|
|
|
|
return unless user_params[:password].present?
|
2019-07-18 04:27:02 -04:00
|
|
|
return unless captcha_enabled? || captcha_on_login_required?
|
2018-06-21 14:13:08 -04:00
|
|
|
return unless Gitlab::Recaptcha.load_configurations!
|
|
|
|
|
2018-06-22 02:25:00 -04:00
|
|
|
if verify_recaptcha
|
|
|
|
increment_successful_login_captcha_counter
|
|
|
|
else
|
|
|
|
increment_failed_login_captcha_counter
|
|
|
|
|
2018-06-21 14:13:08 -04:00
|
|
|
self.resource = resource_class.new
|
2019-04-08 10:17:45 -04:00
|
|
|
flash[:alert] = _('There was an error with the reCAPTCHA. Please solve the reCAPTCHA again.')
|
2018-06-21 14:13:08 -04:00
|
|
|
flash.delete :recaptcha_error
|
|
|
|
|
|
|
|
respond_with_navigational(resource) { render :new }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-06-22 02:25:00 -04:00
|
|
|
def increment_failed_login_captcha_counter
|
|
|
|
Gitlab::Metrics.counter(
|
|
|
|
:failed_login_captcha_total,
|
2019-08-31 15:21:10 -04:00
|
|
|
'Number of failed CAPTCHA attempts for logins'
|
2018-06-22 02:25:00 -04:00
|
|
|
).increment
|
|
|
|
end
|
|
|
|
|
|
|
|
def increment_successful_login_captcha_counter
|
|
|
|
Gitlab::Metrics.counter(
|
|
|
|
:successful_login_captcha_total,
|
2019-08-31 15:21:10 -04:00
|
|
|
'Number of successful CAPTCHA attempts for logins'
|
2018-06-22 02:25:00 -04:00
|
|
|
).increment
|
|
|
|
end
|
|
|
|
|
2018-07-23 09:13:11 -04:00
|
|
|
##
|
|
|
|
# We do have some duplication between lib/gitlab/auth/activity.rb here, but
|
|
|
|
# leaving this method here because of backwards compatibility.
|
|
|
|
#
|
|
|
|
def login_counter
|
|
|
|
@login_counter ||= Gitlab::Metrics.counter(:user_session_logins_total, 'User sign in count')
|
|
|
|
end
|
|
|
|
|
2017-08-23 00:40:16 -04:00
|
|
|
def log_failed_login
|
2017-09-29 17:34:47 -04:00
|
|
|
Gitlab::AppLogger.info("Failed Login: username=#{user_params[:login]} ip=#{request.remote_ip}")
|
2017-08-23 00:40:16 -04:00
|
|
|
end
|
|
|
|
|
2019-07-18 04:27:02 -04:00
|
|
|
def action_new_and_failed_login?
|
|
|
|
action_name == 'new' && failed_login?
|
|
|
|
end
|
|
|
|
|
|
|
|
def save_failed_login
|
|
|
|
session[:failed_login_attempts] ||= 0
|
|
|
|
session[:failed_login_attempts] += 1
|
|
|
|
end
|
|
|
|
|
2017-08-23 00:40:16 -04:00
|
|
|
def failed_login?
|
2018-10-26 12:51:22 -04:00
|
|
|
(options = request.env["warden.options"]) && options[:action] == "unauthenticated"
|
2017-08-23 00:40:16 -04:00
|
|
|
end
|
|
|
|
|
2020-08-28 05:10:32 -04:00
|
|
|
# counting sessions per IP lets us check if there are associated multiple
|
2019-07-18 04:27:02 -04:00
|
|
|
# anonymous sessions with one IP and prevent situations when there are
|
|
|
|
# multiple attempts of logging in
|
|
|
|
def store_unauthenticated_sessions
|
|
|
|
return if current_user
|
|
|
|
|
2020-08-28 05:10:32 -04:00
|
|
|
Gitlab::AnonymousSession.new(request.remote_ip).count_session_ip
|
2019-07-18 04:27:02 -04:00
|
|
|
end
|
|
|
|
|
2016-03-02 17:48:49 -05:00
|
|
|
# Handle an "initial setup" state, where there's only one user, it's an admin,
|
|
|
|
# and they require a password change.
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2016-03-02 17:48:49 -05:00
|
|
|
def check_initial_setup
|
2016-06-06 09:50:58 -04:00
|
|
|
return unless User.limit(2).count == 1 # Count as much 2 to know if we have exactly one
|
2016-03-02 17:48:49 -05:00
|
|
|
|
|
|
|
user = User.admins.last
|
|
|
|
|
2017-11-23 08:16:14 -05:00
|
|
|
return unless user && user.require_password_creation_for_web?
|
2016-03-02 17:48:49 -05:00
|
|
|
|
2017-09-27 05:48:33 -04:00
|
|
|
Users::UpdateService.new(current_user, user: user).execute do |user|
|
2017-06-22 05:27:37 -04:00
|
|
|
@token = user.generate_reset_token
|
|
|
|
end
|
2016-03-02 17:48:49 -05:00
|
|
|
|
2017-06-22 05:27:37 -04:00
|
|
|
redirect_to edit_user_password_path(reset_password_token: @token),
|
2019-04-08 10:17:45 -04:00
|
|
|
notice: _("Please create a password for your new account.")
|
2016-03-02 17:48:49 -05:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2016-03-02 17:48:49 -05:00
|
|
|
|
2018-12-12 09:45:55 -05:00
|
|
|
def ensure_password_authentication_enabled!
|
|
|
|
render_403 unless Gitlab::CurrentSettings.password_authentication_enabled_for_web?
|
|
|
|
end
|
|
|
|
|
|
|
|
def password_based_login?
|
|
|
|
user_params[:login].present? || user_params[:password].present?
|
|
|
|
end
|
|
|
|
|
2015-05-05 22:16:45 -04:00
|
|
|
def user_params
|
2016-06-06 00:50:39 -04:00
|
|
|
params.require(:user).permit(:login, :password, :remember_me, :otp_attempt, :device_response)
|
2015-05-05 22:16:45 -04:00
|
|
|
end
|
|
|
|
|
2015-05-08 20:41:53 -04:00
|
|
|
def find_user
|
2020-09-02 11:10:54 -04:00
|
|
|
strong_memoize(:find_user) do
|
|
|
|
if session[:otp_user_id] && user_params[:login]
|
|
|
|
User.by_id_and_login(session[:otp_user_id], user_params[:login]).first
|
|
|
|
elsif session[:otp_user_id]
|
|
|
|
User.find(session[:otp_user_id])
|
|
|
|
elsif user_params[:login]
|
|
|
|
User.by_login(user_params[:login])
|
|
|
|
end
|
2015-05-08 20:41:53 -04:00
|
|
|
end
|
|
|
|
end
|
2015-12-27 12:03:06 -05:00
|
|
|
|
2017-09-19 03:44:58 -04:00
|
|
|
def stored_redirect_uri
|
|
|
|
@redirect_to ||= stored_location_for(:redirect)
|
|
|
|
end
|
|
|
|
|
|
|
|
def store_redirect_uri
|
|
|
|
redirect_uri =
|
2015-08-12 11:20:01 -04:00
|
|
|
if request.referer.present? && (params['redirect_to_referer'] == 'yes')
|
2017-09-19 03:44:58 -04:00
|
|
|
URI(request.referer)
|
2015-08-12 11:20:01 -04:00
|
|
|
else
|
2017-09-19 03:44:58 -04:00
|
|
|
URI(request.url)
|
2015-08-12 11:20:01 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Prevent a 'you are already signed in' message directly after signing:
|
|
|
|
# we should never redirect to '/users/sign_in' after signing in successfully.
|
2017-09-19 03:44:58 -04:00
|
|
|
return true if redirect_uri.path == new_user_session_path
|
|
|
|
|
2018-05-02 14:25:21 -04:00
|
|
|
redirect_to = redirect_uri.to_s if host_allowed?(redirect_uri)
|
2017-09-19 03:44:58 -04:00
|
|
|
|
|
|
|
@redirect_to = redirect_to
|
|
|
|
store_location_for(:redirect, redirect_to)
|
|
|
|
end
|
|
|
|
|
2016-04-07 05:19:29 -04:00
|
|
|
def two_factor_enabled?
|
2017-09-19 03:44:58 -04:00
|
|
|
find_user&.two_factor_enabled?
|
2016-04-07 05:19:29 -04:00
|
|
|
end
|
|
|
|
|
2015-05-27 11:40:21 -04:00
|
|
|
def auto_sign_in_with_provider
|
2018-07-13 06:39:31 -04:00
|
|
|
return unless Gitlab::Auth.omniauth_enabled?
|
|
|
|
|
2015-05-27 11:40:21 -04:00
|
|
|
provider = Gitlab.config.omniauth.auto_sign_in_with_provider
|
|
|
|
return unless provider.present?
|
|
|
|
|
2017-03-23 09:49:59 -04:00
|
|
|
# If a "auto_sign_in" query parameter is set to a falsy value, don't auto sign-in.
|
|
|
|
# Otherwise, the default is to auto sign-in.
|
|
|
|
return if Gitlab::Utils.to_boolean(params[:auto_sign_in]) == false
|
|
|
|
|
2015-12-27 12:03:06 -05:00
|
|
|
# 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
|
2015-05-27 11:40:21 -04:00
|
|
|
# to do nothing to prevent redirection loops with certain Omniauth providers.
|
|
|
|
return unless flash[:alert].blank? || flash[:alert] == I18n.t('devise.failure.unauthenticated')
|
2015-12-27 12:03:06 -05:00
|
|
|
|
2015-05-27 11:40:21 -04:00
|
|
|
# Prevent alert from popping up on the first page shown after authentication.
|
2015-12-27 12:03:06 -05:00
|
|
|
flash[:alert] = nil
|
|
|
|
|
2016-07-25 13:40:40 -04:00
|
|
|
redirect_to omniauth_authorize_path(:user, provider)
|
2015-05-27 11:40:21 -04:00
|
|
|
end
|
|
|
|
|
2015-05-08 20:41:53 -04:00
|
|
|
def valid_otp_attempt?(user)
|
2015-09-19 21:16:18 -04:00
|
|
|
user.validate_and_consume_otp!(user_params[:otp_attempt]) ||
|
2016-12-15 17:14:20 -05:00
|
|
|
user.invalidate_otp_backup_code!(user_params[:otp_attempt])
|
2015-05-05 22:16:45 -04:00
|
|
|
end
|
2015-07-03 07:54:50 -04:00
|
|
|
|
2017-08-23 00:40:16 -04:00
|
|
|
def log_audit_event(user, resource, options = {})
|
2017-09-27 23:54:52 -04:00
|
|
|
Gitlab::AppLogger.info("Successful Login: username=#{resource.username} ip=#{request.remote_ip} method=#{options[:with]} admin=#{resource.admin?}")
|
2017-06-21 09:48:12 -04:00
|
|
|
AuditEventService.new(user, user, options)
|
|
|
|
.for_authentication.security_event
|
2015-07-03 07:54:50 -04:00
|
|
|
end
|
2015-12-28 15:21:34 -05:00
|
|
|
|
2016-10-05 10:41:32 -04:00
|
|
|
def log_user_activity(user)
|
2017-05-23 10:23:43 -04:00
|
|
|
login_counter.increment
|
2020-01-06 04:07:42 -05:00
|
|
|
Users::ActivityService.new(user).execute
|
2016-10-05 10:41:32 -04:00
|
|
|
end
|
|
|
|
|
2015-12-28 15:21:34 -05:00
|
|
|
def load_recaptcha
|
|
|
|
Gitlab::Recaptcha.load_configurations!
|
|
|
|
end
|
2016-06-06 00:52:06 -04:00
|
|
|
|
2019-07-18 04:27:02 -04:00
|
|
|
def unverified_anonymous_user?
|
|
|
|
exceeded_failed_login_attempts? || exceeded_anonymous_sessions?
|
|
|
|
end
|
|
|
|
|
|
|
|
def exceeded_failed_login_attempts?
|
|
|
|
session.fetch(:failed_login_attempts, 0) > MAX_FAILED_LOGIN_ATTEMPTS
|
|
|
|
end
|
|
|
|
|
|
|
|
def exceeded_anonymous_sessions?
|
2020-08-28 05:10:32 -04:00
|
|
|
Gitlab::AnonymousSession.new(request.remote_ip).session_count >= MAX_FAILED_LOGIN_ATTEMPTS
|
2019-07-18 04:27:02 -04:00
|
|
|
end
|
|
|
|
|
2016-06-06 00:52:06 -04:00
|
|
|
def authentication_method
|
|
|
|
if user_params[:otp_attempt]
|
|
|
|
"two-factor"
|
2020-09-07 08:08:27 -04:00
|
|
|
elsif user_params[:device_response] && Feature.enabled?(:webauthn)
|
|
|
|
"two-factor-via-webauthn-device"
|
|
|
|
elsif user_params[:device_response] && !Feature.enabled?(:webauthn)
|
2016-06-06 00:52:06 -04:00
|
|
|
"two-factor-via-u2f-device"
|
|
|
|
else
|
|
|
|
"standard"
|
|
|
|
end
|
|
|
|
end
|
2020-08-10 14:09:54 -04:00
|
|
|
|
|
|
|
def set_invite_params
|
|
|
|
@invite_email = ActionController::Base.helpers.sanitize(params[:invite_email])
|
|
|
|
end
|
2014-07-11 06:24:11 -04:00
|
|
|
end
|
2019-09-13 09:26:31 -04:00
|
|
|
|
|
|
|
SessionsController.prepend_if_ee('EE::SessionsController')
|