debian-mirror-gitlab/app/finders/merge_requests_finder.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

261 lines
7.1 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
# Finders::MergeRequest class
#
# Used to filter MergeRequests collections by set of params
#
# Arguments:
# current_user - which user use
# params:
2018-11-08 19:23:39 +05:30
# scope: 'created_by_me' or 'assigned_to_me' or 'all'
# state: 'open', 'closed', 'merged', 'locked', or 'all'
2014-09-02 18:07:02 +05:30
# group_id: integer
# project_id: integer
2017-08-17 22:00:37 +05:30
# milestone_title: string
2020-01-01 13:55:28 +05:30
# release_tag: string
2017-09-10 17:25:29 +05:30
# author_id: integer
2020-04-22 19:07:51 +05:30
# author_username: string
2014-09-02 18:07:02 +05:30
# assignee_id: integer
# search: string
2019-03-02 22:35:43 +05:30
# in: 'title', 'description', or a string joining them with comma
2014-09-02 18:07:02 +05:30
# label_name: string
# sort: string
2017-08-17 22:00:37 +05:30
# non_archived: boolean
2018-03-17 18:26:18 +05:30
# my_reaction_emoji: string
2018-03-27 19:54:05 +05:30
# source_branch: string
# target_branch: string
# created_after: datetime
# created_before: datetime
# updated_after: datetime
# updated_before: datetime
2014-09-02 18:07:02 +05:30
#
2015-04-26 12:48:37 +05:30
class MergeRequestsFinder < IssuableFinder
2022-10-11 01:57:18 +05:30
extend ::Gitlab::Utils::Override
2020-10-24 23:57:45 +05:30
include MergedAtFilter
2018-12-05 23:21:45 +05:30
def self.scalar_params
2021-01-03 14:25:43 +05:30
@scalar_params ||= super + [
2023-05-27 22:25:52 +05:30
:approved,
2021-01-03 14:25:43 +05:30
:approved_by_ids,
:deployed_after,
:deployed_before,
:draft,
:environment,
:merged_after,
:merged_before,
2021-02-22 17:27:13 +05:30
:reviewer_id,
:reviewer_username,
2021-01-03 14:25:43 +05:30
:target_branch,
2022-10-11 01:57:18 +05:30
:wip
2021-01-03 14:25:43 +05:30
]
end
def self.array_params
@array_params ||= super.merge(approved_by_usernames: [])
2018-12-05 23:21:45 +05:30
end
2014-09-02 18:07:02 +05:30
def klass
MergeRequest
end
2016-09-29 09:46:39 +05:30
2021-02-22 17:27:13 +05:30
def params_class
2021-03-08 18:12:59 +05:30
MergeRequestsFinder.const_get(:Params, false) # rubocop: disable CodeReuse/Finder
2021-02-22 17:27:13 +05:30
end
2018-03-27 19:54:05 +05:30
def filter_items(_items)
2019-03-13 22:55:13 +05:30
items = by_commit(super)
items = by_source_branch(items)
2020-10-24 23:57:45 +05:30
items = by_draft(items)
2019-07-07 11:18:12 +05:30
items = by_target_branch(items)
2020-10-24 23:57:45 +05:30
items = by_merged_at(items)
2021-01-03 14:25:43 +05:30
items = by_approvals(items)
items = by_deployments(items)
2021-02-22 17:27:13 +05:30
items = by_reviewer(items)
2023-05-27 22:25:52 +05:30
items = by_source_project_id(items)
2021-01-03 14:25:43 +05:30
2023-05-27 22:25:52 +05:30
by_approved(items)
2018-03-27 19:54:05 +05:30
end
2021-01-29 00:20:46 +05:30
def filter_negated_items(items)
items = super(items)
2021-02-22 17:27:13 +05:30
items = by_negated_reviewer(items)
2021-09-30 23:02:18 +05:30
items = by_negated_approved_by(items)
2021-01-29 00:20:46 +05:30
by_negated_target_branch(items)
end
2016-09-29 09:46:39 +05:30
private
2022-10-11 01:57:18 +05:30
override :sort
def sort(items)
items = super(items)
return items unless use_grouping_columns?
grouping_columns = klass.grouping_columns(params[:sort])
items.group(grouping_columns) # rubocop:disable CodeReuse/ActiveRecord
end
2019-03-13 22:55:13 +05:30
def by_commit(items)
return items unless params[:commit_sha].presence
2021-03-08 18:12:59 +05:30
items.by_related_commit_sha(params[:commit_sha])
2019-03-13 22:55:13 +05:30
end
2018-03-27 19:54:05 +05:30
def source_branch
@source_branch ||= params[:source_branch].presence
end
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2018-03-27 19:54:05 +05:30
def by_source_branch(items)
return items unless source_branch
items.where(source_branch: source_branch)
end
2018-12-05 23:21:45 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2018-03-27 19:54:05 +05:30
def target_branch
@target_branch ||= params[:target_branch].presence
end
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2018-03-27 19:54:05 +05:30
def by_target_branch(items)
return items unless target_branch
items.where(target_branch: target_branch)
end
2021-01-03 14:25:43 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2018-12-05 23:21:45 +05:30
2021-01-29 00:20:46 +05:30
# rubocop: disable CodeReuse/ActiveRecord
def by_negated_target_branch(items)
return items unless not_params[:target_branch]
items.where.not(target_branch: not_params[:target_branch])
end
# rubocop: enable CodeReuse/ActiveRecord
2021-09-30 23:02:18 +05:30
def by_negated_approved_by(items)
return items unless not_params[:approved_by_usernames]
items.not_approved_by_users_with_usernames(not_params[:approved_by_usernames])
end
2019-07-07 11:18:12 +05:30
def source_project_id
@source_project_id ||= params[:source_project_id].presence
end
2021-01-03 14:25:43 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2019-07-07 11:18:12 +05:30
def by_source_project_id(items)
return items unless source_project_id
items.where(source_project_id: source_project_id)
end
2021-01-03 14:25:43 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2019-07-07 11:18:12 +05:30
2021-01-03 14:25:43 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2020-10-24 23:57:45 +05:30
def by_draft(items)
2022-03-02 08:16:31 +05:30
draft_param = Gitlab::Utils.to_boolean(params.fetch(:draft) { params.fetch(:wip, nil) })
return items if draft_param.nil?
2020-10-24 23:57:45 +05:30
2022-03-02 08:16:31 +05:30
if draft_param
2018-12-05 23:21:45 +05:30
items.where(wip_match(items.arel_table))
else
2022-03-02 08:16:31 +05:30
items.where.not(wip_match(items.arel_table))
2018-12-05 23:21:45 +05:30
end
end
2021-01-03 14:25:43 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2018-12-05 23:21:45 +05:30
2020-10-24 23:57:45 +05:30
# WIP is deprecated in favor of Draft. Currently both options are supported
2018-12-05 23:21:45 +05:30
def wip_match(table)
2021-10-27 15:23:28 +05:30
table[:title].matches('WIP:%')
.or(table[:title].matches('WIP %'))
.or(table[:title].matches('[WIP]%'))
2020-10-24 23:57:45 +05:30
.or(table[:title].matches('Draft - %'))
.or(table[:title].matches('Draft:%'))
.or(table[:title].matches('[Draft]%'))
.or(table[:title].matches('(Draft)%'))
2018-12-05 23:21:45 +05:30
end
2020-03-13 15:44:24 +05:30
2021-01-03 14:25:43 +05:30
# Filter by merge requests that had been approved by specific users
# rubocop: disable CodeReuse/Finder
def by_approvals(items)
MergeRequests::ByApprovalsFinder
.new(params[:approved_by_usernames], params[:approved_by_ids])
.execute(items)
end
# rubocop: enable CodeReuse/Finder
2023-05-27 22:25:52 +05:30
def by_approved(items)
approved_param = Gitlab::Utils.to_boolean(params.fetch(:approved, nil))
2023-06-20 00:43:36 +05:30
return items if approved_param.nil? || Feature.disabled?(:mr_approved_filter, type: :ops)
2023-05-27 22:25:52 +05:30
if approved_param
items.with_approvals
else
items.without_approvals
end
end
2021-01-03 14:25:43 +05:30
def by_deployments(items)
env = params[:environment]
2022-01-26 12:08:38 +05:30
before = parse_datetime(params[:deployed_before])
after = parse_datetime(params[:deployed_after])
2021-01-03 14:25:43 +05:30
id = params[:deployment_id]
return items if !env && !before && !after && !id
# Each filter depends on the same JOIN+WHERE. To prevent this JOIN+WHERE
# from being duplicated for every filter, we only produce it once. The
# filter methods in turn expect the JOIN+WHERE to already be present.
#
# This approach ensures that query performance doesn't degrade as the number
# of deployment related filters increases.
deploys = DeploymentMergeRequest.join_deployments_for_merge_requests
deploys = deploys.by_deployment_id(id) if id
deploys = deploys.deployed_to(env) if env
deploys = deploys.deployed_before(before) if before
deploys = deploys.deployed_after(after) if after
items.where_exists(deploys)
end
2021-02-22 17:27:13 +05:30
def by_reviewer(items)
return items unless params.reviewer_id? || params.reviewer_username?
if params.filter_by_no_reviewer?
items.no_review_requested
elsif params.filter_by_any_reviewer?
items.review_requested
elsif params.reviewer
items.review_requested_to(params.reviewer)
else # reviewer not found
items.none
end
end
def by_negated_reviewer(items)
return items unless not_params.reviewer_id? || not_params.reviewer_username?
if not_params.reviewer.present?
items.no_review_requested_to(not_params.reviewer)
else # reviewer not found
items.none
end
end
2022-01-26 12:08:38 +05:30
def parse_datetime(input)
# To work around http://www.ruby-lang.org/en/news/2021/11/15/date-parsing-method-regexp-dos-cve-2021-41817/
DateTime.parse(input.byteslice(0, 128)) if input
rescue Date::Error
nil
end
2022-10-11 01:57:18 +05:30
def use_grouping_columns?
return false unless params[:sort].present?
params[:approved_by_usernames].present? || params[:approved_by_ids].present?
end
2014-09-02 18:07:02 +05:30
end
2019-12-04 20:38:33 +05:30
2021-06-08 01:23:25 +05:30
MergeRequestsFinder.prepend_mod_with('MergeRequestsFinder')