projects_helper.rb 17.2 KB
Newer Older
1 2
# frozen_string_literal: true

R
randx 已提交
3
module ProjectsHelper
4
  def link_to_project(project)
5
    link_to namespace_project_path(namespace_id: project.namespace, id: project), title: h(project.name) do
6
      title = content_tag(:span, project.name, class: 'project-name')
7 8

      if project.namespace
D
Dmitriy Zaporozhets 已提交
9
        namespace = content_tag(:span, "#{project.namespace.human_name} / ", class: 'namespace-name')
10 11 12 13 14
        title = namespace + title
      end

      title
    end
15
  end
R
randx 已提交
16

J
Jacob Schatz 已提交
17
  def link_to_member_avatar(author, opts = {})
18
    default_opts = { size: 16, lazy_load: false }
J
Jacob Schatz 已提交
19
    opts = default_opts.merge(opts)
20 21

    classes = %W[avatar avatar-inline s#{opts[:size]}]
M
Maxim Rydkin 已提交
22 23
    classes << opts[:avatar_class] if opts[:avatar_class]

24
    avatar = avatar_icon_for_user(author, opts[:size])
25 26 27
    src = opts[:lazy_load] ? nil : avatar

    image_tag(src, width: opts[:size], class: classes, alt: '', "data-src" => avatar)
J
Jacob Schatz 已提交
28 29
  end

30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
  def author_content_tag(author, opts = {})
    default_opts = { author_class: 'author', tooltip: false, by_username: false }
    opts = default_opts.merge(opts)

    has_tooltip = !opts[:by_username] && opts[:tooltip]

    username = opts[:by_username] ? author.to_reference : author.name
    name_tag_options = { class: [opts[:author_class]] }

    if has_tooltip
      name_tag_options[:title] = author.to_reference
      name_tag_options[:data] = { placement: 'top' }
      name_tag_options[:class] << 'has-tooltip'
    end

45 46
    # NOTE: ActionView::Helpers::TagHelper#content_tag HTML escapes username
    content_tag(:span, username, name_tag_options)
47 48
  end

P
Phil Hughes 已提交
49
  def link_to_member(project, author, opts = {}, &block)
50
    default_opts = { avatar: true, name: true, title: ":name" }
51 52
    opts = default_opts.merge(opts)

53 54 55 56 57 58
    data_attrs = {
      user_id: author.id,
      username: author.username,
      name: author.name
    }

59 60
    return "(deleted)" unless author

61
    author_html = []
62

63
    # Build avatar image tag
64
    author_html << link_to_member_avatar(author, opts) if opts[:avatar]
65

66
    # Build name span tag
67
    author_html << author_content_tag(author, opts) if opts[:name]
68

P
Phil Hughes 已提交
69 70
    author_html << capture(&block) if block

71
    author_html = author_html.join.html_safe
72

73
    if opts[:name]
74
      link_to(author_html, user_path(author), class: "author-link js-user-link #{"#{opts[:extra_class]}" if opts[:extra_class]} #{"#{opts[:mobile_classes]}" if opts[:mobile_classes]}", data: data_attrs).html_safe
75
    else
76
      title = opts[:title].sub(":name", sanitize(author.name))
77
      link_to(author_html, user_path(author), class: "author-link has-tooltip", title: title, data: { container: 'body' }).html_safe
78
    end
R
randx 已提交
79
  end
80

81
  def project_title(project)
82 83
    namespace_link =
      if project.group
P
Phil Hughes 已提交
84
        group_title(project.group, nil, nil)
85 86 87
      else
        owner = project.namespace.owner
        link_to(simple_sanitize(owner.name), user_path(owner))
88
      end
89

P
Phil Hughes 已提交
90
    project_link = link_to project_path(project) do
91 92
      icon = project_icon(project, alt: project.name, class: 'avatar-tile', width: 15, height: 15) if project.avatar_url && !Rails.env.test?
      [icon, content_tag("span", simple_sanitize(project.name), class: "breadcrumb-item-text js-breadcrumb-item-text")].join.html_safe
P
Phil Hughes 已提交
93
    end
94

P
Phil Hughes 已提交
95 96
    namespace_link = breadcrumb_list_item(namespace_link) unless project.group
    project_link = breadcrumb_list_item project_link
P
Phil Hughes 已提交
97

P
Phil Hughes 已提交
98
    "#{namespace_link} #{project_link}".html_safe
99
  end
100 101

  def remove_project_message(project)
102 103
    _("You are going to remove %{project_full_name}. Removed project CANNOT be restored! Are you ABSOLUTELY sure?") %
      { project_full_name: project.full_name }
104
  end
105

106
  def transfer_project_message(project)
107 108
    _("You are going to transfer %{project_full_name} to another owner. Are you ABSOLUTELY sure?") %
      { project_full_name: project.full_name }
109 110
  end

111
  def remove_fork_project_message(project)
112
    _("You are going to remove the fork relationship to source project %{forked_from_project}. Are you ABSOLUTELY sure?") %
113 114 115 116 117 118 119 120 121
      { forked_from_project: fork_source_name(project) }
  end

  def fork_source_name(project)
    if @project.fork_source
      @project.fork_source.full_name
    else
      @project.fork_network&.deleted_root_project_name
    end
122 123
  end

124 125 126 127
  def project_nav_tabs
    @nav_tabs ||= get_project_nav_tabs(@project, current_user)
  end

128 129 130 131 132 133
  def project_search_tabs?(tab)
    abilities = Array(search_tab_ability_map[tab])

    abilities.any? { |ability| can?(current_user, ability, @project) }
  end

134 135 136 137
  def project_nav_tab?(name)
    project_nav_tabs.include? name
  end

D
Douwe Maan 已提交
138
  def project_for_deploy_key(deploy_key)
139
    if deploy_key.has_access_to?(@project)
D
Douwe Maan 已提交
140 141
      @project
    else
L
Lin Jen-Shin 已提交
142 143 144
      deploy_key.projects.find do |project|
        can?(current_user, :read_project, project)
      end
D
Douwe Maan 已提交
145 146 147
    end
  end

V
Valery Sizov 已提交
148 149 150
  def can_change_visibility_level?(project, current_user)
    return false unless can?(current_user, :change_visibility_level, project)

151 152
    if project.fork_source
      project.fork_source.visibility_level > Gitlab::VisibilityLevel::PRIVATE
V
Valery Sizov 已提交
153 154 155 156 157
    else
      true
    end
  end

158
  def last_push_event
159
    current_user&.recent_push(@project)
160 161
  end

162
  def link_to_autodeploy_doc
163
    link_to _('About auto deploy'), help_page_path('ci/autodeploy/index'), target: '_blank'
164 165
  end

V
Valery Sizov 已提交
166
  def autodeploy_flash_notice(branch_name)
167 168 169
    translation = _("Branch <strong>%{branch_name}</strong> 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
V
Valery Sizov 已提交
170 171
  end

172
  def project_list_cache_key(project)
173
    key = [
174
      project.route.cache_key,
175
      project.cache_key,
176
      project.last_activity_date,
177 178
      controller.controller_name,
      controller.action_name,
179
      Gitlab::CurrentSettings.cache_key,
180
      "cross-project:#{can?(current_user, :read_cross_project)}",
181
      max_project_member_access_cache_key(project),
M
Mark Chao 已提交
182
      'v2.6'
183 184
    ]

185 186 187 188 189
    key << pipeline_status_cache_key(project.pipeline_status) if project.pipeline_status.has_status?

    key
  end

190
  def load_pipeline_status(projects)
191 192
    Gitlab::Cache::Ci::ProjectPipelineStatus
      .load_in_batch_for_projects(projects)
193 194
  end

195
  def show_no_ssh_key_message?
196 197 198 199
    Gitlab::CurrentSettings.user_show_add_ssh_key_message? &&
      cookies[:hide_no_ssh_message].blank? &&
      !current_user.hide_no_ssh_key &&
      current_user.require_ssh_key?
200 201 202 203
  end

  def show_no_password_message?
    cookies[:hide_no_password_message].blank? && !current_user.hide_no_password &&
204
      current_user.require_extra_setup_for_git_auth?
205 206
  end

207 208
  def show_auto_devops_implicitly_enabled_banner?(project, user)
    return false unless user_can_see_auto_devops_implicitly_enabled_banner?(project, user)
209

210
    cookies["hide_auto_devops_implicitly_enabled_banner_#{project.id}".to_sym].blank?
211 212
  end

213
  def link_to_set_password
214
    if current_user.require_password_creation_for_git?
215 216 217 218 219 220
      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

221 222 223 224 225 226 227 228
  # 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.
229
  # rubocop: disable CodeReuse/ActiveRecord
230
  def any_projects?(projects)
231 232
    return projects.any? if projects.is_a?(Array)

233 234 235 236 237 238
    if projects.limit_value
      projects.to_a.any?
    else
      projects.except(:offset).any?
    end
  end
239
  # rubocop: enable CodeReuse/ActiveRecord
240

241 242
  def show_projects?(projects, params)
    !!(params[:personal] || params[:name] || any_projects?(projects))
243 244
  end

245 246 247 248 249 250 251 252 253 254 255
  def push_to_create_project_command(user = current_user)
    repository_url =
      if Gitlab::CurrentSettings.current_application_settings.enabled_git_access_protocol == 'http'
        user_url(user)
      else
        Gitlab.config.gitlab_shell.ssh_path_prefix + user.username
      end

    "git push --set-upstream #{repository_url}/$(git rev-parse --show-toplevel | xargs basename).git $(git rev-parse --abbrev-ref HEAD)"
  end

A
André Luís 已提交
256 257 258 259 260 261 262 263
  def show_xcode_link?(project = @project)
    browser.platform.mac? && project.repository.xcode_project?
  end

  def xcode_uri_to_repo(project = @project)
    "xcode://clone?repo=#{CGI.escape(default_url_to_repo(project))}"
  end

264 265 266 267
  def link_to_bfg
    link_to 'BFG', 'https://rtyley.github.io/bfg-repo-cleaner/', target: '_blank', rel: 'noopener noreferrer'
  end

268 269 270 271
  def legacy_render_context(params)
    params[:legacy_render] ? { markdown_engine: :redcarpet } : {}
  end

272 273 274 275 276 277 278 279 280 281 282 283 284 285
  def explore_projects_tab?
    current_page?(explore_projects_path) ||
      current_page?(trending_explore_projects_path) ||
      current_page?(starred_explore_projects_path)
  end

  def show_merge_request_count?(disabled: false, compact_mode: false)
    !disabled && !compact_mode && Feature.enabled?(:project_list_show_mr_count, default_enabled: true)
  end

  def show_issue_count?(disabled: false, compact_mode: false)
    !disabled && !compact_mode && Feature.enabled?(:project_list_show_issue_count, default_enabled: true)
  end

286 287
  # overridden in EE
  def settings_operations_available?
288
    can?(current_user, :read_environment, @project)
289 290
  end

291 292 293
  private

  def get_project_nav_tabs(project, current_user)
294
    nav_tabs = [:home]
295

296
    if !project.empty_repo? && can?(current_user, :download_code, project)
F
Filipa Lacerda 已提交
297
      nav_tabs << [:files, :commits, :network, :graphs, :forks, :releases]
298 299
    end

300
    if project.repo_exists? && can?(current_user, :read_merge_request, project)
301 302 303
      nav_tabs << :merge_requests
    end

304
    if Gitlab.config.registry.enabled && can?(current_user, :read_container_image, project)
305
      nav_tabs << :container_registry
K
Kamil Trzcinski 已提交
306 307
    end

308 309
    # Pipelines feature is tied to presence of builds
    if can?(current_user, :read_build, project)
310
      nav_tabs << :pipelines
311 312 313
    end

    if can?(current_user, :read_environment, project) || can?(current_user, :read_cluster, project)
314
      nav_tabs << :operations
315 316
    end

317 318 319 320 321 322
    tab_ability_map.each do |tab, ability|
      if can?(current_user, ability, project)
        nav_tabs << tab
      end
    end

323 324
    nav_tabs << external_nav_tabs(project)

325 326 327
    nav_tabs.flatten
  end

328 329 330
  def external_nav_tabs(project)
    [].tap do |tabs|
      tabs << :external_issue_tracker if project.external_issue_tracker
331
      tabs << :external_wiki if project.external_wiki
332 333 334
    end
  end

335 336
  def tab_ability_map
    {
337 338 339 340 341
      environments:     :read_environment,
      milestones:       :read_milestone,
      snippets:         :read_project_snippet,
      settings:         :admin_project,
      builds:           :read_build,
342
      clusters:         :read_cluster,
D
Dylan Griffith 已提交
343
      serverless:       :read_cluster,
344
      error_tracking:   :read_sentry_issue,
345 346 347 348
      labels:           :read_label,
      issues:           :read_issue,
      project_members:  :read_project_member,
      wiki:             :read_wiki
349
    }
350
  end
351

352 353 354 355 356 357 358
  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]
    )
359
  end
360

361 362
  def project_lfs_status(project)
    if project.lfs_enabled?
363
      content_tag(:span, class: 'lfs-enabled') do
364
        s_('LFSStatus|Enabled')
365 366
      end
    else
367
      content_tag(:span, class: 'lfs-disabled') do
368
        s_('LFSStatus|Disabled')
369 370 371 372
      end
    end
  end

373 374
  def git_user_name
    if current_user
375
      current_user.name.gsub('"', '\"')
376
    else
377
      _("Your name")
378 379 380 381 382 383 384 385 386 387
    end
  end

  def git_user_email
    if current_user
      current_user.email
    else
      "your@email.com"
    end
  end
388

389
  def default_url_to_repo(project = @project)
390 391
    case default_clone_protocol
    when 'ssh'
392 393
      project.ssh_url_to_repo
    else
394
      project.http_url_to_repo
395
    end
396
  end
397

398 399 400 401
  def default_clone_label
    _("Copy %{protocol} clone URL") % { protocol: default_clone_protocol.upcase }
  end

402
  def default_clone_protocol
403 404
    if allowed_protocols_present?
      enabled_protocol
405
    else
L
Lukas Eipert 已提交
406 407 408 409 410 411 412 413 414
      extra_default_clone_protocol
    end
  end

  def extra_default_clone_protocol
    if !current_user || current_user.require_ssh_key?
      gitlab_config.protocol
    else
      'ssh'
415
    end
416
  end
417

M
Matija Čupić 已提交
418 419 420 421
  def sidebar_operations_link_path(project = @project)
    metrics_project_environments_path(project) if can?(current_user, :read_environment, project)
  end

422 423
  def project_last_activity(project)
    if project.last_activity_at
424
      time_ago_with_tooltip(project.last_activity_at, placement: 'bottom', html_class: 'last_activity_time_ago')
425
    else
426
      s_("ProjectLastActivity|Never")
427 428
    end
  end
D
Dmitriy Zaporozhets 已提交
429

430 431
  def project_wiki_path_with_version(proj, page, version, is_newest)
    url_params = is_newest ? {} : { version_id: version }
432
    project_wiki_path(proj, page, url_params)
433
  end
434

V
Valery Sizov 已提交
435 436 437
  def project_status_css_class(status)
    case status
    when "started"
S
Stan Hu 已提交
438
      "table-active"
V
Valery Sizov 已提交
439
    when "failed"
S
Stan Hu 已提交
440
      "table-danger"
V
Valery Sizov 已提交
441
    when "finished"
S
Stan Hu 已提交
442
      "table-success"
V
Valery Sizov 已提交
443 444
    end
  end
445

446
  def readme_cache_key
447
    sha = @project.commit.try(:sha) || 'nil'
448
    [@project.full_path, sha, "readme"].join('-')
449
  end
450

451 452 453
  def current_ref
    @ref || @repository.try(:root_ref)
  end
454

455 456 457
  def project_child_container_class(view_path)
    view_path == "projects/issues/issues" ? "prepend-top-default" : "project-show-#{view_path}"
  end
458 459 460 461

  def project_issues(project)
    IssuesFinder.new(current_user, project_id: project.id).execute
  end
L
Luke "Jared" Bennett 已提交
462

463 464 465
  def restricted_levels
    return [] if current_user.admin?

466
    Gitlab::CurrentSettings.restricted_visibility_levels || []
L
Luke "Jared" Bennett 已提交
467
  end
468

469 470 471 472 473 474 475 476 477 478 479
  def project_permissions_settings(project)
    feature = project.project_feature
    {
      visibilityLevel: project.visibility_level,
      requestAccessEnabled: !!project.request_access_enabled,
      issuesAccessLevel: feature.issues_access_level,
      repositoryAccessLevel: feature.repository_access_level,
      mergeRequestsAccessLevel: feature.merge_requests_access_level,
      buildsAccessLevel: feature.builds_access_level,
      wikiAccessLevel: feature.wiki_access_level,
      snippetsAccessLevel: feature.snippets_access_level,
480
      pagesAccessLevel: feature.pages_access_level,
481 482 483 484 485 486
      containerRegistryEnabled: !!project.container_registry_enabled,
      lfsEnabled: !!project.lfs_enabled
    }
  end

  def project_permissions_panel_data(project)
487
    {
488 489 490 491 492 493
      currentSettings: project_permissions_settings(project),
      canChangeVisibilityLevel: can_change_visibility_level?(project, current_user),
      allowedVisibilityOptions: project_allowed_visibility_levels(project),
      visibilityHelpPath: help_page_path('public_access/public_access'),
      registryAvailable: Gitlab.config.registry.enabled,
      registryHelpPath: help_page_path('user/project/container_registry'),
494
      lfsAvailable: Gitlab.config.lfs.enabled,
495 496 497
      lfsHelpPath: help_page_path('workflow/lfs/manage_large_binaries_with_git_lfs'),
      pagesAvailable: Gitlab.config.pages.enabled,
      pagesAccessControlEnabled: Gitlab.config.pages.access_control,
498
      pagesHelpPath: help_page_path('user/project/pages/introduction', anchor: 'gitlab-pages-access-control-core-only')
499
    }
500
  end
501

502 503
  def project_permissions_panel_data_json(project)
    project_permissions_panel_data(project).to_json.html_safe
504 505 506 507 508 509 510 511
  end

  def project_allowed_visibility_levels(project)
    Gitlab::VisibilityLevel.values.select do |level|
      project.visibility_level_allowed?(level) && !restricted_levels.include?(level)
    end
  end

512 513 514 515 516 517 518
  def find_file_path
    return unless @project && !@project.empty_repo?

    ref = @ref || @project.repository.root_ref

    project_find_file_path(@project, ref)
  end
519 520 521 522

  def can_show_last_commit_in_list?(project)
    can?(current_user, :read_cross_project) && project.commit
  end
R
Rob Watson 已提交
523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540

  def pages_https_only_disabled?
    !@project.pages_domains.all?(&:https?)
  end

  def pages_https_only_title
    return unless pages_https_only_disabled?

    "You must enable HTTPS for all your domains first"
  end

  def pages_https_only_label_class
    if pages_https_only_disabled?
      "list-label disabled"
    else
      "list-label"
    end
  end
L
Lukas Eipert 已提交
541

542 543 544 545
  def sidebar_projects_paths
    %w[
      projects#show
      projects#activity
F
Filipa Lacerda 已提交
546
      releases#index
547 548 549 550
      cycle_analytics#show
    ]
  end

L
Lukas Eipert 已提交
551 552 553 554 555 556 557 558
  def sidebar_settings_paths
    %w[
      projects#edit
      project_members#index
      integrations#show
      services#edit
      repository#show
      ci_cd#show
559
      operations#show
L
Lukas Eipert 已提交
560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582
      badges#index
      pages#show
    ]
  end

  def sidebar_repository_paths
    %w[
      tree
      blob
      blame
      edit_tree
      new_tree
      find_file
      commit
      commits
      compare
      projects/repositories
      tags
      branches
      graphs
      network
    ]
  end
S
Simon Knox 已提交
583 584 585 586 587

  def sidebar_operations_paths
    %w[
      environments
      clusters
D
Dylan Griffith 已提交
588
      functions
589
      error_tracking
S
Simon Knox 已提交
590 591 592 593
      user
      gcp
    ]
  end
594 595 596 597 598 599 600

  def user_can_see_auto_devops_implicitly_enabled_banner?(project, user)
    Ability.allowed?(user, :admin_project, project) &&
      project.has_auto_devops_implicitly_enabled? &&
      project.builds_enabled? &&
      !project.repository.gitlab_ci_yml
  end
R
randx 已提交
601
end