# frozen_string_literal: true class MergeRequestWidgetEntity < Grape::Entity include RequestAwareEntity include ProjectsHelper include ApplicationHelper include ApplicationSettingsHelper SUGGEST_PIPELINE = 'suggest_pipeline' expose :id expose :iid expose :source_project_full_path do |merge_request| merge_request.source_project&.full_path end expose :target_project_full_path do |merge_request| merge_request.project&.full_path end expose :can_create_pipeline_in_target_project do |merge_request| can?(current_user, :create_pipeline, merge_request.target_project) end expose :email_patches_path do |merge_request| project_merge_request_path(merge_request.project, merge_request, format: :patch) end expose :plain_diff_path do |merge_request| project_merge_request_path(merge_request.project, merge_request, format: :diff) end expose :merge_request_basic_path do |merge_request| project_merge_request_path(merge_request.target_project, merge_request, serializer: :basic, format: :json) end expose :merge_request_widget_path do |merge_request| widget_project_json_merge_request_path(merge_request.target_project, merge_request, format: :json) end expose :merge_request_cached_widget_path do |merge_request| cached_widget_project_json_merge_request_path(merge_request.target_project, merge_request, format: :json) end expose :commit_change_content_path do |merge_request| commit_change_content_project_merge_request_path(merge_request.project, merge_request) end expose :conflicts_docs_path do |merge_request| help_page_path('user/project/merge_requests/resolve_conflicts.md') end expose :reviewing_and_managing_merge_requests_docs_path do |merge_request| help_page_path('user/project/merge_requests/reviewing_and_managing_merge_requests.md', anchor: "checkout-merge-requests-locally-through-the-head-ref") end expose :merge_request_pipelines_docs_path do |merge_request| help_page_path('ci/merge_request_pipelines/index.md') end expose :ci_environments_status_path do |merge_request| ci_environments_status_project_merge_request_path(merge_request.project, merge_request) end expose :merge_request_add_ci_config_path, if: ->(mr, _) { can_add_ci_config_path?(mr) } do |merge_request| project_new_blob_path( merge_request.source_project, merge_request.source_branch, file_name: '.gitlab-ci.yml', commit_message: s_("CommitMessage|Add %{file_name}") % { file_name: Gitlab::FileDetector::PATTERNS[:gitlab_ci] }, mr_path: merge_request_path(merge_request), suggest_gitlab_ci_yml: true ) end expose :user_callouts_path do |_merge_request| user_callouts_path end expose :suggest_pipeline_feature_id do |_merge_request| SUGGEST_PIPELINE end expose :is_dismissed_suggest_pipeline do |_merge_request| current_user && current_user.dismissed_callout?(feature_name: SUGGEST_PIPELINE) end expose :human_access do |merge_request| merge_request.project.team.human_max_access(current_user&.id) end expose :new_project_pipeline_path do |merge_request| new_project_pipeline_path(merge_request.project) end expose :source_project_default_url do |merge_request| merge_request.source_project && default_url_to_repo(merge_request.source_project) end # Rendering and redacting Markdown can be expensive. These links are # just nice to have in the merge request widget, so only # include them if they are explicitly requested on first load. expose :issues_links, if: -> (_, opts) { opts[:issues_links] } do expose :assign_to_closing do |merge_request| presenter(merge_request).assign_to_closing_issues_link end expose :closing do |merge_request| presenter(merge_request).closing_issues_links end expose :mentioned_but_not_closing do |merge_request| presenter(merge_request).mentioned_issues_links end end expose :codeclimate, if: -> (mr, _) { head_pipeline_downloadable_path_for_report_type(:codequality) } do expose :head_path do |merge_request| head_pipeline_downloadable_path_for_report_type(:codequality) end expose :base_path do |merge_request| if use_merge_base_with_merged_results? merge_base_pipeline_downloadable_path_for_report_type(:codequality) else base_pipeline_downloadable_path_for_report_type(:codequality) end end end expose :security_reports_docs_path do |merge_request| help_page_path('user/application_security/index.md', anchor: 'viewing-security-scan-information-in-merge-requests') end expose :enabled_reports do |merge_request| merge_request.enabled_reports end private delegate :current_user, to: :request def presenter(merge_request) @presenters ||= {} @presenters[merge_request] ||= MergeRequestPresenter.new(merge_request, current_user: current_user) # rubocop: disable CodeReuse/Presenter end def can_add_ci_config_path?(merge_request) merge_request.open? && merge_request.source_branch_exists? && merge_request.source_project&.uses_default_ci_config? && !merge_request.source_project.has_ci? && merge_request.commits_count > 0 && can?(current_user, :read_build, merge_request.source_project) && can?(current_user, :create_pipeline, merge_request.source_project) end def use_merge_base_with_merged_results? object.actual_head_pipeline&.merge_request_event_type == :merged_result end def head_pipeline_downloadable_path_for_report_type(file_type) object.head_pipeline&.present(current_user: current_user) &.downloadable_path_for_report_type(file_type) end def base_pipeline_downloadable_path_for_report_type(file_type) object.base_pipeline&.present(current_user: current_user) &.downloadable_path_for_report_type(file_type) end def merge_base_pipeline_downloadable_path_for_report_type(file_type) object.merge_base_pipeline&.present(current_user: current_user) &.downloadable_path_for_report_type(file_type) end end MergeRequestWidgetEntity.prepend_if_ee('EE::MergeRequestWidgetEntity')