gitlab-org--gitlab-foss/app/finders/todos_finder.rb

203 lines
3.7 KiB
Ruby

# frozen_string_literal: true
# TodosFinder
#
# Used to filter Todos by set of params
#
# Arguments:
# current_user - which user use
# params:
# action_id: integer
# author_id: integer
# project_id; integer
# state: 'pending' (default) or 'done'
# type: 'Issue' or 'MergeRequest'
#
class TodosFinder
prepend FinderWithCrossProjectAccess
include FinderMethods
include Gitlab::Utils::StrongMemoize
requires_cross_project_access unless: -> { project? }
NONE = '0'
TODO_TYPES = Set.new(%w(Issue MergeRequest Epic)).freeze
attr_accessor :current_user, :params
def initialize(current_user, params = {})
@current_user = current_user
@params = params
end
def execute
return Todo.none if current_user.nil?
items = current_user.todos
items = by_action_id(items)
items = by_action(items)
items = by_author(items)
items = by_state(items)
items = by_type(items)
items = by_group(items)
# Filtering by project HAS TO be the last because we use
# the project IDs yielded by the todos query thus far
items = by_project(items)
sort(items)
end
# Returns `true` if the current user has any todos for the given target with the optional given state.
#
# target - The value of the `target_type` column, such as `Issue`.
# state - The value of the `state` column, such as `pending` or `done`.
def any_for_target?(target, state = nil)
current_user.todos.any_for_target?(target, state)
end
private
def action_id?
action_id.present? && Todo::ACTION_NAMES.key?(action_id.to_i)
end
def action_id
params[:action_id]
end
def action_array_provided?
params[:action].is_a?(Array)
end
def map_actions_to_ids
params[:action].map { |item| Todo::ACTION_NAMES.key(item.to_sym) }
end
def to_action_id
if action_array_provided?
map_actions_to_ids
else
Todo::ACTION_NAMES.key(action.to_sym)
end
end
def action?
action.present? && to_action_id
end
def action
params[:action]
end
def author?
params[:author_id].present?
end
def author
strong_memoize(:author) do
if author? && params[:author_id] != NONE
User.find(params[:author_id])
end
end
end
def project?
params[:project_id].present?
end
def group?
params[:group_id].present?
end
def project
strong_memoize(:project) do
Project.find_without_deleted(params[:project_id]) if project?
end
end
def group
strong_memoize(:group) do
Group.find(params[:group_id])
end
end
def type?
type.present? && TODO_TYPES.include?(type)
end
def type
params[:type]
end
def sort(items)
if params[:sort]
items.sort_by_attribute(params[:sort])
else
items.order_id_desc
end
end
def by_action(items)
if action?
items.for_action(to_action_id)
else
items
end
end
def action_id_array_provided?
params[:action_id].is_a?(Array) && params[:action_id].any?
end
def by_action_ids(items)
items.for_action(action_id)
end
def by_action_id(items)
return by_action_ids(items) if action_id_array_provided?
if action_id?
by_action_ids(items)
else
items
end
end
def by_author(items)
if author?
items.for_author(author)
else
items
end
end
def by_project(items)
if project?
items.for_project(project)
else
items
end
end
def by_group(items)
return items unless group?
items.for_group_ids_and_descendants(params[:group_id])
end
def by_state(items)
return items.pending if params[:state].blank?
items.with_states(params[:state])
end
def by_type(items)
if type?
items.for_type(type)
else
items
end
end
end