2018-07-24 06:00:56 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-05-09 00:15:34 -04:00
|
|
|
class MergeRequestPresenter < Gitlab::View::Presenter::Delegated
|
|
|
|
include ActionView::Helpers::UrlHelper
|
|
|
|
include GitlabRoutingHelper
|
|
|
|
include MarkupHelper
|
|
|
|
include TreeHelper
|
2018-04-06 10:02:36 -04:00
|
|
|
include ChecksCollaboration
|
2018-02-07 11:32:32 -05:00
|
|
|
include Gitlab::Utils::StrongMemoize
|
2017-05-09 00:15:34 -04:00
|
|
|
|
2021-09-21 05:12:21 -04:00
|
|
|
delegator_override_with Gitlab::Utils::StrongMemoize # TODO: Remove `Gitlab::Utils::StrongMemoize` inclusion as it's duplicate
|
|
|
|
|
2020-07-15 14:09:09 -04:00
|
|
|
APPROVALS_WIDGET_BASE_TYPE = 'base'
|
|
|
|
|
2021-09-21 05:12:21 -04:00
|
|
|
presents ::MergeRequest, as: :merge_request
|
2017-05-09 00:15:34 -04:00
|
|
|
|
|
|
|
def ci_status
|
|
|
|
if pipeline
|
|
|
|
status = pipeline.status
|
2019-04-12 08:39:10 -04:00
|
|
|
status = "success-with-warnings" if pipeline.success? && pipeline.has_warnings?
|
2017-05-09 00:15:34 -04:00
|
|
|
|
|
|
|
status || "preparing"
|
|
|
|
else
|
2021-06-30 02:07:17 -04:00
|
|
|
ci_integration = source_project.try(:ci_integration)
|
|
|
|
ci_integration&.commit_status(diff_head_sha, source_branch)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-05-22 07:45:27 -04:00
|
|
|
def cancel_auto_merge_path
|
|
|
|
if can_cancel_auto_merge?(current_user)
|
|
|
|
cancel_auto_merge_project_merge_request_path(project, merge_request)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_issue_to_resolve_discussions_path
|
|
|
|
if can?(current_user, :create_issue, project) && project.issues_enabled?
|
2017-06-29 13:06:35 -04:00
|
|
|
new_project_issue_path(project, merge_request_to_resolve_discussions_of: iid)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def remove_wip_path
|
2020-10-01 08:10:14 -04:00
|
|
|
if can?(current_user, :update_merge_request, merge_request.project)
|
2017-06-29 13:06:35 -04:00
|
|
|
remove_wip_project_merge_request_path(project, merge_request)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def merge_path
|
|
|
|
if can_be_merged_by?(current_user)
|
2017-06-29 13:06:35 -04:00
|
|
|
merge_project_merge_request_path(project, merge_request)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def revert_in_fork_path
|
2018-02-07 12:30:41 -05:00
|
|
|
if user_can_fork_project? && cached_can_be_reverted?
|
2017-05-09 00:15:34 -04:00
|
|
|
continue_params = {
|
|
|
|
to: merge_request_path(merge_request),
|
2019-04-16 06:32:05 -04:00
|
|
|
notice: _('%{edit_in_new_fork_notice} Try to cherry-pick this commit again.') % { edit_in_new_fork_notice: edit_in_new_fork_notice },
|
2017-05-09 00:15:34 -04:00
|
|
|
notice_now: edit_in_new_fork_notice_now
|
|
|
|
}
|
|
|
|
|
2017-06-29 13:06:35 -04:00
|
|
|
project_forks_path(merge_request.project,
|
2017-05-09 00:15:34 -04:00
|
|
|
namespace_key: current_user.namespace.id,
|
|
|
|
continue: continue_params)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def cherry_pick_in_fork_path
|
|
|
|
if user_can_fork_project? && can_be_cherry_picked?
|
|
|
|
continue_params = {
|
|
|
|
to: merge_request_path(merge_request),
|
2019-04-16 06:32:05 -04:00
|
|
|
notice: _('%{edit_in_new_fork_notice} Try to revert this commit again.') % { edit_in_new_fork_notice: edit_in_new_fork_notice },
|
2017-05-09 00:15:34 -04:00
|
|
|
notice_now: edit_in_new_fork_notice_now
|
|
|
|
}
|
|
|
|
|
2017-06-29 13:06:35 -04:00
|
|
|
project_forks_path(project,
|
2017-05-09 00:15:34 -04:00
|
|
|
namespace_key: current_user.namespace.id,
|
|
|
|
continue: continue_params)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def conflict_resolution_path
|
2017-05-11 11:23:02 -04:00
|
|
|
if conflicts.can_be_resolved_in_ui? && conflicts.can_be_resolved_by?(current_user)
|
2017-06-29 13:06:35 -04:00
|
|
|
conflicts_project_merge_request_path(project, merge_request)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-12-20 04:01:21 -05:00
|
|
|
def rebase_path
|
2018-02-28 03:06:18 -05:00
|
|
|
if !rebase_in_progress? && should_be_rebased? && can_push_to_source_branch?
|
2017-12-20 04:01:21 -05:00
|
|
|
rebase_project_merge_request_path(project, merge_request)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-07-11 10:48:59 -04:00
|
|
|
def target_branch_tree_path
|
2017-05-09 00:15:34 -04:00
|
|
|
if target_branch_exists?
|
2017-07-11 10:48:59 -04:00
|
|
|
project_tree_path(project, target_branch)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-07-11 12:49:22 -04:00
|
|
|
def target_branch_commits_path
|
|
|
|
if target_branch_exists?
|
|
|
|
project_commits_path(project, target_branch)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-01 05:05:04 -05:00
|
|
|
def target_branch_path
|
|
|
|
if target_branch_exists?
|
|
|
|
project_branch_path(project, target_branch)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-06 03:34:38 -05:00
|
|
|
def source_branch_commits_path
|
|
|
|
if source_branch_exists?
|
|
|
|
project_commits_path(source_project, source_branch)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-09 00:15:34 -04:00
|
|
|
def source_branch_path
|
|
|
|
if source_branch_exists?
|
2017-06-29 13:06:35 -04:00
|
|
|
project_branch_path(source_project, source_branch)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def source_branch_with_namespace_link
|
|
|
|
namespace = source_project_namespace
|
|
|
|
branch = source_branch
|
|
|
|
|
2018-10-12 17:39:13 -04:00
|
|
|
namespace_link = source_branch_exists? ? link_to(namespace, project_path(source_project)) : ERB::Util.html_escape(namespace)
|
|
|
|
branch_link = source_branch_exists? ? link_to(branch, project_tree_path(source_project, source_branch)) : ERB::Util.html_escape(branch)
|
2017-05-09 00:15:34 -04:00
|
|
|
|
2018-10-12 17:39:13 -04:00
|
|
|
for_fork? ? "#{namespace_link}:#{branch_link}" : branch_link
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def closing_issues_links
|
2017-06-08 12:09:04 -04:00
|
|
|
markdown(
|
|
|
|
issues_sentence(project, closing_issues),
|
|
|
|
pipeline: :gfm,
|
|
|
|
author: author,
|
|
|
|
project: project,
|
2021-11-24 07:10:21 -05:00
|
|
|
issuable_reference_expansion_enabled: true
|
2017-06-08 12:09:04 -04:00
|
|
|
)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def mentioned_issues_links
|
2017-06-08 12:09:04 -04:00
|
|
|
markdown(
|
|
|
|
issues_sentence(project, mentioned_issues),
|
|
|
|
pipeline: :gfm,
|
|
|
|
author: author,
|
|
|
|
project: project,
|
2021-11-24 07:10:21 -05:00
|
|
|
issuable_reference_expansion_enabled: true
|
2017-06-08 12:09:04 -04:00
|
|
|
)
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def assign_to_closing_issues_link
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ServiceClass
|
2021-05-11 23:10:21 -04:00
|
|
|
issues = MergeRequests::AssignIssuesService.new(project: project,
|
|
|
|
current_user: current_user,
|
|
|
|
params: {
|
|
|
|
merge_request: merge_request,
|
|
|
|
closes_issues: closing_issues
|
|
|
|
}).assignable_issues
|
2017-06-29 13:06:35 -04:00
|
|
|
path = assign_related_issues_project_merge_request_path(project, merge_request)
|
2017-05-09 00:15:34 -04:00
|
|
|
if issues.present?
|
2019-04-16 06:32:05 -04:00
|
|
|
if issues.count > 1
|
|
|
|
link_to _('Assign yourself to these issues'), path, method: :post
|
|
|
|
else
|
|
|
|
link_to _('Assign yourself to this issue'), path, method: :post
|
|
|
|
end
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ServiceClass
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def can_revert_on_current_merge_request?
|
2018-04-06 10:02:36 -04:00
|
|
|
can_collaborate_with_project?(project) && cached_can_be_reverted?
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def can_cherry_pick_on_current_merge_request?
|
2018-04-06 10:02:36 -04:00
|
|
|
can_collaborate_with_project?(project) && can_be_cherry_picked?
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
2017-12-20 04:01:21 -05:00
|
|
|
def can_push_to_source_branch?
|
2018-02-28 03:06:18 -05:00
|
|
|
return false unless source_branch_exists?
|
|
|
|
|
|
|
|
!!::Gitlab::UserAccess
|
2020-07-21 14:09:45 -04:00
|
|
|
.new(current_user, container: source_project)
|
2018-02-28 03:06:18 -05:00
|
|
|
.can_push_to_branch?(source_branch)
|
2017-12-20 04:01:21 -05:00
|
|
|
end
|
|
|
|
|
2021-09-21 05:12:21 -04:00
|
|
|
delegator_override :can_remove_source_branch?
|
2018-06-25 04:59:00 -04:00
|
|
|
def can_remove_source_branch?
|
|
|
|
source_branch_exists? && merge_request.can_remove_source_branch?(current_user)
|
|
|
|
end
|
|
|
|
|
2019-01-28 07:12:30 -05:00
|
|
|
def can_read_pipeline?
|
|
|
|
pipeline && can?(current_user, :read_pipeline, pipeline)
|
|
|
|
end
|
|
|
|
|
2018-05-21 03:52:24 -04:00
|
|
|
def mergeable_discussions_state
|
|
|
|
# This avoids calling MergeRequest#mergeable_discussions_state without
|
|
|
|
# considering the state of the MR first. If a MR isn't mergeable, we can
|
|
|
|
# safely short-circuit it.
|
|
|
|
if merge_request.mergeable_state?(skip_ci_check: true, skip_discussions_check: true)
|
|
|
|
merge_request.mergeable_discussions_state?
|
|
|
|
else
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-21 05:12:21 -04:00
|
|
|
delegator_override :subscribed?
|
2018-05-21 03:52:24 -04:00
|
|
|
def subscribed?
|
|
|
|
merge_request.subscribed?(current_user, merge_request.target_project)
|
|
|
|
end
|
|
|
|
|
2019-04-29 03:29:54 -04:00
|
|
|
def source_branch_link
|
|
|
|
if source_branch_exists?
|
|
|
|
link_to(source_branch, source_branch_commits_path, class: 'ref-name')
|
|
|
|
else
|
|
|
|
content_tag(:span, source_branch, class: 'ref-name')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def target_branch_link
|
|
|
|
if target_branch_exists?
|
|
|
|
link_to(target_branch, target_branch_commits_path, class: 'ref-name')
|
|
|
|
else
|
|
|
|
content_tag(:span, target_branch, class: 'ref-name')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-15 14:09:09 -04:00
|
|
|
def api_approvals_path
|
|
|
|
expose_path(api_v4_projects_merge_requests_approvals_path(id: project.id, merge_request_iid: merge_request.iid))
|
|
|
|
end
|
|
|
|
|
|
|
|
def api_approve_path
|
|
|
|
expose_path(api_v4_projects_merge_requests_approve_path(id: project.id, merge_request_iid: merge_request.iid))
|
|
|
|
end
|
|
|
|
|
|
|
|
def api_unapprove_path
|
|
|
|
expose_path(api_v4_projects_merge_requests_unapprove_path(id: project.id, merge_request_iid: merge_request.iid))
|
|
|
|
end
|
|
|
|
|
|
|
|
def approvals_widget_type
|
|
|
|
APPROVALS_WIDGET_BASE_TYPE
|
|
|
|
end
|
|
|
|
|
2021-08-16 14:10:51 -04:00
|
|
|
def closing_issues
|
|
|
|
strong_memoize(:closing_issues) do
|
|
|
|
visible_closing_issues_for(current_user)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def mentioned_issues
|
|
|
|
strong_memoize(:mentioned_issues) do
|
|
|
|
issues_mentioned_but_not_closing(current_user)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-09 00:15:34 -04:00
|
|
|
private
|
|
|
|
|
2018-02-07 12:30:41 -05:00
|
|
|
def cached_can_be_reverted?
|
2018-02-07 11:32:32 -05:00
|
|
|
strong_memoize(:can_be_reverted) do
|
|
|
|
can_be_reverted?(current_user)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-11 11:23:02 -04:00
|
|
|
def conflicts
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ServiceClass
|
2017-05-11 11:23:02 -04:00
|
|
|
@conflicts ||= MergeRequests::Conflicts::ListService.new(merge_request)
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ServiceClass
|
2017-05-11 11:23:02 -04:00
|
|
|
end
|
|
|
|
|
2017-05-09 00:15:34 -04:00
|
|
|
def pipeline
|
2017-12-01 08:26:51 -05:00
|
|
|
@pipeline ||= actual_head_pipeline
|
2017-05-09 00:15:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def issues_sentence(project, issues)
|
|
|
|
# Sorting based on the `#123` or `group/project#123` reference will sort
|
|
|
|
# local issues first.
|
|
|
|
issues.map do |issue|
|
|
|
|
issue.to_reference(project)
|
|
|
|
end.sort.to_sentence
|
|
|
|
end
|
|
|
|
|
|
|
|
def user_can_fork_project?
|
|
|
|
can?(current_user, :fork_project, project)
|
|
|
|
end
|
|
|
|
end
|
2019-09-13 09:26:31 -04:00
|
|
|
|
2021-05-11 17:10:21 -04:00
|
|
|
MergeRequestPresenter.prepend_mod_with('MergeRequestPresenter')
|