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

375 lines
10 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
2015-03-12 11:08:48 -04:00
class GroupsController < Groups::ApplicationController
include API::Helpers::RelatedResourcesHelpers
include IssuableCollectionsAction
include ParamsBackwardCompatibility
include PreviewMarkdown
include RecordUserLastActivity
include SendFileUpload
include FiltersEvents
include Recaptcha::Verify
extend ::Gitlab::Utils::Override
2012-10-02 13:42:15 -04:00
respond_to :html
prepend_before_action(only: [:show, :issues]) { authenticate_sessionless_user!(:rss) }
prepend_before_action(only: [:issues_calendar]) { authenticate_sessionless_user!(:ics) }
prepend_before_action :ensure_export_enabled, only: [:export, :download_export]
prepend_before_action :check_captcha, only: :create, if: -> { captcha_enabled? }
2016-03-20 16:03:53 -04:00
before_action :authenticate_user!, only: [:new, :create]
before_action :group, except: [:index, :new, :create]
2012-10-02 13:42:15 -04:00
# Authorize
before_action :authorize_admin_group!, only: [:edit, :update, :destroy, :projects, :transfer, :export, :download_export]
before_action :authorize_create_group!, only: [:new]
before_action :load_recaptcha, only: [:new], if: -> { captcha_required? }
2013-01-24 10:47:09 -05:00
before_action :group_projects, only: [:projects, :activity, :issues, :merge_requests]
before_action :event_filter, only: [:activity]
2013-06-08 09:26:57 -04:00
before_action :user_actions, only: [:show]
before_action do
push_frontend_feature_flag(:vue_issuables_list, @group)
push_frontend_feature_flag(:iteration_cadences, @group, default_enabled: :yaml)
end
before_action :export_rate_limit, only: [:export, :download_export]
helper_method :captcha_required?
skip_cross_project_access_check :index, :new, :create, :edit, :update,
:destroy, :projects
# When loading show as an atom feed, we render events that could leak cross
# project information
skip_cross_project_access_check :show, if: -> { request.format.html? }
layout :determine_layout
feature_category :subgroups, [
:index, :new, :create, :show, :edit, :update,
:destroy, :details, :transfer, :activity
]
feature_category :issue_tracking, [:issues, :issues_calendar, :preview_markdown]
feature_category :code_review, [:merge_requests, :unfoldered_environment_names]
feature_category :projects, [:projects]
feature_category :importers, [:export, :download_export]
def index
redirect_to(current_user ? dashboard_groups_path : explore_groups_path)
end
2013-01-24 10:47:09 -05:00
def new
@group = Group.new(params.permit(:parent_id))
2013-01-24 10:47:09 -05:00
end
def create
2016-03-17 18:42:46 -04:00
@group = Groups::CreateService.new(current_user, group_params).execute
2013-01-24 10:47:09 -05:00
2016-03-17 18:42:46 -04:00
if @group.persisted?
successful_creation_hooks
2017-03-01 14:34:29 -05:00
notice = if @group.chat_team.present?
"Group '#{@group.name}' and its Mattermost team were successfully created."
else
2017-03-03 03:01:54 -05:00
"Group '#{@group.name}' was successfully created."
2017-03-01 14:34:29 -05:00
end
redirect_to @group, notice: notice
2013-01-24 10:47:09 -05:00
else
render action: "new"
end
end
2012-10-02 13:42:15 -04:00
def show
respond_to do |format|
format.html do
if @group.import_state&.in_progress?
redirect_to group_import_path(@group)
else
render_show_html
end
end
2015-02-18 03:16:42 -05:00
2015-02-18 12:38:46 -05:00
format.atom do
render_details_view_atom
end
end
end
def details
respond_to do |format|
format.html do
render_details_html
end
format.atom do
render_details_view_atom
2015-02-18 12:38:46 -05:00
end
2012-10-02 13:42:15 -04:00
end
end
def activity
respond_to do |format|
format.html
format.json do
load_events
pager_json("events/_events", @events.count { |event| event.visible_to_user?(current_user) })
end
end
end
def edit
@badge_api_endpoint = expose_path(api_v4_groups_badges_path(id: @group.id))
end
2016-06-22 09:50:19 -04:00
def projects
@projects = @group.projects.with_statistics.page(params[:page])
2016-06-22 09:50:19 -04:00
end
def update
if Groups::UpdateService.new(@group, current_user, group_params).execute
notice = "Group '#{@group.name}' was successfully updated."
redirect_to edit_group_origin_location, notice: notice
else
@group.reset
render action: "edit"
end
end
def edit_group_origin_location
if params.dig(:group, :redirect_target) == 'repository_settings'
group_settings_repository_path(@group, anchor: 'js-default-branch-name')
else
edit_group_path(@group, anchor: params[:update_section])
end
end
def destroy
Groups::DestroyService.new(@group, current_user).async_execute
redirect_to root_path, status: :found, alert: "Group '#{@group.name}' was scheduled for deletion."
end
# rubocop: disable CodeReuse/ActiveRecord
def transfer
parent_group = Group.find_by(id: params[:new_parent_group_id])
service = ::Groups::TransferService.new(@group, current_user)
if service.execute(parent_group)
flash[:notice] = "Group '#{@group.name}' was successfully transferred."
redirect_to group_path(@group)
else
flash[:alert] = service.error.html_safe
redirect_to edit_group_path(@group)
end
end
# rubocop: enable CodeReuse/ActiveRecord
def export
export_service = Groups::ImportExport::ExportService.new(group: @group, user: current_user)
if export_service.async_execute
redirect_to edit_group_path(@group), notice: _('Group export started. A download link will be sent by email and made available on this page.')
else
redirect_to edit_group_path(@group), alert: _('Group export could not be started.')
end
end
def download_export
if @group.export_file_exists?
if @group.export_archive_exists?
send_upload(@group.export_file, attachment: @group.export_file.filename)
else
redirect_to edit_group_path(@group),
alert: _('The file containing the export is not available yet; it may still be transferring. Please try again later.')
end
else
redirect_to edit_group_path(@group),
alert: _('Group export link has expired. Please generate a new export from your group settings.')
end
end
def unfoldered_environment_names
respond_to do |format|
format.json do
render json: Environments::EnvironmentNamesFinder.new(@group, current_user).execute
end
end
end
2012-10-02 13:42:15 -04:00
protected
def render_show_html
record_experiment_user(:invite_members_empty_group_version_a) if ::Gitlab.com?
render 'groups/show', locals: { trial: params[:trial] }
end
def render_details_html
render 'groups/show'
end
def render_details_view_atom
load_events
render layout: 'xml.atom', template: 'groups/show'
end
# rubocop: disable CodeReuse/ActiveRecord
2013-01-25 04:30:49 -05:00
def authorize_create_group!
allowed = if params[:parent_id].present?
parent = Group.find_by(id: params[:parent_id])
can?(current_user, :create_subgroup, parent)
else
can?(current_user, :create_group)
end
render_404 unless allowed
end
# rubocop: enable CodeReuse/ActiveRecord
def determine_layout
2013-06-08 09:26:57 -04:00
if [:new, :create].include?(action_name.to_sym)
'application'
elsif [:edit, :update, :projects].include?(action_name.to_sym)
'group_settings'
else
'group'
2013-06-08 09:26:57 -04:00
end
end
def group_params
2018-01-31 10:23:15 -05:00
params.require(:group).permit(group_params_attributes)
end
2018-01-31 10:23:15 -05:00
def group_params_attributes
[
:avatar,
:description,
:emails_disabled,
:mentions_disabled,
:lfs_enabled,
:name,
:path,
:public,
:request_access_enabled,
:share_with_group_lock,
:visibility_level,
2017-02-20 08:51:47 -05:00
:parent_id,
2017-02-07 02:24:57 -05:00
:create_chat_team,
2017-01-24 16:09:58 -05:00
:chat_team_name,
:require_two_factor_authentication,
:two_factor_grace_period,
:project_creation_level,
:subgroup_creation_level,
:default_branch_protection,
:default_branch_name,
:allow_mfa_for_subgroups,
:resource_access_token_creation_allowed,
:prevent_sharing_groups_outside_hierarchy
]
end
2015-02-18 12:38:46 -05:00
# rubocop: disable CodeReuse/ActiveRecord
2015-02-18 12:38:46 -05:00
def load_events
params[:sort] ||= 'latest_activity_desc'
options = { include_subgroups: true }
projects = GroupProjectsFinder.new(params: params, group: group, options: options, current_user: current_user)
.execute
.includes(:namespace)
@events = EventCollection
.new(projects, offset: params[:offset].to_i, filter: event_filter, groups: groups)
.to_a
.map(&:present)
Events::RenderService
.new(current_user)
.execute(@events, atom_request: request.format.atom?)
2015-02-18 12:38:46 -05:00
end
# rubocop: enable CodeReuse/ActiveRecord
def user_actions
if current_user
@notification_setting = current_user.notification_settings_for(group)
end
end
def build_canonical_path(group)
return group_path(group) if action_name == 'show' # root group path
params[:id] = group.to_param
url_for(safe_params)
end
def export_rate_limit
prefixed_action = "group_#{params[:action]}".to_sym
scope = params[:action] == :download_export ? @group : nil
if Gitlab::ApplicationRateLimiter.throttled?(prefixed_action, scope: [current_user, scope].compact)
Gitlab::ApplicationRateLimiter.log_request(request, "#{prefixed_action}_request_limit".to_sym, current_user)
render plain: _('This endpoint has been requested too many times. Try again later.'), status: :too_many_requests
end
end
def ensure_export_enabled
render_404 unless Feature.enabled?(:group_import_export, @group, default_enabled: true)
end
private
def load_recaptcha
Gitlab::Recaptcha.load_configurations!
end
def check_captcha
return if group_params[:parent_id].present? # Only require for top-level groups
load_recaptcha
return if verify_recaptcha
flash[:alert] = _('There was an error with the reCAPTCHA. Please solve the reCAPTCHA again.')
flash.delete :recaptcha_error
@group = Group.new(group_params)
render action: 'new'
end
def successful_creation_hooks; end
def groups
if @group.supports_events?
@group.self_and_descendants.public_or_visible_to_user(current_user)
end
end
override :markdown_service_params
def markdown_service_params
params.merge(group: group)
end
override :has_project_list?
def has_project_list?
%w(details show index).include?(action_name)
end
def captcha_enabled?
Gitlab::Recaptcha.enabled? && Feature.enabled?(:recaptcha_on_top_level_group_creation, type: :ops)
end
def captcha_required?
captcha_enabled? && !params[:parent_id]
end
2012-10-02 13:42:15 -04:00
end
GroupsController.prepend_mod_with('GroupsController')