Commit dda023d6 authored by Bob Van Landuyt's avatar Bob Van Landuyt

Optimize queries and pagination in `GroupDescendantsFinder`

parent 57bd3bb3
......@@ -36,10 +36,13 @@ class GroupDescendantsFinder
def execute
# The children array might be extended with the ancestors of projects when
# filtering. In that case, take the maximum so the aray does not get limited
# Otherwise, allow paginating through the search results
# Otherwise, allow paginating through all results
#
total_count = [children.size, subgroup_count + project_count].max
Kaminari.paginate_array(children, total_count: total_count)
all_required_elements = children
all_required_elements |= ancestors_for_projects if params[:filter]
total_count = [all_required_elements.size, paginator.total_count].max
Kaminari.paginate_array(all_required_elements, total_count: total_count)
end
def subgroup_count
......@@ -53,50 +56,15 @@ class GroupDescendantsFinder
private
def children
return @children if @children
subgroups_with_counts = subgroups.with_route
.page(params[:page]).per(per_page)
.select(GROUP_SELECTS)
paginated_projects = paginate_projects_after_groups(subgroups_with_counts)
subgroups_with_counts = add_project_ancestors_when_searching(subgroups_with_counts, paginated_projects)
@children = subgroups_with_counts + paginated_projects
@children ||= paginator.paginate(params[:page])
end
def add_project_ancestors_when_searching(groups, projects)
return groups unless params[:filter]
project_ancestors = ancestors_for_projects(projects)
.with_route.select(GROUP_SELECTS)
groups | project_ancestors
end
def paginate_projects_after_groups(loaded_subgroups)
# We adjust the pagination for projects for the combination with groups:
# - We limit the first page (page 0) where we show projects:
# Page size = 20: 17 groups, 3 projects
# - We ofset the page to start at 0 after the group pages:
# 3 pages of projects:
# - currently on page 3: Show page 0 (first page) limited to the number of
# projects that still fit the page (no offset)
# - currently on page 4: Show page 1 show all projects, offset by the number
# of projects shown on project-page 0.
group_page_count = loaded_subgroups.total_pages
subgroup_page = loaded_subgroups.current_page
group_last_page_count = subgroups.page(group_page_count).count
project_page = subgroup_page - group_page_count
offset = if project_page.zero? || group_page_count.zero?
0
else
per_page - group_last_page_count
def collections
[subgroups.with_route.select(GROUP_SELECTS), projects]
end
projects.with_route.page(project_page)
.per(per_page - loaded_subgroups.size)
.padding(offset)
def paginator
Gitlab::MultiCollectionPaginator.new(*collections, per_page: params[:per_page])
end
def direct_child_groups
......@@ -107,22 +75,22 @@ class GroupDescendantsFinder
def all_visible_descendant_groups
groups_table = Group.arel_table
visible_for_user = groups_table[:visibility_level]
.in(Gitlab::VisibilityLevel.levels_for_user(current_user))
visible_for_user = if current_user
groups_table[:id].in(
Arel::Nodes::SqlLiteral.new(GroupsFinder.new(current_user, all_available: true).execute.select(:id).to_sql)
)
else
groups_table[:visibility_level].eq(Gitlab::VisibilityLevel::PUBLIC)
end
Gitlab::GroupHierarchy.new(Group.where(id: parent_group))
.base_and_descendants
visible_projects = GroupsFinder.new(current_user).execute.as('visible')
authorized = groups_table.project(1).from(visible_projects)
.where(visible_projects[:id].eq(groups_table[:id]))
.exists
visible_for_user.or(authorized)
end
hierarchy_for_parent
.descendants
.where(visible_for_user)
end
def subgroups_matching_filter
all_visible_descendant_groups
.where.not(id: parent_group)
.search(params[:filter])
end
......@@ -137,14 +105,15 @@ class GroupDescendantsFinder
# So when searching 'project', on the 'subgroup' page we want to preload
# 'nested-group' but not 'subgroup' or 'root'
def ancestors_for_groups(base_for_ancestors)
ancestors_for_parent = Gitlab::GroupHierarchy.new(Group.where(id: parent_group))
.base_and_ancestors
Gitlab::GroupHierarchy.new(base_for_ancestors)
.base_and_ancestors.where.not(id: ancestors_for_parent)
.base_and_ancestors(upto: parent_group.id)
end
def ancestors_for_projects(projects)
ancestors_for_groups(Group.where(id: projects.select(:namespace_id)))
def ancestors_for_projects
projects_to_load_ancestors_of = projects.where.not(namespace: parent_group)
groups_to_load_ancestors_of = Group.where(id: projects_to_load_ancestors_of.select(:namespace_id))
ancestors_for_groups(groups_to_load_ancestors_of)
.with_route.select(GROUP_SELECTS)
end
def subgroups
......@@ -169,9 +138,11 @@ class GroupDescendantsFinder
projects_for_user.where(namespace: parent_group)
end
# Finds all projects nested under `parent_group` or any of it's descendant
# groups
def projects_matching_filter
projects_for_user.search(params[:filter])
.where(namespace: all_visible_descendant_groups)
.where(namespace_id: hierarchy_for_parent.base_and_descendants.select(:id))
end
def projects
......@@ -182,14 +153,14 @@ class GroupDescendantsFinder
else
direct_child_projects
end
projects.order_by(sort)
projects.with_route.order_by(sort)
end
def sort
params.fetch(:sort, 'id_asc')
end
def per_page
params.fetch(:per_page, Kaminari.config.default_per_page)
def hierarchy_for_parent
@hierarchy ||= Gitlab::GroupHierarchy.new(Group.where(id: parent_group.id))
end
end
module Gitlab
class MultiCollectionPaginator
attr_reader :first_collection, :second_collection, :per_page
def initialize(*collections, per_page: nil)
raise ArgumentError.new('Only 2 collections are supported') if collections.size != 2
@per_page = per_page || Kaminari.config.default_per_page
@first_collection, @second_collection = collections
end
def paginate(page)
page = page.to_i
paginated_first_collection(page) + paginated_second_collection(page)
end
def total_count
@total_count ||= first_collection.size + second_collection.size
end
private
def paginated_first_collection(page)
@first_collection_pages ||= Hash.new do |hash, page|
hash[page] = first_collection.page(page).per(per_page)
end
@first_collection_pages[page]
end
def paginated_second_collection(page)
@second_collection_pages ||= Hash.new do |hash, page|
second_collection_page = page - first_collection_page_count
offset = if second_collection_page < 1 || first_collection_page_count.zero?
0
else
per_page - first_collection_last_page_size
end
hash[page] = second_collection.page(second_collection_page)
.per(per_page - paginated_first_collection(page).size)
.padding(offset)
end
@second_collection_pages[page]
end
def first_collection_page_count
return @first_collection_page_count if defined?(@first_collection_page_count)
first_collection_page = paginated_first_collection(0)
@first_collection_page_count = first_collection_page.total_pages
end
def first_collection_last_page_size
return @first_collection_last_page_size if defined?(@first_collection_last_page_size)
@first_collection_last_page_size = paginated_first_collection(first_collection_page_count).count
end
end
end
require 'spec_helper'
describe Gitlab::MultiCollectionPaginator do
subject(:paginator) { described_class.new(Project.all, Group.all, per_page: 3) }
it 'combines both collections' do
project = create(:project)
group = create(:group)
expect(paginator.paginate(1)).to eq([project, group])
end
it 'includes elements second collection if first collection is empty' do
group = create(:group)
expect(paginator.paginate(1)).to eq([group])
end
context 'with a full first page' do
let!(:all_groups) { create_list(:group, 4) }
let!(:all_projects) { create_list(:project, 4) }
it 'knows the total count of the collection' do
expect(paginator.total_count).to eq(8)
end
it 'fills the first page with elements of the first collection' do
expect(paginator.paginate(1)).to eq(all_projects.take(3))
end
it 'fils the second page with a mixture of of the first & second collection' do
first_collection_element = all_projects.last
second_collection_elements = all_groups.take(2)
expected_collection = [first_collection_element] + second_collection_elements
expect(paginator.paginate(2)).to eq(expected_collection)
end
it 'fils the last page with elements from the second collection' do
expected_collection = all_groups[-2..-1]
expect(paginator.paginate(3)).to eq(expected_collection)
end
end
end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment