debian-mirror-gitlab/app/controllers/sessions_controller.rb

327 lines
11 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
class SessionsController < Devise::SessionsController
2018-10-15 14:42:47 +05:30
include InternalRedirect
2015-09-11 14:41:01 +05:30
include AuthenticatesWithTwoFactor
include Devise::Controllers::Rememberable
include Recaptcha::ClientHelper
2018-11-08 19:23:39 +05:30
include Recaptcha::Verify
2020-05-24 23:13:21 +05:30
include RendersLdapServers
include KnownSignIn
2020-09-03 11:15:55 +05:30
include Gitlab::Utils::StrongMemoize
2015-09-11 14:41:01 +05:30
2017-08-17 22:00:37 +05:30
skip_before_action :check_two_factor_requirement, only: [:destroy]
2020-11-24 15:15:51 +05:30
skip_before_action :check_password_expiration, only: [:destroy]
2019-07-31 22:56:46 +05:30
# replaced with :require_no_authentication_without_flash
skip_before_action :require_no_authentication, only: [:new, :create]
2016-04-02 18:10:28 +05:30
2016-06-02 11:05:42 +05:30
prepend_before_action :check_initial_setup, only: [:new]
prepend_before_action :authenticate_with_two_factor,
2019-10-12 21:52:04 +05:30
if: -> { action_name == 'create' && two_factor_enabled? }
2018-11-08 19:23:39 +05:30
prepend_before_action :check_captcha, only: [:create]
2018-03-17 18:26:18 +05:30
prepend_before_action :store_redirect_uri, only: [:new]
2019-07-31 22:56:46 +05:30
prepend_before_action :require_no_authentication_without_flash, only: [:new, :create]
2021-06-08 01:23:25 +05:30
prepend_before_action :check_forbidden_password_based_login, if: -> { action_name == 'create' && password_based_login? }
2019-10-12 21:52:04 +05:30
prepend_before_action :ensure_password_authentication_enabled!, if: -> { action_name == 'create' && password_based_login? }
2019-06-05 12:25:43 +05:30
2015-09-11 14:41:01 +05:30
before_action :auto_sign_in_with_provider, only: [:new]
2019-09-04 21:01:54 +05:30
before_action :store_unauthenticated_sessions, only: [:new]
before_action :save_failed_login, if: :action_new_and_failed_login?
before_action :load_recaptcha
2020-10-24 23:57:45 +05:30
before_action :set_invite_params, only: [:new]
2020-11-24 15:15:51 +05:30
before_action do
push_frontend_feature_flag(:webauthn)
end
2015-09-11 14:41:01 +05:30
2019-09-04 21:01:54 +05:30
after_action :log_failed_login, if: :action_new_and_failed_login?
2020-05-24 23:13:21 +05:30
after_action :verify_known_sign_in, only: [:create]
2019-09-04 21:01:54 +05:30
helper_method :captcha_enabled?, :captcha_on_login_required?
2018-03-17 18:26:18 +05:30
2019-10-12 21:52:04 +05:30
# 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-04-08 14:13:33 +05:30
protect_from_forgery with: :exception, prepend: true, except: :destroy
2018-11-08 19:23:39 +05:30
2021-01-03 14:25:43 +05:30
feature_category :authentication_and_authorization
2019-12-04 20:38:33 +05:30
CAPTCHA_HEADER = 'X-GitLab-Show-Login-Captcha'
2019-09-04 21:01:54 +05:30
MAX_FAILED_LOGIN_ATTEMPTS = 5
2018-11-08 19:23:39 +05:30
2014-09-02 18:07:02 +05:30
def new
set_minimum_password_length
2015-09-11 14:41:01 +05:30
super
end
def create
super do |resource|
# User has successfully signed in, so clear any unused reset token
if resource.reset_password_token.present?
2018-11-18 11:00:15 +05:30
resource.update(reset_password_token: nil,
reset_password_sent_at: nil)
2015-09-11 14:41:01 +05:30
end
2018-03-17 18:26:18 +05:30
2019-12-21 20:55:43 +05:30
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
2018-03-17 18:26:18 +05:30
log_audit_event(current_user, resource, with: authentication_method)
2017-08-17 22:00:37 +05:30
log_user_activity(current_user)
2015-09-11 14:41:01 +05:30
end
end
2017-08-17 22:00:37 +05:30
def destroy
2018-03-17 18:26:18 +05:30
Gitlab::AppLogger.info("User Logout: username=#{current_user.username} ip=#{request.remote_ip}")
2017-08-17 22:00:37 +05:30
super
# hide the signed_out notice
flash[:notice] = nil
end
2015-09-11 14:41:01 +05:30
private
2019-07-31 22:56:46 +05:30
def require_no_authentication_without_flash
require_no_authentication
if flash[:alert] == I18n.t('devise.failure.already_authenticated')
flash[:alert] = nil
end
end
2018-11-08 19:23:39 +05:30
def captcha_enabled?
request.headers[CAPTCHA_HEADER] && Gitlab::Recaptcha.enabled?
end
2019-09-04 21:01:54 +05:30
def captcha_on_login_required?
Gitlab::Recaptcha.enabled_on_login? && unverified_anonymous_user?
end
2018-11-08 19:23:39 +05:30
# From https://github.com/plataformatec/devise/wiki/How-To:-Use-Recaptcha-with-Devise#devisepasswordscontroller
def check_captcha
return unless user_params[:password].present?
2019-09-04 21:01:54 +05:30
return unless captcha_enabled? || captcha_on_login_required?
2018-11-08 19:23:39 +05:30
return unless Gitlab::Recaptcha.load_configurations!
if verify_recaptcha
increment_successful_login_captcha_counter
else
increment_failed_login_captcha_counter
self.resource = resource_class.new
2019-07-31 22:56:46 +05:30
flash[:alert] = _('There was an error with the reCAPTCHA. Please solve the reCAPTCHA again.')
2018-11-08 19:23:39 +05:30
flash.delete :recaptcha_error
respond_with_navigational(resource) { render :new }
end
end
def increment_failed_login_captcha_counter
Gitlab::Metrics.counter(
:failed_login_captcha_total,
2019-12-04 20:38:33 +05:30
'Number of failed CAPTCHA attempts for logins'
2018-11-08 19:23:39 +05:30
).increment
end
def increment_successful_login_captcha_counter
Gitlab::Metrics.counter(
:successful_login_captcha_total,
2019-12-04 20:38:33 +05:30
'Number of successful CAPTCHA attempts for logins'
2018-11-08 19:23:39 +05:30
).increment
end
2018-11-18 11:00:15 +05:30
##
# 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
2018-03-17 18:26:18 +05:30
def log_failed_login
Gitlab::AppLogger.info("Failed Login: username=#{user_params[:login]} ip=#{request.remote_ip}")
end
2019-09-04 21:01:54 +05:30
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
2018-03-17 18:26:18 +05:30
def failed_login?
2019-02-15 15:39:39 +05:30
(options = request.env["warden.options"]) && options[:action] == "unauthenticated"
2018-03-17 18:26:18 +05:30
end
2020-11-24 15:15:51 +05:30
# counting sessions per IP lets us check if there are associated multiple
2019-09-04 21:01:54 +05:30
# 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-11-24 15:15:51 +05:30
Gitlab::AnonymousSession.new(request.remote_ip).count_session_ip
2019-09-04 21:01:54 +05:30
end
2016-06-02 11:05:42 +05:30
# Handle an "initial setup" state, where there's only one user, it's an admin,
# and they require a password change.
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2016-06-02 11:05:42 +05:30
def check_initial_setup
return unless User.limit(2).count == 1 # Count as much 2 to know if we have exactly one
2016-06-02 11:05:42 +05:30
user = User.admins.last
2018-03-17 18:26:18 +05:30
return unless user && user.require_password_creation_for_web?
2016-06-02 11:05:42 +05:30
2018-03-17 18:26:18 +05:30
Users::UpdateService.new(current_user, user: user).execute do |user|
2017-09-10 17:25:29 +05:30
@token = user.generate_reset_token
end
2016-06-02 11:05:42 +05:30
2017-09-10 17:25:29 +05:30
redirect_to edit_user_password_path(reset_password_token: @token),
2019-07-31 22:56:46 +05:30
notice: _("Please create a password for your new account.")
2016-06-02 11:05:42 +05:30
end
2018-12-05 23:21:45 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2016-06-02 11:05:42 +05:30
2019-06-05 12:25:43 +05:30
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-09-11 14:41:01 +05:30
def user_params
params.require(:user).permit(:login, :password, :remember_me, :otp_attempt, :device_response)
2015-09-11 14:41:01 +05:30
end
def find_user
2020-09-03 11:15:55 +05:30
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-09-11 14:41:01 +05:30
end
end
2018-03-17 18:26:18 +05:30
def stored_redirect_uri
@redirect_to ||= stored_location_for(:redirect)
end
def store_redirect_uri
redirect_uri =
2015-04-26 12:48:37 +05:30
if request.referer.present? && (params['redirect_to_referer'] == 'yes')
2018-03-17 18:26:18 +05:30
URI(request.referer)
2015-04-26 12:48:37 +05:30
else
2018-03-17 18:26:18 +05:30
URI(request.url)
2015-04-26 12:48:37 +05:30
end
2014-09-02 18:07:02 +05:30
# Prevent a 'you are already signed in' message directly after signing:
# we should never redirect to '/users/sign_in' after signing in successfully.
2018-03-17 18:26:18 +05:30
return true if redirect_uri.path == new_user_session_path
2018-10-15 14:42:47 +05:30
redirect_to = redirect_uri.to_s if host_allowed?(redirect_uri)
2018-03-17 18:26:18 +05:30
@redirect_to = redirect_to
store_location_for(:redirect, redirect_to)
end
2016-06-02 11:05:42 +05:30
def two_factor_enabled?
2018-03-17 18:26:18 +05:30
find_user&.two_factor_enabled?
2016-06-02 11:05:42 +05:30
end
2015-09-11 14:41:01 +05:30
def auto_sign_in_with_provider
2018-11-18 11:00:15 +05:30
return unless Gitlab::Auth.omniauth_enabled?
2015-09-11 14:41:01 +05:30
provider = Gitlab.config.omniauth.auto_sign_in_with_provider
return unless provider.present?
2017-09-10 17:25:29 +05:30
# 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
# 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-09-11 14:41:01 +05:30
# to do nothing to prevent redirection loops with certain Omniauth providers.
return unless flash[:alert].blank? || flash[:alert] == I18n.t('devise.failure.unauthenticated')
2015-09-11 14:41:01 +05:30
# Prevent alert from popping up on the first page shown after authentication.
flash[:alert] = nil
2016-09-13 17:45:13 +05:30
redirect_to omniauth_authorize_path(:user, provider)
2015-09-11 14:41:01 +05:30
end
def valid_otp_attempt?(user)
2021-01-03 14:25:43 +05:30
otp_validation_result =
::Users::ValidateOtpService.new(user).execute(user_params[:otp_attempt])
return true if otp_validation_result[:status] == :success
user.invalidate_otp_backup_code!(user_params[:otp_attempt])
2015-09-11 14:41:01 +05:30
end
2018-03-17 18:26:18 +05:30
def log_audit_event(user, resource, options = {})
Gitlab::AppLogger.info("Successful Login: username=#{resource.username} ip=#{request.remote_ip} method=#{options[:with]} admin=#{resource.admin?}")
2017-09-10 17:25:29 +05:30
AuditEventService.new(user, user, options)
.for_authentication.security_event
2015-09-11 14:41:01 +05:30
end
2017-08-17 22:00:37 +05:30
def log_user_activity(user)
2017-09-10 17:25:29 +05:30
login_counter.increment
2020-03-13 15:44:24 +05:30
Users::ActivityService.new(user).execute
2017-08-17 22:00:37 +05:30
end
def load_recaptcha
Gitlab::Recaptcha.load_configurations!
end
2019-09-04 21:01:54 +05:30
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-11-24 15:15:51 +05:30
Gitlab::AnonymousSession.new(request.remote_ip).session_count >= MAX_FAILED_LOGIN_ATTEMPTS
2019-09-04 21:01:54 +05:30
end
def authentication_method
if user_params[:otp_attempt]
2021-01-29 00:20:46 +05:30
AuthenticationEvent::TWO_FACTOR
2020-11-24 15:15:51 +05:30
elsif user_params[:device_response] && Feature.enabled?(:webauthn)
2021-01-29 00:20:46 +05:30
AuthenticationEvent::TWO_FACTOR_WEBAUTHN
2020-11-24 15:15:51 +05:30
elsif user_params[:device_response] && !Feature.enabled?(:webauthn)
2021-01-29 00:20:46 +05:30
AuthenticationEvent::TWO_FACTOR_U2F
else
2021-01-29 00:20:46 +05:30
AuthenticationEvent::STANDARD
end
end
2019-12-26 22:10:19 +05:30
2020-10-24 23:57:45 +05:30
def set_invite_params
@invite_email = ActionController::Base.helpers.sanitize(params[:invite_email])
2019-12-26 22:10:19 +05:30
end
2021-06-08 01:23:25 +05:30
def check_forbidden_password_based_login
if find_user&.password_based_login_forbidden?
flash[:alert] = _('You are not allowed to log in using password')
redirect_to new_user_session_path
end
end
2014-09-02 18:07:02 +05:30
end
2019-12-04 20:38:33 +05:30
2021-06-08 01:23:25 +05:30
SessionsController.prepend_mod_with('SessionsController')