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
07554d47
Commit
07554d47
authored
Apr 09, 2021
by
Sarah Yasonik
Committed by
Alex Kalderimis
Apr 09, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Improve performance of on-call schedules SQL queries
parent
4a8a1ff1
Changes
14
Show whitespace changes
Inline
Side-by-side
Showing
14 changed files
with
314 additions
and
21 deletions
+314
-21
app/graphql/resolvers/concerns/looks_ahead.rb
app/graphql/resolvers/concerns/looks_ahead.rb
+9
-6
app/graphql/resolvers/group_members_resolver.rb
app/graphql/resolvers/group_members_resolver.rb
+0
-6
app/graphql/resolvers/members_resolver.rb
app/graphql/resolvers/members_resolver.rb
+6
-0
doc/development/api_graphql_styleguide.md
doc/development/api_graphql_styleguide.md
+20
-0
ee/app/graphql/ee/types/project_type.rb
ee/app/graphql/ee/types/project_type.rb
+1
-0
ee/app/graphql/resolvers/incident_management/oncall_schedule_resolver.rb
...resolvers/incident_management/oncall_schedule_resolver.rb
+38
-2
ee/app/graphql/types/incident_management/oncall_rotation_type.rb
...graphql/types/incident_management/oncall_rotation_type.rb
+1
-1
ee/app/graphql/types/incident_management/oncall_shift_type.rb
...pp/graphql/types/incident_management/oncall_shift_type.rb
+6
-2
ee/changelogs/unreleased/sy-resolve-oncall-n-plus-one.yml
ee/changelogs/unreleased/sy-resolve-oncall-n-plus-one.yml
+5
-0
ee/lib/gitlab/graphql/loaders/oncall_participant_loader.rb
ee/lib/gitlab/graphql/loaders/oncall_participant_loader.rb
+25
-0
ee/spec/graphql/types/incident_management/oncall_shift_type_spec.rb
...aphql/types/incident_management/oncall_shift_type_spec.rb
+0
-2
ee/spec/lib/gitlab/graphql/loaders/oncall_participant_loader_spec.rb
.../gitlab/graphql/loaders/oncall_participant_loader_spec.rb
+27
-0
ee/spec/requests/api/graphql/project/incident_management/oncall_participants_spec.rb
...l/project/incident_management/oncall_participants_spec.rb
+88
-0
ee/spec/requests/api/graphql/project/incident_management/oncall_shifts_spec.rb
...graphql/project/incident_management/oncall_shifts_spec.rb
+88
-2
No files found.
app/graphql/resolvers/concerns/looks_ahead.rb
View file @
07554d47
...
@@ -15,12 +15,7 @@ module LooksAhead
...
@@ -15,12 +15,7 @@ module LooksAhead
end
end
def
apply_lookahead
(
query
)
def
apply_lookahead
(
query
)
selection
=
node_selection
all_preloads
=
(
unconditional_includes
+
filtered_preloads
).
uniq
includes
=
preloads
.
each
.
flat_map
do
|
name
,
requirements
|
selection
&
.
selects?
(
name
)
?
requirements
:
[]
end
all_preloads
=
(
unconditional_includes
+
includes
).
uniq
return
query
if
all_preloads
.
empty?
return
query
if
all_preloads
.
empty?
...
@@ -37,6 +32,14 @@ module LooksAhead
...
@@ -37,6 +32,14 @@ module LooksAhead
{}
{}
end
end
def
filtered_preloads
selection
=
node_selection
preloads
.
each
.
flat_map
do
|
name
,
requirements
|
selection
&
.
selects?
(
name
)
?
requirements
:
[]
end
end
def
node_selection
def
node_selection
return
unless
lookahead
return
unless
lookahead
...
...
app/graphql/resolvers/group_members_resolver.rb
View file @
07554d47
...
@@ -13,12 +13,6 @@ module Resolvers
...
@@ -13,12 +13,6 @@ module Resolvers
private
private
def
preloads
{
user:
[
:user
,
:source
]
}
end
def
finder_class
def
finder_class
GroupMembersFinder
GroupMembersFinder
end
end
...
...
app/graphql/resolvers/members_resolver.rb
View file @
07554d47
...
@@ -21,6 +21,12 @@ module Resolvers
...
@@ -21,6 +21,12 @@ module Resolvers
private
private
def
preloads
{
user:
[
:user
,
:source
]
}
end
def
finder_class
def
finder_class
# override in subclass
# override in subclass
end
end
...
...
doc/development/api_graphql_styleguide.md
View file @
07554d47
...
@@ -1014,6 +1014,26 @@ class MyThingResolver < BaseResolver
...
@@ -1014,6 +1014,26 @@ class MyThingResolver < BaseResolver
end
end
```
```
By default, fields defined in
`#preloads`
will be preloaded if that field
is selected in the query. Occasionally, finer control may be
needed to avoid preloading too much or incorrect content.
Extending the above example, we might want to preload a different
association if certain fields are requested together. This can
be done by overriding
`#filtered_preloads`
:
```
ruby
class
MyThingResolver
<
BaseResolver
# ...
def
filtered_preloads
return
[
:alternate_attribute
]
if
lookahead
.
selects?
(
:field_one
)
&&
lookahead
.
selects?
(
:field_two
)
super
end
end
```
The final thing that is needed is that every field that uses this resolver needs
The final thing that is needed is that every field that uses this resolver needs
to advertise the need for lookahead:
to advertise the need for lookahead:
...
...
ee/app/graphql/ee/types/project_type.rb
View file @
07554d47
...
@@ -128,6 +128,7 @@ module EE
...
@@ -128,6 +128,7 @@ module EE
::
Types
::
IncidentManagement
::
OncallScheduleType
.
connection_type
,
::
Types
::
IncidentManagement
::
OncallScheduleType
.
connection_type
,
null:
true
,
null:
true
,
description:
'Incident Management On-call schedules of the project.'
,
description:
'Incident Management On-call schedules of the project.'
,
extras:
[
:lookahead
],
resolver:
::
Resolvers
::
IncidentManagement
::
OncallScheduleResolver
resolver:
::
Resolvers
::
IncidentManagement
::
OncallScheduleResolver
field
:api_fuzzing_ci_configuration
,
field
:api_fuzzing_ci_configuration
,
...
...
ee/app/graphql/resolvers/incident_management/oncall_schedule_resolver.rb
View file @
07554d47
...
@@ -3,12 +3,48 @@
...
@@ -3,12 +3,48 @@
module
Resolvers
module
Resolvers
module
IncidentManagement
module
IncidentManagement
class
OncallScheduleResolver
<
BaseResolver
class
OncallScheduleResolver
<
BaseResolver
extend
::
Gitlab
::
Utils
::
Override
include
LooksAhead
alias_method
:project
,
:object
alias_method
:project
,
:object
type
Types
::
IncidentManagement
::
OncallScheduleType
.
connection_type
,
null:
true
type
Types
::
IncidentManagement
::
OncallScheduleType
.
connection_type
,
null:
true
def
resolve
(
**
args
)
def
resolve_with_lookahead
(
**
args
)
::
IncidentManagement
::
OncallSchedulesFinder
.
new
(
context
[
:current_user
],
project
).
execute
apply_lookahead
(
::
IncidentManagement
::
OncallSchedulesFinder
.
new
(
context
[
:current_user
],
project
).
execute
)
end
private
# Tailor preloads to requested rotation fields instead of
# using LooksAhead#preloads to bulk-load all rotation associations
override
:filtered_preloads
def
filtered_preloads
rotation
=
rotation_selection
return
[]
unless
rotation
return
[{
rotations:
{
active_participants: :user
}
}]
if
rotation
.
selects?
(
:participants
)
return
[{
rotations: :active_participants
}]
if
will_generate_shifts?
(
rotation
)
[
:rotations
]
end
# @param rotation [GraphQL::Execution::Lookahead]
def
will_generate_shifts?
(
rotation
)
return
false
unless
rotation
.
selects?
(
:shifts
)
rotation
.
selection
(
:shifts
).
arguments
[
:end_time
]
>
Time
.
current
end
def
rotation_selection
rotations
=
node_selection
&
.
selection
(
:rotations
)
return
unless
rotations
&
.
selected?
if
rotations
.
selects?
(
:nodes
)
rotations
.
selection
(
:nodes
)
elsif
rotations
.
selects?
(
:edges
)
rotations
.
selection
(
:edges
).
selection
(
:node
)
end
end
end
end
end
end
end
...
...
ee/app/graphql/types/incident_management/oncall_rotation_type.rb
View file @
07554d47
...
@@ -58,7 +58,7 @@ module Types
...
@@ -58,7 +58,7 @@ module Types
resolver:
::
Resolvers
::
IncidentManagement
::
OncallShiftsResolver
resolver:
::
Resolvers
::
IncidentManagement
::
OncallShiftsResolver
def
participants
def
participants
object
.
participants
.
not_removed
object
.
active_participants
end
end
end
end
end
end
...
...
ee/app/graphql/types/incident_management/oncall_shift_type.rb
View file @
07554d47
...
@@ -2,12 +2,11 @@
...
@@ -2,12 +2,11 @@
module
Types
module
Types
module
IncidentManagement
module
IncidentManagement
# rubocop: disable Graphql/AuthorizeTypes
class
OncallShiftType
<
BaseObject
class
OncallShiftType
<
BaseObject
graphql_name
'IncidentManagementOncallShift'
graphql_name
'IncidentManagementOncallShift'
description
'A block of time for which a participant is on-call.'
description
'A block of time for which a participant is on-call.'
authorize
:read_incident_management_oncall_schedule
field
:participant
,
field
:participant
,
::
Types
::
IncidentManagement
::
OncallParticipantType
,
::
Types
::
IncidentManagement
::
OncallParticipantType
,
null:
true
,
null:
true
,
...
@@ -22,6 +21,11 @@ module Types
...
@@ -22,6 +21,11 @@ module Types
Types
::
TimeType
,
Types
::
TimeType
,
null:
true
,
null:
true
,
description:
'End time of the on-call shift.'
description:
'End time of the on-call shift.'
def
participant
Gitlab
::
Graphql
::
Loaders
::
OncallParticipantLoader
.
new
(
object
.
participant_id
).
find
end
end
end
# rubocop: enable Graphql/AuthorizeTypes
end
end
end
end
ee/changelogs/unreleased/sy-resolve-oncall-n-plus-one.yml
0 → 100644
View file @
07554d47
---
title
:
Improve performance of on-call schedules SQL queries
merge_request
:
58209
author
:
type
:
performance
ee/lib/gitlab/graphql/loaders/oncall_participant_loader.rb
0 → 100644
View file @
07554d47
# frozen_string_literal: true
module
Gitlab
module
Graphql
module
Loaders
class
OncallParticipantLoader
attr_reader
:participant_id
def
initialize
(
participant_id
)
@participant_id
=
participant_id
end
# rubocop: disable CodeReuse/ActiveRecord
def
find
BatchLoader
::
GraphQL
.
for
(
participant_id
.
to_i
).
batch
do
|
ids
,
loader
|
results
=
::
IncidentManagement
::
OncallParticipant
.
includes
(
:user
).
id_in
(
ids
)
results
.
each
{
|
participant
|
loader
.
call
(
participant
.
id
,
participant
)
}
end
end
# rubocop: enable CodeReuse/ActiveRecord
end
end
end
end
ee/spec/graphql/types/incident_management/oncall_shift_type_spec.rb
View file @
07554d47
...
@@ -5,8 +5,6 @@ require 'spec_helper'
...
@@ -5,8 +5,6 @@ require 'spec_helper'
RSpec
.
describe
GitlabSchema
.
types
[
'IncidentManagementOncallShift'
]
do
RSpec
.
describe
GitlabSchema
.
types
[
'IncidentManagementOncallShift'
]
do
specify
{
expect
(
described_class
.
graphql_name
).
to
eq
(
'IncidentManagementOncallShift'
)
}
specify
{
expect
(
described_class
.
graphql_name
).
to
eq
(
'IncidentManagementOncallShift'
)
}
specify
{
expect
(
described_class
).
to
require_graphql_authorizations
(
:read_incident_management_oncall_schedule
)
}
it
'exposes the expected fields'
do
it
'exposes the expected fields'
do
expected_fields
=
%i[
expected_fields
=
%i[
participant
participant
...
...
ee/spec/lib/gitlab/graphql/loaders/oncall_participant_loader_spec.rb
0 → 100644
View file @
07554d47
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
Gitlab
::
Graphql
::
Loaders
::
OncallParticipantLoader
do
describe
'#find'
do
let_it_be
(
:participant1
)
{
create
(
:incident_management_oncall_participant
)
}
let_it_be
(
:participant2
)
{
create
(
:incident_management_oncall_participant
)
}
let_it_be
(
:participant3
)
{
create
(
:incident_management_oncall_participant
)
}
it
'finds a participant by id'
do
first_result
=
described_class
.
new
(
participant1
.
id
).
find
second_result
=
described_class
.
new
(
participant2
.
id
).
find
expect
(
first_result
.
sync
).
to
eq
(
participant1
)
expect
(
second_result
.
sync
).
to
eq
(
participant2
)
end
it
'includes the user association'
do
expect
do
[
described_class
.
new
(
participant3
.
id
).
find
,
described_class
.
new
(
participant2
.
id
).
find
,
described_class
.
new
(
participant1
.
id
).
find
].
map
(
&
:sync
).
map
(
&
:user
)
end
.
not_to
exceed_query_limit
(
2
)
end
end
end
ee/spec/requests/api/graphql/project/incident_management/oncall_participants_spec.rb
0 → 100644
View file @
07554d47
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
'getting Incident Management on-call shifts'
do
include
GraphqlHelpers
let_it_be
(
:participant
)
{
create
(
:incident_management_oncall_participant
,
:utc
,
:with_developer_access
)
}
let_it_be
(
:rotation
)
{
participant
.
rotation
}
let_it_be
(
:project
)
{
rotation
.
project
}
let_it_be
(
:current_user
)
{
participant
.
user
}
let
(
:fields
)
do
<<~
QUERY
nodes {
rotations {
nodes {
participants {
nodes {
id
colorPalette
colorWeight
user { id }
}
}
}
}
}
QUERY
end
let
(
:query
)
do
graphql_query_for
(
'project'
,
{
'fullPath'
=>
project
.
full_path
},
query_graphql_field
(
'incidentManagementOncallSchedules'
,
{},
fields
)
)
end
let
(
:participants
)
do
graphql_data
.
dig
(
'project'
,
'incidentManagementOncallSchedules'
,
'nodes'
).
first
.
dig
(
'rotations'
,
'nodes'
).
first
.
dig
(
'participants'
,
'nodes'
)
end
before
do
stub_licensed_features
(
oncall_schedules:
true
)
post_graphql
(
query
,
current_user:
current_user
)
end
it_behaves_like
'a working graphql query'
it
'returns the correct properties of the on-call shifts'
do
expect
(
participants
.
first
).
to
include
(
'id'
=>
participant
.
to_global_id
.
to_s
,
'user'
=>
{
'id'
=>
participant
.
user
.
to_global_id
.
to_s
},
'colorWeight'
=>
'50'
,
'colorPalette'
=>
'blue'
)
end
context
'performance'
do
shared_examples
'avoids N+1 queries'
do
specify
do
base_count
=
ActiveRecord
::
QueryRecorder
.
new
do
post_graphql
(
query
,
current_user:
current_user
)
end
action
expect
{
post_graphql
(
query
,
current_user:
current_user
)
}.
not_to
exceed_query_limit
(
base_count
)
end
end
context
'for additional participant'
do
let
(
:action
)
{
create
(
:incident_management_oncall_participant
,
rotation:
rotation
)
}
it_behaves_like
'avoids N+1 queries'
end
context
'for additional rotation with participants'
do
let
(
:action
)
{
create
(
:incident_management_oncall_rotation
,
:with_participants
,
schedule:
rotation
.
schedule
)
}
it_behaves_like
'avoids N+1 queries'
end
end
end
ee/spec/requests/api/graphql/project/incident_management/oncall_shifts_spec.rb
View file @
07554d47
...
@@ -17,7 +17,7 @@ RSpec.describe 'getting Incident Management on-call shifts' do
...
@@ -17,7 +17,7 @@ RSpec.describe 'getting Incident Management on-call shifts' do
let
(
:shift_fields
)
do
let
(
:shift_fields
)
do
<<~
QUERY
<<~
QUERY
nodes {
nodes {
participant { id }
participant { id
user { id }
}
endsAt
endsAt
startsAt
startsAt
}
}
...
@@ -60,12 +60,98 @@ RSpec.describe 'getting Incident Management on-call shifts' do
...
@@ -60,12 +60,98 @@ RSpec.describe 'getting Incident Management on-call shifts' do
it
'returns the correct properties of the on-call shifts'
do
it
'returns the correct properties of the on-call shifts'
do
expect
(
shifts
.
first
).
to
include
(
expect
(
shifts
.
first
).
to
include
(
'participant'
=>
{
'id'
=>
participant
.
to_global_id
.
to_s
},
'participant'
=>
{
'id'
=>
participant
.
to_global_id
.
to_s
,
'user'
=>
{
'id'
=>
participant
.
user
.
to_global_id
.
to_s
}
},
'startsAt'
=>
params
[
:start_time
],
'startsAt'
=>
params
[
:start_time
],
'endsAt'
=>
params
[
:end_time
]
'endsAt'
=>
params
[
:end_time
]
)
)
end
end
context
'performance'
do
shared_examples
'avoids N+1 queries'
do
specify
do
base_count
=
ActiveRecord
::
QueryRecorder
.
new
do
post_graphql
(
query
,
current_user:
current_user
)
end
action
expect
{
post_graphql
(
query
,
current_user:
current_user
)
}.
not_to
exceed_query_limit
(
base_count
)
end
end
shared_examples
'avoids N+1 queries for additional generated shift'
do
include_examples
'avoids N+1 queries'
do
let
(
:action
)
{
params
[
:end_time
]
=
ends_at
.
next_day
.
iso8601
}
end
end
shared_examples
'avoids N+1 queries for additional historical shift'
do
include_examples
'avoids N+1 queries'
do
let
(
:action
)
{
create
(
:incident_management_oncall_shift
,
participant:
participant
,
starts_at:
last_shift
.
ends_at
)
}
end
end
shared_examples
'avoids N+1 queries for additional participant'
do
include_examples
'avoids N+1 queries'
do
let
(
:action
)
{
create
(
:incident_management_oncall_participant
,
rotation:
rotation
)
}
end
end
shared_examples
'avoids N+1 queries for additional rotation with participants'
do
include_examples
'avoids N+1 queries'
do
let
(
:action
)
{
create
(
:incident_management_oncall_rotation
,
:with_participants
,
schedule:
rotation
.
schedule
)
}
end
end
shared_examples
'adds only one query for each additional rotation with participants'
do
specify
do
base_count
=
ActiveRecord
::
QueryRecorder
.
new
do
post_graphql
(
query
,
current_user:
current_user
)
end
create
(
:incident_management_oncall_rotation
,
:with_participants
,
schedule:
rotation
.
schedule
)
create
(
:incident_management_oncall_rotation
,
:with_participants
,
schedule:
rotation
.
schedule
)
expect
{
post_graphql
(
query
,
current_user:
current_user
)
}.
not_to
exceed_query_limit
(
base_count
).
with_threshold
(
2
)
end
end
context
'for past and future shifts'
do
let_it_be
(
:last_shift
)
{
create
(
:incident_management_oncall_shift
,
participant:
participant
)
}
let
(
:ends_at
)
{
rotation
.
starts_at
+
2
*
rotation
.
shift_cycle_duration
}
it_behaves_like
'avoids N+1 queries for additional generated shift'
it_behaves_like
'avoids N+1 queries for additional historical shift'
it_behaves_like
'avoids N+1 queries for additional participant'
it_behaves_like
'adds only one query for each additional rotation with participants'
end
context
'for future shifts only'
do
let
(
:starts_at
)
{
rotation
.
starts_at
+
rotation
.
shift_cycle_duration
}
let
(
:ends_at
)
{
rotation
.
starts_at
+
2
*
rotation
.
shift_cycle_duration
}
it_behaves_like
'avoids N+1 queries for additional generated shift'
it_behaves_like
'avoids N+1 queries for additional participant'
it_behaves_like
'avoids N+1 queries for additional rotation with participants'
end
context
'for past shifts only'
do
let_it_be
(
:last_shift
)
{
create
(
:incident_management_oncall_shift
,
participant:
participant
)
}
around
do
|
example
|
travel_to
(
starts_at
+
1.5
*
rotation
.
shift_cycle_duration
)
{
example
.
run
}
end
it_behaves_like
'avoids N+1 queries for additional historical shift'
it_behaves_like
'avoids N+1 queries for additional participant'
it_behaves_like
'adds only one query for each additional rotation with participants'
end
end
context
"without required argument starts_at"
do
context
"without required argument starts_at"
do
let
(
:params
)
{
{
end_time:
ends_at
.
iso8601
}
}
let
(
:params
)
{
{
end_time:
ends_at
.
iso8601
}
}
...
...
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