module ProjectsHelper def link_to_project(project) link_to [project.namespace.becomes(Namespace), project], title: h(project.name) do title = content_tag(:span, project.name, class: 'project-name') if project.namespace namespace = content_tag(:span, "#{project.namespace.human_name} / ", class: 'namespace-name') title = namespace + title end title end end def link_to_member_avatar(author, opts = {}) default_opts = { avatar: true, name: true, size: 16, author_class: 'author', title: ":name" } opts = default_opts.merge(opts) image_tag(avatar_icon(author, opts[:size]), width: opts[:size], class: "avatar avatar-inline #{"s#{opts[:size]}" if opts[:size]}", alt: '') if opts[:avatar] end def link_to_member(project, author, opts = {}, &block) default_opts = { avatar: true, name: true, size: 16, author_class: 'author', title: ":name", tooltip: false } opts = default_opts.merge(opts) return "(deleted)" unless author author_html = "" # Build avatar image tag author_html << image_tag(avatar_icon(author, opts[:size]), width: opts[:size], class: "avatar avatar-inline #{"s#{opts[:size]}" if opts[:size]} #{opts[:avatar_class] if opts[:avatar_class]}", alt: '') if opts[:avatar] # Build name span tag if opts[:by_username] author_html << content_tag(:span, sanitize("@#{author.username}"), class: opts[:author_class]) if opts[:name] else tooltip_data = { placement: 'top' } author_html << content_tag(:span, sanitize(author.name), class: [opts[:author_class], ('has-tooltip' if opts[:tooltip])], title: (author.to_reference if opts[:tooltip]), data: (tooltip_data if opts[:tooltip])) if opts[:name] end author_html << capture(&block) if block author_html = author_html.html_safe if opts[:name] link_to(author_html, user_path(author), class: "author_link #{"#{opts[:extra_class]}" if opts[:extra_class]} #{"#{opts[:mobile_classes]}" if opts[:mobile_classes]}").html_safe else title = opts[:title].sub(":name", sanitize(author.name)) link_to(author_html, user_path(author), class: "author_link has-tooltip", title: title, data: { container: 'body' }).html_safe end end def project_title(project) namespace_link = if project.group group_title(project.group) else owner = project.namespace.owner link_to(simple_sanitize(owner.name), user_path(owner)) end project_link = link_to project_path(project), { class: "project-item-select-holder" } do output = if show_new_nav? project_icon(project, alt: project.name, class: 'avatar-tile', width: 16, height: 16) else "" end output << simple_sanitize(project.name) output.html_safe end if current_user project_link << button_tag(type: 'button', class: 'dropdown-toggle-caret js-projects-dropdown-toggle', aria: { label: 'Toggle switch project dropdown' }, data: { target: '.js-dropdown-menu-projects', toggle: 'dropdown', order_by: 'last_activity_at' }) do icon("chevron-down") end end "#{namespace_link} / #{project_link}".html_safe end def remove_project_message(project) _("You are going to remove %{project_name_with_namespace}. Removed project CANNOT be restored! Are you ABSOLUTELY sure?") % { project_name_with_namespace: project.name_with_namespace } end def transfer_project_message(project) _("You are going to transfer %{project_name_with_namespace} to another owner. Are you ABSOLUTELY sure?") % { project_name_with_namespace: project.name_with_namespace } end def remove_fork_project_message(project) _("You are going to remove the fork relationship to source project %{forked_from_project}. Are you ABSOLUTELY sure?") % { forked_from_project: @project.forked_from_project.name_with_namespace } end def project_nav_tabs @nav_tabs ||= get_project_nav_tabs(@project, current_user) end def project_search_tabs?(tab) abilities = Array(search_tab_ability_map[tab]) abilities.any? { |ability| can?(current_user, ability, @project) } end def project_nav_tab?(name) project_nav_tabs.include? name end def project_for_deploy_key(deploy_key) if deploy_key.has_access_to?(@project) @project else deploy_key.projects.find do |project| can?(current_user, :read_project, project) end end end def can_change_visibility_level?(project, current_user) return false unless can?(current_user, :change_visibility_level, project) if project.forked? project.forked_from_project.visibility_level > Gitlab::VisibilityLevel::PRIVATE else true end end def license_short_name(project) license = project.repository.license license&.nickname || license&.name || 'LICENSE' end def last_push_event return unless current_user return current_user.recent_push unless @project project_ids = [@project.id] if fork = current_user.fork_of(@project) project_ids << fork.id end current_user.recent_push(project_ids) end def project_feature_access_select(field) # Don't show option "everyone with access" if project is private options = project_feature_options level = @project.project_feature.public_send(field) # rubocop:disable GitlabSecurity/PublicSend if @project.private? disabled_option = ProjectFeature::ENABLED highest_available_option = ProjectFeature::PRIVATE if level == disabled_option end options = options_for_select( options.invert, selected: highest_available_option || level, disabled: disabled_option ) content_tag :div, class: "select-wrapper" do concat( content_tag( :select, options, name: "project[project_feature_attributes][#{field}]", id: "project_project_feature_attributes_#{field}", class: "pull-right form-control select-control #{repo_children_classes(field)} ", data: { field: field } ) ) concat( icon('chevron-down') ) end.html_safe end def link_to_autodeploy_doc link_to _('About auto deploy'), help_page_path('ci/autodeploy/index'), target: '_blank' end def autodeploy_flash_notice(branch_name) translation = _("Branch %{branch_name} was created. To set up auto deploy, choose a GitLab CI Yaml template and commit your changes. %{link_to_autodeploy_doc}") % { branch_name: truncate(sanitize(branch_name)), link_to_autodeploy_doc: link_to_autodeploy_doc } translation.html_safe end def project_list_cache_key(project) key = [ project.route.cache_key, project.cache_key, controller.controller_name, controller.action_name, current_application_settings.cache_key, 'v2.5' ] key << pipeline_status_cache_key(project.pipeline_status) if project.pipeline_status.has_status? key end def load_pipeline_status(projects) Gitlab::Cache::Ci::ProjectPipelineStatus .load_in_batch_for_projects(projects) end def show_no_ssh_key_message? cookies[:hide_no_ssh_message].blank? && !current_user.hide_no_ssh_key && current_user.require_ssh_key? end def show_no_password_message? cookies[:hide_no_password_message].blank? && !current_user.hide_no_password && ( current_user.require_password_creation? || current_user.require_personal_access_token_creation_for_git_auth? ) end def link_to_set_password if current_user.require_password_creation? link_to s_('SetPasswordToCloneLink|set a password'), edit_profile_password_path else link_to s_('CreateTokenToCloneLink|create a personal access token'), profile_personal_access_tokens_path end end # Returns true if any projects are present. # # If the relation has a LIMIT applied we'll cast the relation to an Array # since repeated any? checks would otherwise result in multiple COUNT queries # being executed. # # If no limit is applied we'll just issue a COUNT since the result set could # be too large to load into memory. def any_projects?(projects) return projects.any? if projects.is_a?(Array) if projects.limit_value projects.to_a.any? else projects.except(:offset).any? end end def has_projects_or_name?(projects, params) !!(params[:name] || any_projects?(projects)) end private def repo_children_classes(field) needs_repo_check = [:merge_requests_access_level, :builds_access_level] return unless needs_repo_check.include?(field) classes = "project-repo-select js-repo-select" classes << " disabled" unless @project.feature_available?(:repository, current_user) classes end def get_project_nav_tabs(project, current_user) nav_tabs = [:home] if !project.empty_repo? && can?(current_user, :download_code, project) nav_tabs << [:files, :commits, :network, :graphs, :forks] end if project.repo_exists? && can?(current_user, :read_merge_request, project) nav_tabs << :merge_requests end if Gitlab.config.registry.enabled && can?(current_user, :read_container_image, project) nav_tabs << :container_registry end if project.builds_enabled? && can?(current_user, :read_pipeline, project) nav_tabs << :pipelines end tab_ability_map.each do |tab, ability| if can?(current_user, ability, project) nav_tabs << tab end end nav_tabs.flatten end def tab_ability_map { environments: :read_environment, milestones: :read_milestone, snippets: :read_project_snippet, settings: :admin_project, builds: :read_build, labels: :read_label, issues: :read_issue, project_members: :read_project_member, wiki: :read_wiki } end def search_tab_ability_map @search_tab_ability_map ||= tab_ability_map.merge( blobs: :download_code, commits: :download_code, merge_requests: :read_merge_request, notes: [:read_merge_request, :download_code, :read_issue, :read_project_snippet] ) end def project_lfs_status(project) if project.lfs_enabled? content_tag(:span, class: 'lfs-enabled') do s_('LFSStatus|Enabled') end else content_tag(:span, class: 'lfs-disabled') do s_('LFSStatus|Disabled') end end end def git_user_name if current_user current_user.name else _("Your name") end end def git_user_email if current_user current_user.email else "your@email.com" end end def default_url_to_repo(project = @project) case default_clone_protocol when 'ssh' project.ssh_url_to_repo else project.http_url_to_repo end end def default_clone_protocol if allowed_protocols_present? enabled_protocol else if !current_user || current_user.require_ssh_key? gitlab_config.protocol else 'ssh' end end end def project_last_activity(project) if project.last_activity_at time_ago_with_tooltip(project.last_activity_at, placement: 'bottom', html_class: 'last_activity_time_ago') else s_("ProjectLastActivity|Never") end end def add_special_file_path(project, file_name:, commit_message: nil, branch_name: nil, context: nil) commit_message ||= s_("CommitMessage|Add %{file_name}") % { file_name: file_name.downcase } project_new_blob_path( project, project.default_branch || 'master', file_name: file_name, commit_message: commit_message, branch_name: branch_name, context: context ) end def add_koding_stack_path(project) project_new_blob_path( project, project.default_branch || 'master', file_name: '.koding.yml', commit_message: "Add Koding stack script", content: <<-CONTENT.strip_heredoc provider: aws: access_key: '${var.aws_access_key}' secret_key: '${var.aws_secret_key}' resource: aws_instance: #{project.path}-vm: instance_type: t2.nano user_data: |- # Created by GitLab UI for :> echo _KD_NOTIFY_@Installing Base packages...@ apt-get update -y apt-get install git -y echo _KD_NOTIFY_@Cloning #{project.name}...@ export KODING_USER=${var.koding_user_username} export REPO_URL=#{root_url}${var.koding_queryString_repo}.git export BRANCH=${var.koding_queryString_branch} sudo -i -u $KODING_USER git clone $REPO_URL -b $BRANCH echo _KD_NOTIFY_@#{project.name} cloned.@ CONTENT ) end def koding_project_url(project = nil, branch = nil, sha = nil) if project import_path = "/Home/Stacks/import" repo = project.full_path branch ||= project.default_branch sha ||= project.commit.short_id path = "#{import_path}?repo=#{repo}&branch=#{branch}&sha=#{sha}" return URI.join(current_application_settings.koding_url, path).to_s end current_application_settings.koding_url end def contribution_guide_path(project) if project && contribution_guide = project.repository.contribution_guide project_blob_path( project, tree_join(project.default_branch, contribution_guide.name) ) end end def readme_path(project) filename_path(project, :readme) end def changelog_path(project) filename_path(project, :changelog) end def license_path(project) filename_path(project, :license_blob) end def version_path(project) filename_path(project, :version) end def ci_configuration_path(project) filename_path(project, :gitlab_ci_yml) end def project_wiki_path_with_version(proj, page, version, is_newest) url_params = is_newest ? {} : { version_id: version } project_wiki_path(proj, page, url_params) end def project_status_css_class(status) case status when "started" "active" when "failed" "danger" when "finished" "success" end end def readme_cache_key sha = @project.commit.try(:sha) || 'nil' [@project.full_path, sha, "readme"].join('-') end def current_ref @ref || @repository.try(:root_ref) end def filename_path(project, filename) if project && blob = project.repository.public_send(filename) # rubocop:disable GitlabSecurity/PublicSend project_blob_path( project, tree_join(project.default_branch, blob.name) ) end end def sanitize_repo_path(project, message) return '' unless message.present? exports_path = File.join(Settings.shared['path'], 'tmp/project_exports') filtered_message = message.strip.gsub(exports_path, "[REPO EXPORT PATH]") filtered_message.gsub(project.repository_storage_path.chomp('/'), "[REPOS PATH]") end def project_feature_options { ProjectFeature::DISABLED => s_('ProjectFeature|Disabled'), ProjectFeature::PRIVATE => s_('ProjectFeature|Only team members'), ProjectFeature::ENABLED => s_('ProjectFeature|Everyone with access') } end def project_child_container_class(view_path) view_path == "projects/issues/issues" ? "prepend-top-default" : "project-show-#{view_path}" end def project_issues(project) IssuesFinder.new(current_user, project_id: project.id).execute end def visibility_select_options(project, selected_level) level_options = Gitlab::VisibilityLevel.values.each_with_object([]) do |level, level_options| next if restricted_levels.include?(level) level_options << [ visibility_level_label(level), { data: { description: visibility_level_description(level, project) } }, level ] end options_for_select(level_options, selected_level) end def restricted_levels return [] if current_user.admin? current_application_settings.restricted_visibility_levels || [] end def find_file_path return unless @project && !@project.empty_repo? ref = @ref || @project.repository.root_ref project_find_file_path(@project, ref) end end