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
1
Merge Requests
1
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
nexedi
gitlab-ce
Commits
3f5d0012
Commit
3f5d0012
authored
Apr 07, 2021
by
Terri Chu
Committed by
Heinrich Lee Yu
Apr 07, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Reduce queries on Controller Projects::LabelsController#index [RUN AS-IF-FOSS] [RUN ALL RSPEC]
parent
6af841ba
Changes
8
Show whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
136 additions
and
7 deletions
+136
-7
app/controllers/groups/labels_controller.rb
app/controllers/groups/labels_controller.rb
+2
-2
app/controllers/projects/labels_controller.rb
app/controllers/projects/labels_controller.rb
+7
-4
app/models/preloaders/labels_preloader.rb
app/models/preloaders/labels_preloader.rb
+32
-0
changelogs/unreleased/21078-controller-projects-labelscontroller-index-executes-more-than-100-s.yml
...jects-labelscontroller-index-executes-more-than-100-s.yml
+5
-0
ee/app/models/ee/preloaders/labels_preloader.rb
ee/app/models/ee/preloaders/labels_preloader.rb
+17
-0
spec/controllers/groups/labels_controller_spec.rb
spec/controllers/groups/labels_controller_spec.rb
+1
-1
spec/controllers/projects/labels_controller_spec.rb
spec/controllers/projects/labels_controller_spec.rb
+20
-0
spec/models/preloaders/labels_preloader_spec.rb
spec/models/preloaders/labels_preloader_spec.rb
+52
-0
No files found.
app/controllers/groups/labels_controller.rb
View file @
3f5d0012
...
@@ -16,8 +16,8 @@ class Groups::LabelsController < Groups::ApplicationController
...
@@ -16,8 +16,8 @@ class Groups::LabelsController < Groups::ApplicationController
format
.
html
do
format
.
html
do
# at group level we do not want to list project labels,
# at group level we do not want to list project labels,
# we only want `only_group_labels = false` when pulling labels for label filter dropdowns, fetched through json
# we only want `only_group_labels = false` when pulling labels for label filter dropdowns, fetched through json
@labels
=
available_labels
(
params
.
merge
(
only_group_labels:
true
)).
includes
(
:group
).
page
(
params
[
:page
])
# rubocop: disable CodeReuse/ActiveRecord
@labels
=
available_labels
(
params
.
merge
(
only_group_labels:
true
)).
page
(
params
[
:page
])
# rubocop: disable CodeReuse/ActiveRecord
@labels
.
each
{
|
label
|
label
.
lazy_subscription
(
current_user
)
}
# preload subscriptions
Preloaders
::
LabelsPreloader
.
new
(
@labels
,
current_user
).
preload_all
end
end
format
.
json
do
format
.
json
do
render
json:
LabelSerializer
.
new
.
represent_appearance
(
available_labels
)
render
json:
LabelSerializer
.
new
.
represent_appearance
(
available_labels
)
...
...
app/controllers/projects/labels_controller.rb
View file @
3f5d0012
...
@@ -17,11 +17,14 @@ class Projects::LabelsController < Projects::ApplicationController
...
@@ -17,11 +17,14 @@ class Projects::LabelsController < Projects::ApplicationController
feature_category
:issue_tracking
feature_category
:issue_tracking
def
index
def
index
respond_to
do
|
format
|
format
.
html
do
@prioritized_labels
=
@available_labels
.
prioritized
(
@project
)
@prioritized_labels
=
@available_labels
.
prioritized
(
@project
)
@labels
=
@available_labels
.
unprioritized
(
@project
).
page
(
params
[
:page
])
@labels
=
@available_labels
.
unprioritized
(
@project
).
page
(
params
[
:page
])
# preload group, project, and subscription data
respond_to
do
|
format
|
Preloaders
::
LabelsPreloader
.
new
(
@prioritized_labels
,
current_user
,
@project
).
preload_all
format
.
html
Preloaders
::
LabelsPreloader
.
new
(
@labels
,
current_user
,
@project
).
preload_all
end
format
.
json
do
format
.
json
do
render
json:
LabelSerializer
.
new
.
represent_appearance
(
@available_labels
)
render
json:
LabelSerializer
.
new
.
represent_appearance
(
@available_labels
)
end
end
...
...
app/models/preloaders/labels_preloader.rb
0 → 100644
View file @
3f5d0012
# frozen_string_literal: true
module
Preloaders
# This class preloads the `project`, `group`, and subscription associations for the given
# labels, user, and project (if provided). A Label can be of type ProjectLabel or GroupLabel
# and the preloader supports both.
#
# Usage:
# labels = Label.where(...)
# Preloaders::LabelsPreloader.new(labels, current_user, @project).preload_all
# labels.first.project # won't fire any query
class
LabelsPreloader
attr_reader
:labels
,
:user
,
:project
def
initialize
(
labels
,
user
,
project
=
nil
)
@labels
,
@user
,
@project
=
labels
,
user
,
project
end
def
preload_all
preloader
=
ActiveRecord
::
Associations
::
Preloader
.
new
preloader
.
preload
(
labels
.
select
{
|
l
|
l
.
is_a?
ProjectLabel
},
{
project:
[
:project_feature
,
namespace: :route
]
})
preloader
.
preload
(
labels
.
select
{
|
l
|
l
.
is_a?
GroupLabel
},
{
group: :route
})
labels
.
each
do
|
label
|
label
.
lazy_subscription
(
user
)
label
.
lazy_subscription
(
user
,
project
)
if
project
.
present?
end
end
end
end
Preloaders
::
LabelsPreloader
.
prepend_if_ee
(
'EE::Preloaders::LabelsPreloader'
)
changelogs/unreleased/21078-controller-projects-labelscontroller-index-executes-more-than-100-s.yml
0 → 100644
View file @
3f5d0012
---
title
:
Reduce queries on projects labels controller
merge_request
:
57864
author
:
type
:
performance
ee/app/models/ee/preloaders/labels_preloader.rb
0 → 100644
View file @
3f5d0012
# frozen_string_literal: true
module
EE
module
Preloaders
module
LabelsPreloader
extend
::
Gitlab
::
Utils
::
Override
override
:preload_all
def
preload_all
super
preloader
=
ActiveRecord
::
Associations
::
Preloader
.
new
preloader
.
preload
(
labels
.
select
{
|
l
|
l
.
is_a?
GroupLabel
},
{
group:
[
:ip_restrictions
,
:saml_provider
]
})
end
end
end
end
spec/controllers/groups/labels_controller_spec.rb
View file @
3f5d0012
...
@@ -60,7 +60,7 @@ RSpec.describe Groups::LabelsController do
...
@@ -60,7 +60,7 @@ RSpec.describe Groups::LabelsController do
create_list
(
:group_label
,
3
,
group:
group
)
create_list
(
:group_label
,
3
,
group:
group
)
# some n+1 queries still exist
# some n+1 queries still exist
expect
{
get
:index
,
params:
{
group_id:
group
.
to_param
}
}.
not_to
exceed_all_query_limit
(
control
.
count
).
with_threshold
(
1
2
)
expect
{
get
:index
,
params:
{
group_id:
group
.
to_param
}
}.
not_to
exceed_all_query_limit
(
control
.
count
).
with_threshold
(
1
0
)
expect
(
assigns
(
:labels
).
count
).
to
eq
(
4
)
expect
(
assigns
(
:labels
).
count
).
to
eq
(
4
)
end
end
end
end
...
...
spec/controllers/projects/labels_controller_spec.rb
View file @
3f5d0012
...
@@ -93,6 +93,26 @@ RSpec.describe Projects::LabelsController do
...
@@ -93,6 +93,26 @@ RSpec.describe Projects::LabelsController do
end
end
end
end
context
'with views rendered'
do
render_views
before
do
list_labels
end
it
'avoids N+1 queries'
do
control
=
ActiveRecord
::
QueryRecorder
.
new
(
skip_cached:
false
)
{
list_labels
}
create_list
(
:label
,
3
,
project:
project
)
create_list
(
:group_label
,
3
,
group:
group
)
# some n+1 queries still exist
# calls to get max project authorization access level
expect
{
list_labels
}.
not_to
exceed_all_query_limit
(
control
.
count
).
with_threshold
(
25
)
expect
(
assigns
(
:labels
).
count
).
to
eq
(
10
)
end
end
def
list_labels
def
list_labels
get
:index
,
params:
{
namespace_id:
project
.
namespace
.
to_param
,
project_id:
project
}
get
:index
,
params:
{
namespace_id:
project
.
namespace
.
to_param
,
project_id:
project
}
end
end
...
...
spec/models/preloaders/labels_preloader_spec.rb
0 → 100644
View file @
3f5d0012
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
Preloaders
::
LabelsPreloader
do
let_it_be
(
:user
)
{
create
(
:user
)
}
shared_examples
'an efficient database query'
do
let
(
:subscriptions
)
{
labels
.
each
{
|
l
|
create
(
:subscription
,
subscribable:
l
,
project:
l
.
project
,
user:
user
)
}}
it
'does not make n+1 queries'
do
first_label
=
labels_with_preloaded_data
.
first
clean_labels
=
labels_with_preloaded_data
expect
{
access_data
(
clean_labels
)
}.
to
issue_same_number_of_queries_as
{
access_data
([
first_label
])
}
end
end
context
'project labels'
do
let_it_be
(
:projects
)
{
create_list
(
:project
,
3
,
:public
,
:repository
)
}
let_it_be
(
:labels
)
{
projects
.
each
{
|
p
|
create
(
:label
,
project:
p
)
}
}
it_behaves_like
'an efficient database query'
end
context
'group labels'
do
let_it_be
(
:groups
)
{
create_list
(
:group
,
3
)
}
let_it_be
(
:labels
)
{
groups
.
each
{
|
g
|
create
(
:group_label
,
group:
g
)
}
}
it_behaves_like
'an efficient database query'
end
private
def
labels_with_preloaded_data
l
=
Label
.
where
(
id:
labels
.
map
(
&
:id
))
described_class
.
new
(
l
,
user
).
preload_all
l
end
def
access_data
(
labels
)
labels
.
each
do
|
label
|
if
label
.
is_a?
(
ProjectLabel
)
label
.
project
.
project_feature
label
.
lazy_subscription
(
user
,
label
.
project
)
elsif
label
.
is_a?
(
GroupLabel
)
label
.
group
.
route
label
.
lazy_subscription
(
user
)
end
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