gitlab-org--gitlab-foss/app/controllers/application_controller.rb

307 lines
8.8 KiB
Ruby
Raw Normal View History

require 'gon'
2015-08-04 22:21:12 +00:00
require 'fogbugz'
2013-08-21 09:33:12 +00:00
class ApplicationController < ActionController::Base
include Gitlab::CurrentSettings
include Gitlab::GonHelper
include GitlabRoutingHelper
include PageLayoutHelper
include SentryHelper
include WorkhorseHelper
before_action :authenticate_user_from_private_token!
before_action :authenticate_user!
2015-11-12 04:25:31 +00:00
before_action :validate_user_service_ticket!
before_action :check_password_expiration
before_action :check_2fa_requirement
before_action :ldap_security_check
before_action :sentry_context
before_action :default_headers
before_action :add_gon_variables
before_action :configure_permitted_parameters, if: :devise_controller?
before_action :require_email, unless: :devise_controller?
protect_from_forgery with: :exception
2016-08-08 18:55:13 +00:00
helper_method :can?, :current_application_settings
helper_method :import_sources_enabled?, :github_import_enabled?, :gitea_import_enabled?, :github_import_configured?, :gitlab_import_enabled?, :gitlab_import_configured?, :bitbucket_import_enabled?, :bitbucket_import_configured?, :google_code_import_enabled?, :fogbugz_import_enabled?, :git_import_enabled?, :gitlab_project_import_enabled?
helper_method :two_factor_grace_period_expired?, :two_factor_skippable?
2011-10-08 21:36:38 +00:00
rescue_from Encoding::CompatibilityError do |exception|
2012-11-06 20:15:25 +00:00
log_exception(exception)
2012-09-26 20:24:52 +00:00
render "errors/encoding", layout: "errors", status: 500
end
rescue_from ActiveRecord::RecordNotFound do |exception|
2012-11-06 20:15:25 +00:00
log_exception(exception)
2015-10-09 17:07:29 +00:00
render_404
2011-10-09 21:15:28 +00:00
end
rescue_from Gitlab::Access::AccessDeniedError do |exception|
render_403
end
rescue_from Gitlab::Auth::TooManyIps do |e|
head :forbidden, retry_after: Gitlab::Auth::UniqueIpsLimiter.config.unique_ips_limit_time_window
end
def redirect_back_or_default(default: root_path, options: {})
redirect_to request.referer.present? ? :back : default, options
end
def not_found
render_404
end
def route_not_found
if current_user
not_found
else
redirect_to new_user_session_path
end
end
protected
2011-10-08 21:36:38 +00:00
# This filter handles both private tokens and personal access tokens
def authenticate_user_from_private_token!
token = params[:private_token].presence || request.headers['PRIVATE-TOKEN'].presence
return unless token.present?
user = User.find_by_authentication_token(token) || User.find_by_personal_access_token(token)
2017-02-28 21:19:52 +00:00
if user && can?(user, :log_in)
# Notice we are passing store false, so the user is not
# actually stored in the session and a token is needed
# for every request. If you want the token to work as a
# sign in token, you can simply remove store: false.
sign_in user, store: false
end
end
2012-11-06 20:15:25 +00:00
def log_exception(exception)
application_trace = ActionDispatch::ExceptionWrapper.new(env, exception).application_trace
application_trace.map!{ |t| " #{t}\n" }
logger.error "\n#{exception.class.name} (#{exception.message}):\n#{application_trace.join}"
end
def after_sign_in_path_for(resource)
stored_location_for(:redirect) || stored_location_for(resource) || root_path
2012-04-13 05:12:34 +00:00
end
def after_sign_out_path_for(resource)
2016-04-30 15:38:39 +00:00
current_application_settings.after_sign_out_path.presence || new_user_session_path
end
def can?(object, action, subject = :global)
2016-08-08 18:55:13 +00:00
Ability.allowed?(object, action, subject)
2011-10-08 21:36:38 +00:00
end
2011-10-17 10:39:03 +00:00
def access_denied!
2012-09-26 20:24:52 +00:00
render "errors/access_denied", layout: "errors", status: 404
end
def git_not_found!
render "errors/git_not_found.html", layout: "errors", status: 404
2011-10-08 21:36:38 +00:00
end
2014-02-07 16:59:55 +00:00
def render_403
head :forbidden
2011-10-14 15:08:25 +00:00
end
2011-10-15 15:51:58 +00:00
2014-02-07 16:59:55 +00:00
def render_404
respond_to do |format|
2016-09-01 15:20:29 +00:00
format.html do
render file: Rails.root.join("public", "404"), layout: false, status: "404"
end
2016-09-01 15:20:29 +00:00
format.any { head :not_found }
end
2012-12-05 03:14:05 +00:00
end
2011-11-20 20:32:12 +00:00
def no_cache_headers
response.headers["Cache-Control"] = "no-cache, no-store, max-age=0, must-revalidate"
response.headers["Pragma"] = "no-cache"
response.headers["Expires"] = "Fri, 01 Jan 1990 00:00:00 GMT"
end
2012-01-27 23:49:14 +00:00
2013-02-02 18:32:13 +00:00
def default_headers
headers['X-Frame-Options'] = 'DENY'
headers['X-XSS-Protection'] = '1; mode=block'
2013-12-27 11:38:29 +00:00
headers['X-UA-Compatible'] = 'IE=edge'
headers['X-Content-Type-Options'] = 'nosniff'
2013-02-02 18:32:13 +00:00
end
2015-11-12 04:25:31 +00:00
def validate_user_service_ticket!
return unless signed_in? && session[:service_tickets]
valid = session[:service_tickets].all? do |provider, ticket|
Gitlab::OAuth::Session.valid?(provider, ticket)
end
unless valid
session[:service_tickets] = nil
sign_out current_user
redirect_to new_user_session_path
end
end
def check_password_expiration
if current_user && current_user.password_expires_at && current_user.password_expires_at < Time.now && !current_user.ldap_user?
2017-02-21 22:31:14 +00:00
return redirect_to new_profile_password_path
end
end
def check_2fa_requirement
if two_factor_authentication_required? && current_user && !current_user.two_factor_enabled? && !skip_two_factor?
redirect_to profile_two_factor_auth_path
2015-12-18 20:29:13 +00:00
end
end
def ldap_security_check
if current_user && current_user.requires_ldap_check?
2016-03-10 11:37:14 +00:00
return unless current_user.try_obtain_ldap_lease
2016-03-09 18:11:24 +00:00
unless Gitlab::LDAP::Access.allowed?(current_user)
sign_out current_user
flash[:alert] = "Access denied for your LDAP account."
redirect_to new_user_session_path
end
end
end
def event_filter
# Split using comma to maintain backward compatibility Ex/ "filter1,filter2"
filters = cookies['event_filter'].split(',')[0] if cookies['event_filter'].present?
@event_filter ||= EventFilter.new(filters)
end
def gitlab_ldap_access(&block)
2017-02-21 23:59:42 +00:00
Gitlab::LDAP::Access.open { |access| yield(access) }
end
# JSON for infinite scroll via Pager object
def pager_json(partial, count, locals = {})
html = render_to_string(
partial,
locals: locals,
layout: false,
formats: [:html]
)
render json: {
html: html,
count: count
}
end
2016-02-03 23:21:14 +00:00
def view_to_html_string(partial, locals = {})
render_to_string(
2016-02-03 23:21:14 +00:00
partial,
locals: locals,
layout: false,
formats: [:html]
)
end
def configure_permitted_parameters
devise_parameter_sanitizer.permit(:sign_in, keys: [:username, :email, :password, :login, :remember_me, :otp_attempt])
end
def hexdigest(string)
Digest::SHA1.hexdigest string
end
def require_email
if current_user && current_user.temp_oauth_email? && session[:impersonator_id].nil?
2017-02-21 22:31:14 +00:00
return redirect_to profile_path, notice: 'Please complete your profile with email address'
end
end
def import_sources_enabled?
!current_application_settings.import_sources.empty?
end
2015-02-17 21:52:32 +00:00
def github_import_enabled?
current_application_settings.import_sources.include?('github')
end
def gitea_import_enabled?
current_application_settings.import_sources.include?('gitea')
2016-10-17 15:58:57 +00:00
end
def github_import_configured?
Gitlab::OAuth::Provider.enabled?(:github)
2015-02-17 21:52:32 +00:00
end
def gitlab_import_enabled?
request.host != 'gitlab.com' && current_application_settings.import_sources.include?('gitlab')
end
def gitlab_import_configured?
Gitlab::OAuth::Provider.enabled?(:gitlab)
2015-02-17 21:52:32 +00:00
end
def bitbucket_import_enabled?
current_application_settings.import_sources.include?('bitbucket')
end
def bitbucket_import_configured?
Gitlab::OAuth::Provider.enabled?(:bitbucket)
2015-02-17 21:52:32 +00:00
end
def google_code_import_enabled?
current_application_settings.import_sources.include?('google_code')
end
2015-08-04 22:21:12 +00:00
def fogbugz_import_enabled?
current_application_settings.import_sources.include?('fogbugz')
end
def git_import_enabled?
current_application_settings.import_sources.include?('git')
end
def gitlab_project_import_enabled?
current_application_settings.import_sources.include?('gitlab_project')
end
2015-12-18 20:29:13 +00:00
def two_factor_authentication_required?
2017-01-24 21:09:58 +00:00
current_application_settings.require_two_factor_authentication ||
current_user.try(:require_two_factor_authentication)
2015-12-18 20:29:13 +00:00
end
2015-12-24 02:02:52 +00:00
def two_factor_grace_period
2017-01-24 21:09:58 +00:00
if current_user.try(:require_two_factor_authentication)
[
current_application_settings.two_factor_grace_period,
current_user.two_factor_grace_period
].min
else
current_application_settings.two_factor_grace_period
end
2015-12-24 02:02:52 +00:00
end
def two_factor_grace_period_expired?
date = current_user.otp_grace_period_started_at
2015-12-24 02:02:52 +00:00
date && (date + two_factor_grace_period.hours) < Time.current
end
def two_factor_skippable?
two_factor_authentication_required? &&
!current_user.two_factor_enabled? &&
!two_factor_grace_period_expired?
end
2015-12-24 02:02:52 +00:00
def skip_two_factor?
session[:skip_tfa] && session[:skip_tfa] > Time.current
end
# U2F (universal 2nd factor) devices need a unique identifier for the application
# to perform authentication.
# https://developers.yubico.com/U2F/App_ID.html
def u2f_app_id
request.base_url
end
2011-10-08 21:36:38 +00:00
end