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
ac9d4190
Commit
ac9d4190
authored
Dec 24, 2019
by
GitLab Bot
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add latest changes from gitlab-org/gitlab@master
parent
c6373a2c
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
68 additions
and
24 deletions
+68
-24
.gitlab/ci/rails.gitlab-ci.yml
.gitlab/ci/rails.gitlab-ci.yml
+16
-4
app/models/ci/pipeline.rb
app/models/ci/pipeline.rb
+3
-3
app/serializers/pipeline_entity.rb
app/serializers/pipeline_entity.rb
+1
-1
app/serializers/pipeline_serializer.rb
app/serializers/pipeline_serializer.rb
+5
-0
changelogs/unreleased/sh-optimize-pipeline-sql.yml
changelogs/unreleased/sh-optimize-pipeline-sql.yml
+5
-0
spec/controllers/projects/pipelines_controller_spec.rb
spec/controllers/projects/pipelines_controller_spec.rb
+34
-14
spec/lib/gitlab/import_export/all_models.yml
spec/lib/gitlab/import_export/all_models.yml
+2
-0
spec/serializers/pipeline_serializer_spec.rb
spec/serializers/pipeline_serializer_spec.rb
+2
-2
No files found.
.gitlab/ci/rails.gitlab-ci.yml
View file @
ac9d4190
...
@@ -57,7 +57,7 @@ setup-test-env:
...
@@ -57,7 +57,7 @@ setup-test-env:
dependencies
:
[
"
setup-test-env"
,
"
retrieve-tests-metadata"
,
"
compile-assets
pull-cache"
]
dependencies
:
[
"
setup-test-env"
,
"
retrieve-tests-metadata"
,
"
compile-assets
pull-cache"
]
script
:
script
:
-
source scripts/rspec_helpers.sh
-
source scripts/rspec_helpers.sh
-
rspec_paralellized_job "--tag ~quarantine --tag ~geo"
-
rspec_paralellized_job "--tag ~quarantine --tag ~geo
--tag ~level:migration
"
artifacts
:
artifacts
:
expire_in
:
31d
expire_in
:
31d
when
:
always
when
:
always
...
@@ -92,12 +92,21 @@ setup-test-env:
...
@@ -92,12 +92,21 @@ setup-test-env:
-
.use-pg10
-
.use-pg10
-
.only-master
-
.only-master
.rspec-base-migration
:
script
:
-
source scripts/rspec_helpers.sh
-
rspec_paralellized_job "--tag ~quarantine --tag ~geo --tag level:migration"
rspec migration pg9
:
rspec migration pg9
:
extends
:
.rspec-base-pg9
extends
:
-
.rspec-base-pg9
-
.rspec-base-migration
parallel
:
4
parallel
:
4
rspec migration pg9-foss
:
rspec migration pg9-foss
:
extends
:
.rspec-base-pg9-foss
extends
:
-
.rspec-base-pg9-foss
-
.rspec-base-migration
parallel
:
4
parallel
:
4
rspec unit pg9
:
rspec unit pg9
:
...
@@ -149,7 +158,9 @@ rspec system pg10:
...
@@ -149,7 +158,9 @@ rspec system pg10:
-
.use-pg10-ee
-
.use-pg10-ee
rspec-ee migration pg9
:
rspec-ee migration pg9
:
extends
:
.rspec-ee-base-pg9
extends
:
-
.rspec-ee-base-pg9
-
.rspec-base-migration
parallel
:
2
parallel
:
2
rspec-ee unit pg9
:
rspec-ee unit pg9
:
...
@@ -167,6 +178,7 @@ rspec-ee system pg9:
...
@@ -167,6 +178,7 @@ rspec-ee system pg9:
rspec-ee migration pg10
:
rspec-ee migration pg10
:
extends
:
extends
:
-
.rspec-ee-base-pg10
-
.rspec-ee-base-pg10
-
.rspec-base-migration
-
.only-master
-
.only-master
parallel
:
2
parallel
:
2
...
...
app/models/ci/pipeline.rb
View file @
ac9d4190
...
@@ -32,6 +32,7 @@ module Ci
...
@@ -32,6 +32,7 @@ module Ci
has_many
:stages
,
->
{
order
(
position: :asc
)
},
inverse_of: :pipeline
has_many
:stages
,
->
{
order
(
position: :asc
)
},
inverse_of: :pipeline
has_many
:statuses
,
class_name:
'CommitStatus'
,
foreign_key: :commit_id
,
inverse_of: :pipeline
has_many
:statuses
,
class_name:
'CommitStatus'
,
foreign_key: :commit_id
,
inverse_of: :pipeline
has_many
:latest_statuses_ordered_by_stage
,
->
{
latest
.
order
(
:stage_idx
,
:stage
)
},
class_name:
'CommitStatus'
,
foreign_key: :commit_id
,
inverse_of: :pipeline
has_many
:processables
,
->
{
processables
},
has_many
:processables
,
->
{
processables
},
class_name:
'CommitStatus'
,
foreign_key: :commit_id
,
inverse_of: :pipeline
class_name:
'CommitStatus'
,
foreign_key: :commit_id
,
inverse_of: :pipeline
has_many
:builds
,
foreign_key: :commit_id
,
inverse_of: :pipeline
has_many
:builds
,
foreign_key: :commit_id
,
inverse_of: :pipeline
...
@@ -45,6 +46,7 @@ module Ci
...
@@ -45,6 +46,7 @@ module Ci
has_many
:merge_requests_as_head_pipeline
,
foreign_key:
"head_pipeline_id"
,
class_name:
'MergeRequest'
has_many
:merge_requests_as_head_pipeline
,
foreign_key:
"head_pipeline_id"
,
class_name:
'MergeRequest'
has_many
:pending_builds
,
->
{
pending
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
has_many
:pending_builds
,
->
{
pending
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
has_many
:failed_builds
,
->
{
latest
.
failed
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
,
inverse_of: :pipeline
has_many
:retryable_builds
,
->
{
latest
.
failed_or_canceled
.
includes
(
:project
)
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
has_many
:retryable_builds
,
->
{
latest
.
failed_or_canceled
.
includes
(
:project
)
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
has_many
:cancelable_statuses
,
->
{
cancelable
},
foreign_key: :commit_id
,
class_name:
'CommitStatus'
has_many
:cancelable_statuses
,
->
{
cancelable
},
foreign_key: :commit_id
,
class_name:
'CommitStatus'
has_many
:manual_actions
,
->
{
latest
.
manual_actions
.
includes
(
:project
)
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
has_many
:manual_actions
,
->
{
latest
.
manual_actions
.
includes
(
:project
)
},
foreign_key: :commit_id
,
class_name:
'Ci::Build'
...
@@ -383,9 +385,7 @@ module Ci
...
@@ -383,9 +385,7 @@ module Ci
end
end
def
legacy_stages_using_composite_status
def
legacy_stages_using_composite_status
stages
=
statuses
.
latest
stages
=
latest_statuses_ordered_by_stage
.
group_by
(
&
:stage
)
.
order
(
:stage_idx
,
:stage
)
.
group_by
(
&
:stage
)
stages
.
map
do
|
stage_name
,
jobs
|
stages
.
map
do
|
stage_name
,
jobs
|
composite_status
=
Gitlab
::
Ci
::
Status
::
Composite
composite_status
=
Gitlab
::
Ci
::
Status
::
Composite
...
...
app/serializers/pipeline_entity.rb
View file @
ac9d4190
...
@@ -78,7 +78,7 @@ class PipelineEntity < Grape::Entity
...
@@ -78,7 +78,7 @@ class PipelineEntity < Grape::Entity
end
end
expose
:failed_builds
,
if:
->
(
*
)
{
can_retry?
},
using:
JobEntity
do
|
pipeline
|
expose
:failed_builds
,
if:
->
(
*
)
{
can_retry?
},
using:
JobEntity
do
|
pipeline
|
pipeline
.
builds
.
failed
pipeline
.
failed_builds
end
end
private
private
...
...
app/serializers/pipeline_serializer.rb
View file @
ac9d4190
...
@@ -40,7 +40,11 @@ class PipelineSerializer < BaseSerializer
...
@@ -40,7 +40,11 @@ class PipelineSerializer < BaseSerializer
def
preloaded_relations
def
preloaded_relations
[
[
:latest_statuses_ordered_by_stage
,
:stages
,
:stages
,
{
failed_builds:
%i(project metadata)
},
:retryable_builds
,
:retryable_builds
,
:cancelable_statuses
,
:cancelable_statuses
,
:trigger_requests
,
:trigger_requests
,
...
@@ -48,6 +52,7 @@ class PipelineSerializer < BaseSerializer
...
@@ -48,6 +52,7 @@ class PipelineSerializer < BaseSerializer
:scheduled_actions
,
:scheduled_actions
,
:artifacts
,
:artifacts
,
:merge_request
,
:merge_request
,
:user
,
{
{
pending_builds: :project
,
pending_builds: :project
,
project:
[
:route
,
{
namespace: :route
}],
project:
[
:route
,
{
namespace: :route
}],
...
...
changelogs/unreleased/sh-optimize-pipeline-sql.yml
0 → 100644
View file @
ac9d4190
---
title
:
Eliminate N+1 queries in PipelinesController#index
merge_request
:
22189
author
:
type
:
performance
spec/controllers/projects/pipelines_controller_spec.rb
View file @
ac9d4190
...
@@ -6,7 +6,7 @@ describe Projects::PipelinesController do
...
@@ -6,7 +6,7 @@ describe Projects::PipelinesController do
include
ApiHelpers
include
ApiHelpers
let_it_be
(
:user
)
{
create
(
:user
)
}
let_it_be
(
:user
)
{
create
(
:user
)
}
let
(
:project
)
{
create
(
:project
,
:public
,
:repository
)
}
let
_it_be
(
:project
)
{
create
(
:project
,
:public
,
:repository
)
}
let
(
:feature
)
{
ProjectFeature
::
ENABLED
}
let
(
:feature
)
{
ProjectFeature
::
ENABLED
}
before
do
before
do
...
@@ -19,12 +19,12 @@ describe Projects::PipelinesController do
...
@@ -19,12 +19,12 @@ describe Projects::PipelinesController do
describe
'GET index.json'
do
describe
'GET index.json'
do
before
do
before
do
%w(pending running success failed canceled)
.
each_with_index
do
|
status
,
index
|
create_all_pipeline_types
create_pipeline
(
status
,
project
.
commit
(
"HEAD~
#{
index
}
"
))
end
end
end
context
'when using persisted stages'
,
:request_store
do
context
'when using persisted stages'
,
:request_store
do
render_views
before
do
before
do
stub_feature_flags
(
ci_pipeline_persisted_stages:
true
)
stub_feature_flags
(
ci_pipeline_persisted_stages:
true
)
end
end
...
@@ -32,9 +32,7 @@ describe Projects::PipelinesController do
...
@@ -32,9 +32,7 @@ describe Projects::PipelinesController do
it
'returns serialized pipelines'
,
:request_store
do
it
'returns serialized pipelines'
,
:request_store
do
expect
(
::
Gitlab
::
GitalyClient
).
to
receive
(
:allow_ref_name_caching
).
and_call_original
expect
(
::
Gitlab
::
GitalyClient
).
to
receive
(
:allow_ref_name_caching
).
and_call_original
queries
=
ActiveRecord
::
QueryRecorder
.
new
do
get_pipelines_index_json
get_pipelines_index_json
end
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
).
to
match_response_schema
(
'pipeline'
)
expect
(
response
).
to
match_response_schema
(
'pipeline'
)
...
@@ -49,8 +47,22 @@ describe Projects::PipelinesController do
...
@@ -49,8 +47,22 @@ describe Projects::PipelinesController do
json_response
.
dig
(
'pipelines'
,
0
,
'details'
,
'stages'
).
tap
do
|
stages
|
json_response
.
dig
(
'pipelines'
,
0
,
'details'
,
'stages'
).
tap
do
|
stages
|
expect
(
stages
.
count
).
to
eq
3
expect
(
stages
.
count
).
to
eq
3
end
end
end
it
'does not execute N+1 queries'
do
get_pipelines_index_json
control_count
=
ActiveRecord
::
QueryRecorder
.
new
do
get_pipelines_index_json
end
.
count
expect
(
queries
.
count
).
to
be
create_all_pipeline_types
# There appears to be one extra query for Pipelines#has_warnings? for some reason
expect
{
get_pipelines_index_json
}.
not_to
exceed_query_limit
(
control_count
+
1
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
json_response
[
'pipelines'
].
count
).
to
eq
10
end
end
end
end
...
@@ -133,19 +145,27 @@ describe Projects::PipelinesController do
...
@@ -133,19 +145,27 @@ describe Projects::PipelinesController do
format: :json
format: :json
end
end
def
create_all_pipeline_types
%w(pending running success failed canceled)
.
each_with_index
do
|
status
,
index
|
create_pipeline
(
status
,
project
.
commit
(
"HEAD~
#{
index
}
"
))
end
end
def
create_pipeline
(
status
,
sha
)
def
create_pipeline
(
status
,
sha
)
user
=
create
(
:user
)
pipeline
=
create
(
:ci_empty_pipeline
,
status:
status
,
pipeline
=
create
(
:ci_empty_pipeline
,
status:
status
,
project:
project
,
project:
project
,
sha:
sha
)
sha:
sha
,
user:
user
)
create_build
(
pipeline
,
'build'
,
1
,
'build'
)
create_build
(
pipeline
,
'build'
,
1
,
'build'
,
user
)
create_build
(
pipeline
,
'test'
,
2
,
'test'
)
create_build
(
pipeline
,
'test'
,
2
,
'test'
,
user
)
create_build
(
pipeline
,
'deploy'
,
3
,
'deploy'
)
create_build
(
pipeline
,
'deploy'
,
3
,
'deploy'
,
user
)
end
end
def
create_build
(
pipeline
,
stage
,
stage_idx
,
name
)
def
create_build
(
pipeline
,
stage
,
stage_idx
,
name
,
user
=
nil
)
status
=
%w[created running pending success failed canceled]
.
sample
status
=
%w[created running pending success failed canceled]
.
sample
create
(
:ci_build
,
pipeline:
pipeline
,
stage:
stage
,
stage_idx:
stage_idx
,
name:
name
,
status:
status
)
create
(
:ci_build
,
pipeline:
pipeline
,
stage:
stage
,
stage_idx:
stage_idx
,
name:
name
,
status:
status
,
user:
user
)
end
end
end
end
...
...
spec/lib/gitlab/import_export/all_models.yml
View file @
ac9d4190
...
@@ -160,6 +160,7 @@ ci_pipelines:
...
@@ -160,6 +160,7 @@ ci_pipelines:
-
user
-
user
-
stages
-
stages
-
statuses
-
statuses
-
latest_statuses_ordered_by_stage
-
builds
-
builds
-
processables
-
processables
-
trigger_requests
-
trigger_requests
...
@@ -168,6 +169,7 @@ ci_pipelines:
...
@@ -168,6 +169,7 @@ ci_pipelines:
-
auto_canceled_pipelines
-
auto_canceled_pipelines
-
auto_canceled_jobs
-
auto_canceled_jobs
-
pending_builds
-
pending_builds
-
failed_builds
-
retryable_builds
-
retryable_builds
-
cancelable_statuses
-
cancelable_statuses
-
manual_actions
-
manual_actions
...
...
spec/serializers/pipeline_serializer_spec.rb
View file @
ac9d4190
...
@@ -159,7 +159,7 @@ describe PipelineSerializer do
...
@@ -159,7 +159,7 @@ describe PipelineSerializer do
it
'verifies number of queries'
,
:request_store
do
it
'verifies number of queries'
,
:request_store
do
recorded
=
ActiveRecord
::
QueryRecorder
.
new
{
subject
}
recorded
=
ActiveRecord
::
QueryRecorder
.
new
{
subject
}
expected_queries
=
Gitlab
.
ee?
?
38
:
35
expected_queries
=
Gitlab
.
ee?
?
42
:
39
expect
(
recorded
.
count
).
to
be_within
(
2
).
of
(
expected_queries
)
expect
(
recorded
.
count
).
to
be_within
(
2
).
of
(
expected_queries
)
expect
(
recorded
.
cached_count
).
to
eq
(
0
)
expect
(
recorded
.
cached_count
).
to
eq
(
0
)
...
@@ -180,7 +180,7 @@ describe PipelineSerializer do
...
@@ -180,7 +180,7 @@ describe PipelineSerializer do
# pipeline. With the same ref this check is cached but if refs are
# pipeline. With the same ref this check is cached but if refs are
# different then there is an extra query per ref
# different then there is an extra query per ref
# https://gitlab.com/gitlab-org/gitlab-foss/issues/46368
# https://gitlab.com/gitlab-org/gitlab-foss/issues/46368
expected_queries
=
Gitlab
.
ee?
?
4
1
:
38
expected_queries
=
Gitlab
.
ee?
?
4
5
:
42
expect
(
recorded
.
count
).
to
be_within
(
2
).
of
(
expected_queries
)
expect
(
recorded
.
count
).
to
be_within
(
2
).
of
(
expected_queries
)
expect
(
recorded
.
cached_count
).
to
eq
(
0
)
expect
(
recorded
.
cached_count
).
to
eq
(
0
)
...
...
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