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
Boxiang Sun
gitlab-ce
Commits
1efb2195
Commit
1efb2195
authored
Nov 30, 2017
by
Kamil Trzcinski
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Perform SQL matching of Build&Runner tags to greatly speed-up job picking
parent
844d95fb
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
120 additions
and
16 deletions
+120
-16
app/models/ci/build.rb
app/models/ci/build.rb
+19
-0
app/models/ci/runner.rb
app/models/ci/runner.rb
+3
-3
app/services/ci/register_job_service.rb
app/services/ci/register_job_service.rb
+10
-0
changelogs/unreleased/perform-sql-matching-of-tags.yml
changelogs/unreleased/perform-sql-matching-of-tags.yml
+5
-0
spec/models/ci/build_spec.rb
spec/models/ci/build_spec.rb
+73
-0
spec/services/ci/register_job_service_spec.rb
spec/services/ci/register_job_service_spec.rb
+10
-13
No files found.
app/models/ci/build.rb
View file @
1efb2195
...
@@ -47,6 +47,25 @@ module Ci
...
@@ -47,6 +47,25 @@ module Ci
scope
:manual_actions
,
->
()
{
where
(
when: :manual
,
status:
COMPLETED_STATUSES
+
[
:manual
])
}
scope
:manual_actions
,
->
()
{
where
(
when: :manual
,
status:
COMPLETED_STATUSES
+
[
:manual
])
}
scope
:ref_protected
,
->
{
where
(
protected:
true
)
}
scope
:ref_protected
,
->
{
where
(
protected:
true
)
}
scope
:matches_tag_ids
,
->
(
tag_ids
)
do
matcher
=
::
ActsAsTaggableOn
::
Tagging
.
where
(
taggable_type:
CommitStatus
)
.
where
(
context:
'tags'
)
.
where
(
'taggable_id = ci_builds.id'
)
.
where
.
not
(
tag_id:
tag_ids
).
select
(
'1'
)
where
(
"NOT EXISTS (?)"
,
matcher
)
end
scope
:with_any_tags
,
->
do
matcher
=
::
ActsAsTaggableOn
::
Tagging
.
where
(
taggable_type:
CommitStatus
)
.
where
(
context:
'tags'
)
.
where
(
'taggable_id = ci_builds.id'
).
select
(
'1'
)
where
(
"EXISTS (?)"
,
matcher
)
end
mount_uploader
:legacy_artifacts_file
,
LegacyArtifactUploader
,
mount_on: :artifacts_file
mount_uploader
:legacy_artifacts_file
,
LegacyArtifactUploader
,
mount_on: :artifacts_file
mount_uploader
:legacy_artifacts_metadata
,
LegacyArtifactUploader
,
mount_on: :artifacts_metadata
mount_uploader
:legacy_artifacts_metadata
,
LegacyArtifactUploader
,
mount_on: :artifacts_metadata
...
...
app/models/ci/runner.rb
View file @
1efb2195
...
@@ -112,7 +112,7 @@ module Ci
...
@@ -112,7 +112,7 @@ module Ci
def
can_pick?
(
build
)
def
can_pick?
(
build
)
return
false
if
self
.
ref_protected?
&&
!
build
.
protected?
return
false
if
self
.
ref_protected?
&&
!
build
.
protected?
assignable_for?
(
build
.
project
)
&&
accepting_tags?
(
build
)
assignable_for?
(
build
.
project
_id
)
&&
accepting_tags?
(
build
)
end
end
def
only_for?
(
project
)
def
only_for?
(
project
)
...
@@ -171,8 +171,8 @@ module Ci
...
@@ -171,8 +171,8 @@ module Ci
end
end
end
end
def
assignable_for?
(
project
)
def
assignable_for?
(
project
_id
)
is_shared?
||
projects
.
exists?
(
id:
project
.
id
)
is_shared?
||
projects
.
exists?
(
id:
project
_
id
)
end
end
def
accepting_tags?
(
build
)
def
accepting_tags?
(
build
)
...
...
app/services/ci/register_job_service.rb
View file @
1efb2195
...
@@ -22,6 +22,16 @@ module Ci
...
@@ -22,6 +22,16 @@ module Ci
valid
=
true
valid
=
true
if
Feature
.
enabled?
(
'ci_job_request_with_tags_matcher'
)
# pick builds that does not have other tags than runner's one
builds
=
builds
.
matches_tag_ids
(
runner
.
tags
.
ids
)
# pick builds that have at least one tag
unless
runner
.
run_untagged?
builds
=
builds
.
with_any_tags
end
end
builds
.
find
do
|
build
|
builds
.
find
do
|
build
|
next
unless
runner
.
can_pick?
(
build
)
next
unless
runner
.
can_pick?
(
build
)
...
...
changelogs/unreleased/perform-sql-matching-of-tags.yml
0 → 100644
View file @
1efb2195
---
title
:
Perform SQL matching of Build&Runner tags to greatly speed-up job picking
merge_request
:
author
:
type
:
performance
spec/models/ci/build_spec.rb
View file @
1efb2195
...
@@ -1921,4 +1921,77 @@ describe Ci::Build do
...
@@ -1921,4 +1921,77 @@ describe Ci::Build do
end
end
end
end
end
end
describe
'.matches_tag_ids'
do
set
(
:build
)
{
create
(
:ci_build
,
project:
project
,
user:
user
)
}
let
(
:tag_ids
)
{
::
ActsAsTaggableOn
::
Tag
.
named_any
(
tag_list
).
ids
}
subject
{
described_class
.
where
(
id:
build
).
matches_tag_ids
(
tag_ids
)
}
before
do
build
.
update
(
tag_list:
build_tag_list
)
end
context
'when have different tags'
do
let
(
:build_tag_list
)
{
%w(A B)
}
let
(
:tag_list
)
{
%w(C D)
}
it
"does not match a build"
do
is_expected
.
not_to
contain_exactly
(
build
)
end
end
context
'when have a subset of tags'
do
let
(
:build_tag_list
)
{
%w(A B)
}
let
(
:tag_list
)
{
%w(A B C D)
}
it
"does match a build"
do
is_expected
.
to
contain_exactly
(
build
)
end
end
context
'when build does not have tags'
do
let
(
:build_tag_list
)
{
[]
}
let
(
:tag_list
)
{
%w(C D)
}
it
"does match a build"
do
is_expected
.
to
contain_exactly
(
build
)
end
end
context
'when does not have a subset of tags'
do
let
(
:build_tag_list
)
{
%w(A B C)
}
let
(
:tag_list
)
{
%w(C D)
}
it
"does not match a build"
do
is_expected
.
not_to
contain_exactly
(
build
)
end
end
end
describe
'.matches_tags'
do
set
(
:build
)
{
create
(
:ci_build
,
project:
project
,
user:
user
)
}
subject
{
described_class
.
where
(
id:
build
).
with_any_tags
}
before
do
build
.
update
(
tag_list:
tag_list
)
end
context
'when does have tags'
do
let
(
:tag_list
)
{
%w(A B)
}
it
"does match a build"
do
is_expected
.
to
contain_exactly
(
build
)
end
end
context
'when does not have tags'
do
let
(
:tag_list
)
{
[]
}
it
"does not match a build"
do
is_expected
.
not_to
contain_exactly
(
build
)
end
end
end
end
end
spec/services/ci/register_job_service_spec.rb
View file @
1efb2195
...
@@ -15,16 +15,14 @@ module Ci
...
@@ -15,16 +15,14 @@ module Ci
describe
'#execute'
do
describe
'#execute'
do
context
'runner follow tag list'
do
context
'runner follow tag list'
do
it
"picks build with the same tag"
do
it
"picks build with the same tag"
do
pending_job
.
tag_list
=
[
"linux"
]
pending_job
.
update
(
tag_list:
[
"linux"
])
pending_job
.
save
specific_runner
.
update
(
tag_list:
[
"linux"
])
specific_runner
.
tag_list
=
[
"linux"
]
expect
(
execute
(
specific_runner
)).
to
eq
(
pending_job
)
expect
(
execute
(
specific_runner
)).
to
eq
(
pending_job
)
end
end
it
"does not pick build with different tag"
do
it
"does not pick build with different tag"
do
pending_job
.
tag_list
=
[
"linux"
]
pending_job
.
update
(
tag_list:
[
"linux"
])
pending_job
.
save
specific_runner
.
update
(
tag_list:
[
"win32"
])
specific_runner
.
tag_list
=
[
"win32"
]
expect
(
execute
(
specific_runner
)).
to
be_falsey
expect
(
execute
(
specific_runner
)).
to
be_falsey
end
end
...
@@ -33,13 +31,12 @@ module Ci
...
@@ -33,13 +31,12 @@ module Ci
end
end
it
"does not pick build with tag"
do
it
"does not pick build with tag"
do
pending_job
.
tag_list
=
[
"linux"
]
pending_job
.
update
(
tag_list:
[
"linux"
])
pending_job
.
save
expect
(
execute
(
specific_runner
)).
to
be_falsey
expect
(
execute
(
specific_runner
)).
to
be_falsey
end
end
it
"pick build without tag"
do
it
"pick build without tag"
do
specific_runner
.
tag_list
=
[
"win32"
]
specific_runner
.
update
(
tag_list:
[
"win32"
])
expect
(
execute
(
specific_runner
)).
to
eq
(
pending_job
)
expect
(
execute
(
specific_runner
)).
to
eq
(
pending_job
)
end
end
end
end
...
@@ -172,7 +169,7 @@ module Ci
...
@@ -172,7 +169,7 @@ module Ci
context
'when first build is stalled'
do
context
'when first build is stalled'
do
before
do
before
do
pending_job
.
lock_version
=
10
pending_job
.
update
(
lock_version:
0
)
end
end
subject
{
described_class
.
new
(
specific_runner
).
execute
}
subject
{
described_class
.
new
(
specific_runner
).
execute
}
...
@@ -182,7 +179,7 @@ module Ci
...
@@ -182,7 +179,7 @@ module Ci
before
do
before
do
allow_any_instance_of
(
Ci
::
RegisterJobService
).
to
receive
(
:builds_for_specific_runner
)
allow_any_instance_of
(
Ci
::
RegisterJobService
).
to
receive
(
:builds_for_specific_runner
)
.
and_return
(
[
pending_job
,
other_build
]
)
.
and_return
(
Ci
::
Build
.
where
(
id:
[
pending_job
,
other_build
])
)
end
end
it
"receives second build from the queue"
do
it
"receives second build from the queue"
do
...
@@ -194,7 +191,7 @@ module Ci
...
@@ -194,7 +191,7 @@ module Ci
context
'when single build is in queue'
do
context
'when single build is in queue'
do
before
do
before
do
allow_any_instance_of
(
Ci
::
RegisterJobService
).
to
receive
(
:builds_for_specific_runner
)
allow_any_instance_of
(
Ci
::
RegisterJobService
).
to
receive
(
:builds_for_specific_runner
)
.
and_return
(
[
pending_job
]
)
.
and_return
(
Ci
::
Build
.
where
(
id:
pending_job
)
)
end
end
it
"does not receive any valid result"
do
it
"does not receive any valid result"
do
...
@@ -205,7 +202,7 @@ module Ci
...
@@ -205,7 +202,7 @@ module Ci
context
'when there is no build in queue'
do
context
'when there is no build in queue'
do
before
do
before
do
allow_any_instance_of
(
Ci
::
RegisterJobService
).
to
receive
(
:builds_for_specific_runner
)
allow_any_instance_of
(
Ci
::
RegisterJobService
).
to
receive
(
:builds_for_specific_runner
)
.
and_return
(
[]
)
.
and_return
(
Ci
::
Build
.
none
)
end
end
it
"does not receive builds but result is valid"
do
it
"does not receive builds but result is valid"
do
...
...
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