Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Léo-Paul Géneau
gitlab-ce
Commits
dda023d6
Commit
dda023d6
authored
Oct 04, 2017
by
Bob Van Landuyt
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Optimize queries and pagination in `GroupDescendantsFinder`
parent
57bd3bb3
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
139 additions
and
61 deletions
+139
-61
app/finders/group_descendants_finder.rb
app/finders/group_descendants_finder.rb
+32
-61
lib/gitlab/multi_collection_paginator.rb
lib/gitlab/multi_collection_paginator.rb
+61
-0
spec/lib/gitlab/multi_collection_paginator_spec.rb
spec/lib/gitlab/multi_collection_paginator_spec.rb
+46
-0
No files found.
app/finders/group_descendants_finder.rb
View file @
dda023d6
...
@@ -36,10 +36,13 @@ class GroupDescendantsFinder
...
@@ -36,10 +36,13 @@ class GroupDescendantsFinder
def
execute
def
execute
# The children array might be extended with the ancestors of projects when
# 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
# 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
all_required_elements
=
children
Kaminari
.
paginate_array
(
children
,
total_count:
total_count
)
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
end
def
subgroup_count
def
subgroup_count
...
@@ -53,50 +56,15 @@ class GroupDescendantsFinder
...
@@ -53,50 +56,15 @@ class GroupDescendantsFinder
private
private
def
children
def
children
return
@children
if
@children
@children
||=
paginator
.
paginate
(
params
[
:page
])
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
end
end
def
add_project_ancestors_when_searching
(
groups
,
projects
)
def
collections
return
groups
unless
params
[
:filter
]
[
subgroups
.
with_route
.
select
(
GROUP_SELECTS
),
projects
]
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
end
end
projects
.
with_route
.
page
(
project_page
)
def
paginator
.
per
(
per_page
-
loaded_subgroups
.
size
)
Gitlab
::
MultiCollectionPaginator
.
new
(
*
collections
,
per_page:
params
[
:per_page
])
.
padding
(
offset
)
end
end
def
direct_child_groups
def
direct_child_groups
...
@@ -107,22 +75,22 @@ class GroupDescendantsFinder
...
@@ -107,22 +75,22 @@ class GroupDescendantsFinder
def
all_visible_descendant_groups
def
all_visible_descendant_groups
groups_table
=
Group
.
arel_table
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
visible_for_user
=
if
current_user
groups_table
[
:id
].
in
(
visible_projects
=
GroupsFinder
.
new
(
current_user
).
execute
.
as
(
'visible'
)
Arel
::
Nodes
::
SqlLiteral
.
new
(
GroupsFinder
.
new
(
current_user
,
all_available:
true
).
execute
.
select
(
:id
).
to_sql
)
authorized
=
groups_table
.
project
(
1
).
from
(
visible_projects
)
)
.
where
(
visible_projects
[
:id
].
eq
(
groups_table
[
:id
]))
else
.
exists
groups_table
[
:visibility_level
].
eq
(
Gitlab
::
VisibilityLevel
::
PUBLIC
)
visible_for_user
.
or
(
authorized
)
end
end
hierarchy_for_parent
Gitlab
::
GroupHierarchy
.
new
(
Group
.
where
(
id:
parent_group
))
.
descendants
.
base_and_descendants
.
where
(
visible_for_user
)
.
where
(
visible_for_user
)
end
end
def
subgroups_matching_filter
def
subgroups_matching_filter
all_visible_descendant_groups
all_visible_descendant_groups
.
where
.
not
(
id:
parent_group
)
.
search
(
params
[
:filter
])
.
search
(
params
[
:filter
])
end
end
...
@@ -137,14 +105,15 @@ class GroupDescendantsFinder
...
@@ -137,14 +105,15 @@ class GroupDescendantsFinder
# So when searching 'project', on the 'subgroup' page we want to preload
# So when searching 'project', on the 'subgroup' page we want to preload
# 'nested-group' but not 'subgroup' or 'root'
# 'nested-group' but not 'subgroup' or 'root'
def
ancestors_for_groups
(
base_for_ancestors
)
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
)
Gitlab
::
GroupHierarchy
.
new
(
base_for_ancestors
)
.
base_and_ancestors
.
where
.
not
(
id:
ancestors_for_parent
)
.
base_and_ancestors
(
upto:
parent_group
.
id
)
end
end
def
ancestors_for_projects
(
projects
)
def
ancestors_for_projects
ancestors_for_groups
(
Group
.
where
(
id:
projects
.
select
(
:namespace_id
)))
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
end
def
subgroups
def
subgroups
...
@@ -169,9 +138,11 @@ class GroupDescendantsFinder
...
@@ -169,9 +138,11 @@ class GroupDescendantsFinder
projects_for_user
.
where
(
namespace:
parent_group
)
projects_for_user
.
where
(
namespace:
parent_group
)
end
end
# Finds all projects nested under `parent_group` or any of it's descendant
# groups
def
projects_matching_filter
def
projects_matching_filter
projects_for_user
.
search
(
params
[
: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
end
def
projects
def
projects
...
@@ -182,14 +153,14 @@ class GroupDescendantsFinder
...
@@ -182,14 +153,14 @@ class GroupDescendantsFinder
else
else
direct_child_projects
direct_child_projects
end
end
projects
.
order_by
(
sort
)
projects
.
with_route
.
order_by
(
sort
)
end
end
def
sort
def
sort
params
.
fetch
(
:sort
,
'id_asc'
)
params
.
fetch
(
:sort
,
'id_asc'
)
end
end
def
per_page
def
hierarchy_for_parent
params
.
fetch
(
:per_page
,
Kaminari
.
config
.
default_per_page
)
@hierarchy
||=
Gitlab
::
GroupHierarchy
.
new
(
Group
.
where
(
id:
parent_group
.
id
)
)
end
end
end
end
lib/gitlab/multi_collection_paginator.rb
0 → 100644
View file @
dda023d6
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
spec/lib/gitlab/multi_collection_paginator_spec.rb
0 → 100644
View file @
dda023d6
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
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment