# frozen_string_literal: true class ProjectsController < Projects::ApplicationController include API::Helpers::RelatedResourcesHelpers include IssuableCollections include ExtractsPath include PreviewMarkdown include SendFileUpload include RecordUserLastActivity include ImportUrlParams include FiltersEvents include SourcegraphDecorator include PlanningHierarchy REFS_LIMIT = 100 prepend_before_action(only: [:show]) { authenticate_sessionless_user!(:rss) } around_action :allow_gitaly_ref_name_caching, only: [:index, :show] before_action :disable_query_limiting, only: [:show, :create] before_action :authenticate_user!, except: [:index, :show, :activity, :refs, :unfoldered_environment_names] before_action :redirect_git_extension, only: [:show] before_action :project, except: [:index, :new, :create] before_action :repository, except: [:index, :new, :create] before_action :verify_git_import_enabled, only: [:create] before_action :project_export_enabled, only: [:export, :download_export, :remove_export, :generate_new_export] before_action :present_project, only: [:edit] before_action :authorize_download_code!, only: [:refs] # Authorize before_action :authorize_admin_project!, only: [:edit, :update, :housekeeping, :download_export, :export, :remove_export, :generate_new_export] before_action :authorize_archive_project!, only: [:archive, :unarchive] before_action :event_filter, only: [:show, :activity] # Project Export Rate Limit before_action :check_export_rate_limit!, only: [:export, :download_export, :generate_new_export] before_action do push_frontend_feature_flag(:lazy_load_commits, @project) push_frontend_feature_flag(:highlight_js, @project) push_frontend_feature_flag(:file_line_blame, @project) push_frontend_feature_flag(:increase_page_size_exponentially, @project) push_licensed_feature(:file_locks) if @project.present? && @project.licensed_feature_available?(:file_locks) push_licensed_feature(:security_orchestration_policies) if @project.present? && @project.licensed_feature_available?(:security_orchestration_policies) push_force_frontend_feature_flag(:work_items, @project&.work_items_feature_flag_enabled?) push_force_frontend_feature_flag(:work_items_mvc_2, @project&.work_items_mvc_2_feature_flag_enabled?) push_frontend_feature_flag(:package_registry_access_level) push_frontend_feature_flag(:work_items_hierarchy, @project) end before_action only: :edit do push_frontend_feature_flag(:split_operations_visibility_permissions, @project) end layout :determine_layout feature_category :projects, [ :index, :show, :new, :create, :edit, :update, :transfer, :destroy, :archive, :unarchive, :toggle_star, :activity ] feature_category :source_code_management, [:remove_fork, :housekeeping, :refs] feature_category :team_planning, [:preview_markdown, :new_issuable_address] feature_category :importers, [:export, :remove_export, :generate_new_export, :download_export] feature_category :code_review, [:unfoldered_environment_names] feature_category :portfolio_management, [:planning_hierarchy] urgency :low, [:export, :remove_export, :generate_new_export, :download_export] urgency :low, [:preview_markdown, :new_issuable_address] # TODO: Set high urgency for #show https://gitlab.com/gitlab-org/gitlab/-/issues/334444 urgency :low, [:refs, :show, :toggle_star, :transfer, :archive, :destroy, :update, :create, :activity, :edit, :new, :export, :remove_export, :generate_new_export, :download_export] urgency :high, [:unfoldered_environment_names] def index redirect_to(current_user ? root_path : explore_root_path) end # rubocop: disable CodeReuse/ActiveRecord def new @namespace = Namespace.find_by(id: params[:namespace_id]) if params[:namespace_id] return access_denied! if @namespace && !can?(current_user, :create_projects, @namespace) @current_user_group = if current_user.manageable_groups(include_groups_with_developer_maintainer_access: true).count == 1 current_user.manageable_groups(include_groups_with_developer_maintainer_access: true).first else nil end @project = Project.new(namespace_id: @namespace&.id) end # rubocop: enable CodeReuse/ActiveRecord def edit @badge_api_endpoint = expose_path(api_v4_projects_badges_path(id: @project.id)) render_edit end def create @project = ::Projects::CreateService.new(current_user, project_params(attributes: project_params_create_attributes)).execute if @project.saved? redirect_to( project_path(@project, custom_import_params), notice: _("Project '%{project_name}' was successfully created.") % { project_name: @project.name } ) else render 'new' end end def update result = ::Projects::UpdateService.new(@project, current_user, project_params).execute # Refresh the repo in case anything changed @repository = @project.repository if result[:status] == :success flash[:notice] = _("Project '%{project_name}' was successfully updated.") % { project_name: @project.name } redirect_to(edit_project_path(@project, anchor: 'js-general-project-settings')) else flash[:alert] = result[:message] @project.reset render 'edit' end end # rubocop: disable CodeReuse/ActiveRecord def transfer return access_denied! unless can?(current_user, :change_namespace, @project) namespace = Namespace.find_by(id: params[:new_namespace_id]) ::Projects::TransferService.new(project, current_user).execute(namespace) if @project.errors[:new_namespace].present? flash[:alert] = @project.errors[:new_namespace].first return redirect_to edit_project_path(@project) end redirect_to edit_project_path(@project) end # rubocop: enable CodeReuse/ActiveRecord def remove_fork return access_denied! unless can?(current_user, :remove_fork_project, @project) if ::Projects::UnlinkForkService.new(@project, current_user).execute flash[:notice] = _('The fork relationship has been removed.') end redirect_to edit_project_path(@project) 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 show @id, @ref, @path = extract_ref_path if @project.import_in_progress? redirect_to project_import_path(@project, custom_import_params) return end if @project.pending_delete? flash.now[:alert] = _("Project '%{project_name}' queued for deletion.") % { project_name: @project.name } end respond_to do |format| format.html do @notification_setting = current_user.notification_settings_for(@project) if current_user @project = @project.present(current_user: current_user) render_landing_page end format.atom do load_events @events = @events.select { |event| event.visible_to_user?(current_user) } render layout: 'xml' end end end def destroy return access_denied! unless can?(current_user, :remove_project, @project) ::Projects::DestroyService.new(@project, current_user, {}).async_execute flash[:notice] = _("Project '%{project_name}' is in the process of being deleted.") % { project_name: @project.full_name } redirect_to dashboard_projects_path, status: :found rescue Projects::DestroyService::DestroyError => e redirect_to edit_project_path(@project), status: :found, alert: e.message end def new_issuable_address return render_404 unless Gitlab::IncomingEmail.supports_issue_creation? current_user.reset_incoming_email_token! render json: { new_address: @project.new_issuable_address(current_user, params[:issuable_type]) } end def archive ::Projects::UpdateService.new(@project, current_user, archived: true).execute respond_to do |format| format.html { redirect_to project_path(@project) } end end def unarchive ::Projects::UpdateService.new(@project, current_user, archived: false).execute respond_to do |format| format.html { redirect_to project_path(@project) } end end def housekeeping ::Repositories::HousekeepingService.new(@project, :gc).execute redirect_to( project_path(@project), notice: _("Housekeeping successfully started") ) rescue ::Repositories::HousekeepingService::LeaseTaken => e redirect_to( edit_project_path(@project, anchor: 'js-project-advanced-settings'), alert: e.to_s ) end def export @project.add_export_job(current_user: current_user) redirect_to( edit_project_path(@project, anchor: 'js-export-project'), notice: _("Project export started. A download link will be sent by email and made available on this page.") ) rescue Project::ExportLimitExceeded => e redirect_to( edit_project_path(@project, anchor: 'js-export-project'), alert: e.to_s ) end def download_export if @project.export_file_exists? if @project.export_archive_exists? send_upload(@project.export_file, attachment: @project.export_file.filename) else redirect_to( edit_project_path(@project, anchor: 'js-export-project'), alert: _("The file containing the export is not available yet; it may still be transferring. Please try again later.") ) end else redirect_to( edit_project_path(@project, anchor: 'js-export-project'), alert: _("Project export link has expired. Please generate a new export from your project settings.") ) end end def remove_export if @project.remove_exports flash[:notice] = _("Project export has been deleted.") else flash[:alert] = _("Project export could not be deleted.") end redirect_to(edit_project_path(@project, anchor: 'js-export-project')) end def generate_new_export if @project.remove_exports export else redirect_to( edit_project_path(@project, anchor: 'js-export-project'), alert: _("Project export could not be deleted.") ) end end def toggle_star current_user.toggle_star(@project) @project.reset render json: { star_count: @project.star_count } end # rubocop: disable CodeReuse/ActiveRecord def refs find_refs = refs_params['find'] find_branches = true find_tags = true find_commits = true unless find_refs.nil? find_branches = find_refs.include?('branches') find_tags = find_refs.include?('tags') find_commits = find_refs.include?('commits') end options = {} if find_branches branches = BranchesFinder.new(@repository, refs_params.merge(per_page: REFS_LIMIT)) .execute(gitaly_pagination: true) .take(REFS_LIMIT) .map(&:name) options['Branches'] = branches end if find_tags && @repository.tag_count.nonzero? tags = TagsFinder.new(@repository, refs_params.merge(per_page: REFS_LIMIT)) .execute(gitaly_pagination: true) .take(REFS_LIMIT) .map(&:name) options['Tags'] = tags end # If reference is commit id - we should add it to branch/tag selectbox ref = Addressable::URI.unescape(refs_params[:ref]) if find_commits && ref && options.flatten(2).exclude?(ref) && ref =~ /\A[0-9a-zA-Z]{6,52}\z/ options['Commits'] = [ref] end render json: options.to_json rescue Gitlab::Git::CommandError render json: { error: _('Unable to load refs') }, status: :service_unavailable end # rubocop: enable CodeReuse/ActiveRecord def unfoldered_environment_names respond_to do |format| format.json do render json: Environments::EnvironmentNamesFinder.new(@project, current_user).execute end end end private def refs_params params.permit(:search, :sort, :ref, find: []) end # Render project landing depending of which features are available # So if page is not available in the list it renders the next page # # pages list order: repository readme, wiki home, issues list, customize workflow def render_landing_page Gitlab::Tracking.event('project_overview', 'render', user: current_user, project: @project.project) if can?(current_user, :download_code, @project) return render 'projects/no_repo' unless @project.repository_exists? render 'projects/empty' if @project.empty_repo? else if can?(current_user, :read_wiki, @project) @wiki = @project.wiki @wiki_home = @wiki.find_page('home', params[:version_id]) elsif @project.feature_available?(:issues, current_user) @issues = issuables_collection.page(params[:page]) @issuable_meta_data = Gitlab::IssuableMetadata.new(current_user, @issues).data end render :show end end def finder_type IssuesFinder end def determine_layout if [:new, :create].include?(action_name.to_sym) 'application' elsif [:edit, :update].include?(action_name.to_sym) 'project_settings' else 'project' end end # rubocop: disable CodeReuse/ActiveRecord def load_events projects = Project.where(id: @project.id) @events = EventCollection .new(projects, offset: params[:offset].to_i, filter: event_filter) .to_a .map(&:present) end # rubocop: enable CodeReuse/ActiveRecord def project_params(attributes: []) params.require(:project) .permit(project_params_attributes + attributes) .merge(import_url_params) end def project_feature_attributes %i[ builds_access_level issues_access_level forking_access_level merge_requests_access_level repository_access_level snippets_access_level wiki_access_level package_registry_access_level pages_access_level metrics_dashboard_access_level analytics_access_level security_and_compliance_access_level container_registry_access_level releases_access_level ] + operations_feature_attributes end def operations_feature_attributes if Feature.enabled?(:split_operations_visibility_permissions, project) %i[ environments_access_level feature_flags_access_level monitor_access_level ] else %i[operations_access_level] end end def project_setting_attributes %i[ show_default_award_emojis squash_option mr_default_target_self warn_about_potentially_unwanted_characters enforce_auth_checks_on_uploads ] end def project_params_attributes [ :allow_merge_on_skipped_pipeline, :avatar, :build_allow_git_fetch, :build_timeout_human_readable, :resolve_outdated_diff_discussions, :container_registry_enabled, :default_branch, :description, :emails_disabled, :external_authorization_classification_label, :import_url, :issues_tracker, :issues_tracker_id, :last_activity_at, :lfs_enabled, :name, :only_allow_merge_if_all_discussions_are_resolved, :only_allow_merge_if_pipeline_succeeds, :path, :printing_merge_request_link_enabled, :public_builds, :remove_source_branch_after_merge, :request_access_enabled, :runners_token, :tag_list, :topics, :visibility_level, :template_name, :template_project_id, :merge_method, :initialize_with_sast, :initialize_with_readme, :autoclose_referenced_issues, :ci_separated_caches, :suggestion_commit_message, :packages_enabled, :service_desk_enabled, :merge_commit_template_or_default, :squash_commit_template_or_default, project_setting_attributes: project_setting_attributes ] + [project_feature_attributes: project_feature_attributes] end def project_params_create_attributes [:namespace_id] end def custom_import_params {} end def active_new_project_tab project_params[:import_url].present? ? 'import' : 'blank' end def repo_exists? project.repository_exists? && !project.empty_repo? rescue Gitlab::Git::Repository::NoRepository project.repository.expire_exists_cache false end def project_view_files? if current_user current_user.project_view == 'files' else project_view_files_allowed? end end # Override extract_ref from ExtractsPath, which returns the branch and file path # for the blob/tree, which in this case is just the root of the default branch. # This way we avoid to access the repository.ref_names. def extract_ref(_id) [get_id, ''] end # Override get_id from ExtractsPath in this case is just the root of the default branch. def get_id project.repository.root_ref end def project_view_files_allowed? !project.empty_repo? && can?(current_user, :download_code, project) end def build_canonical_path(project) params[:namespace_id] = project.namespace.to_param params[:id] = project.to_param url_for(safe_params) end def verify_git_import_enabled render_404 if project_params[:import_url] && !git_import_enabled? end def project_export_enabled render_404 unless Gitlab::CurrentSettings.project_export_enabled? end # Redirect from localhost/group/project.git to localhost/group/project def redirect_git_extension return unless params[:format] == 'git' # `project` calls `find_routable!`, so this will trigger the usual not-found # behaviour when the user isn't authorized to see the project return if project.nil? || performed? redirect_to(request.original_url.sub(%r{\.git/?\Z}, '')) end def disable_query_limiting Gitlab::QueryLimiting.disable!('https://gitlab.com/gitlab-org/gitlab/-/issues/20826') end def present_project @project = @project.present(current_user: current_user) end def check_export_rate_limit! prefixed_action = "project_#{params[:action]}".to_sym group_scope = params[:action] == 'download_export' ? @project.namespace : nil check_rate_limit!(prefixed_action, scope: [current_user, group_scope].compact) end def render_edit render 'edit' end end ProjectsController.prepend_mod_with('ProjectsController')