# frozen_string_literal: true module MergeRequests # Used to filter MergeRequest collections by approvers class ByApprovalsFinder attr_reader :usernames, :ids # We apply a limitation to the amount of elements that can be part of the filter condition MAX_FILTER_ELEMENTS = 5 # Initialize the finder # # @param [Array] usernames # @param [Array] ids def initialize(usernames, ids) # rubocop:disable CodeReuse/ActiveRecord @usernames = Array(usernames).map(&:to_s).uniq.take(MAX_FILTER_ELEMENTS) @ids = Array(ids).uniq.take(MAX_FILTER_ELEMENTS) # rubocop:enable CodeReuse/ActiveRecord end # Filter MergeRequest collections by approvers # # @param [ActiveRecord::Relation] items the activerecord relation def execute(items) if by_no_approvals? without_approvals(items) elsif by_any_approvals? with_any_approvals(items) elsif ids.present? find_approved_by_ids(items) elsif usernames.present? find_approved_by_names(items) else items end end private # Is param using special condition: "None" ? # # @return [Boolean] whether special condition "None" is being used def by_no_approvals? includes_special_label?(IssuableFinder::Params::FILTER_NONE) end # Is param using special condition: "Any" ? # # @return [Boolean] whether special condition "Any" is being used def by_any_approvals? includes_special_label?(IssuableFinder::Params::FILTER_ANY) end # Check if we have the special label in ids or usernames field # # @param [String] label the special label # @return [Boolean] whether ids or usernames includes the special label def includes_special_label?(label) ids.first.to_s.downcase == label || usernames.map(&:downcase).include?(label) end # Merge requests without any approval # # @param [ActiveRecord::Relation] items def without_approvals(items) items.without_approvals end # Merge requests with any number of approvals # # @param [ActiveRecord::Relation] items the activerecord relation def with_any_approvals(items) items.select_from_union([ items.with_approvals ]) end # Merge requests approved by given usernames # # @param [ActiveRecord::Relation] items the activerecord relation def find_approved_by_names(items) items.approved_by_users_with_usernames(*usernames) end # Merge requests approved by given user IDs # # @param [ActiveRecord::Relation] items the activerecord relation def find_approved_by_ids(items) items.approved_by_users_with_ids(*ids) end end end