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

158 lines
4.8 KiB
Ruby
Raw Normal View History

2014-09-02 18:07:02 +05:30
class SessionsController < Devise::SessionsController
2015-09-11 14:41:01 +05:30
include AuthenticatesWithTwoFactor
include Devise::Controllers::Rememberable
include Recaptcha::ClientHelper
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]
2016-04-02 18:10:28 +05:30
2017-09-10 17:25:29 +05:30
# Explicitly call protect from forgery before anything else. Otherwise the
# CSFR-token might be cleared before authentication is done. This was the case
# when LDAP was enabled and the `OmniauthCallbacksController` is loaded
#
# *Note:* `prepend: true` is the default for rails4, but this will be changed
# to `prepend: false` in rails5.
protect_from_forgery prepend: true, with: :exception
2016-06-02 11:05:42 +05:30
prepend_before_action :check_initial_setup, only: [:new]
prepend_before_action :authenticate_with_two_factor,
if: :two_factor_enabled?, only: [:create]
2015-09-11 14:41:01 +05:30
prepend_before_action :store_redirect_path, only: [:new]
2016-06-02 11:05:42 +05:30
2015-09-11 14:41:01 +05:30
before_action :auto_sign_in_with_provider, only: [:new]
before_action :load_recaptcha
2015-09-11 14:41:01 +05:30
2014-09-02 18:07:02 +05:30
def new
set_minimum_password_length
2017-09-10 17:25:29 +05:30
@ldap_servers = Gitlab::LDAP::Config.available_servers
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?
resource.update_attributes(reset_password_token: nil,
reset_password_sent_at: nil)
end
2017-08-17 22:00:37 +05:30
# hide the signed-in notification
flash[:notice] = nil
log_audit_event(current_user, 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
super
# hide the signed_out notice
flash[:notice] = nil
end
2015-09-11 14:41:01 +05:30
private
2017-09-10 17:25:29 +05:30
def login_counter
@login_counter ||= Gitlab::Metrics.counter(:user_session_logins_total, 'User sign in count')
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.
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
2017-09-10 17:25:29 +05:30
return unless user && user.require_password_creation?
2016-06-02 11:05:42 +05:30
2017-09-10 17:25:29 +05:30
Users::UpdateService.new(user).execute do |user|
@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),
2016-06-02 11:05:42 +05:30
notice: "Please create a password for your new account."
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
2016-06-02 11:05:42 +05:30
if session[:otp_user_id]
2015-09-11 14:41:01 +05:30
User.find(session[:otp_user_id])
2016-06-02 11:05:42 +05:30
elsif user_params[:login]
User.by_login(user_params[:login])
2015-09-11 14:41:01 +05:30
end
end
2015-09-11 14:41:01 +05:30
def store_redirect_path
2015-04-26 12:48:37 +05:30
redirect_path =
if request.referer.present? && (params['redirect_to_referer'] == 'yes')
referer_uri = URI(request.referer)
if referer_uri.host == Gitlab.config.gitlab.host
2017-08-17 22:00:37 +05:30
referer_uri.request_uri
2015-04-26 12:48:37 +05:30
else
request.fullpath
end
else
request.fullpath
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.
2017-09-10 17:25:29 +05:30
unless URI(redirect_path).path == new_user_session_path
2014-09-02 18:07:02 +05:30
store_location_for(:redirect, redirect_path)
end
2015-09-11 14:41:01 +05:30
end
2014-09-02 18:07:02 +05:30
2016-06-02 11:05:42 +05:30
def two_factor_enabled?
find_user.try(:two_factor_enabled?)
end
2015-09-11 14:41:01 +05:30
def auto_sign_in_with_provider
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)
2015-09-25 12:07:36 +05:30
user.validate_and_consume_otp!(user_params[:otp_attempt]) ||
2017-08-17 22:00:37 +05:30
user.invalidate_otp_backup_code!(user_params[:otp_attempt])
2015-09-11 14:41:01 +05:30
end
def log_audit_event(user, options = {})
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
2017-08-17 22:00:37 +05:30
Users::ActivityService.new(user, 'login').execute
end
def load_recaptcha
Gitlab::Recaptcha.load_configurations!
end
def authentication_method
if user_params[:otp_attempt]
"two-factor"
elsif user_params[:device_response]
"two-factor-via-u2f-device"
else
"standard"
end
end
2014-09-02 18:07:02 +05:30
end