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
8d1cdc94
Commit
8d1cdc94
authored
Feb 07, 2017
by
Tomasz Maczukin
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Update stuck and outdated builds cleanup worker
parent
e5bafed8
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
101 additions
and
35 deletions
+101
-35
app/workers/stuck_ci_builds_worker.rb
app/workers/stuck_ci_builds_worker.rb
+31
-7
spec/workers/stuck_ci_builds_worker_spec.rb
spec/workers/stuck_ci_builds_worker_spec.rb
+70
-28
No files found.
app/workers/stuck_ci_builds_worker.rb
View file @
8d1cdc94
...
@@ -2,18 +2,42 @@ class StuckCiBuildsWorker
...
@@ -2,18 +2,42 @@ class StuckCiBuildsWorker
include
Sidekiq
::
Worker
include
Sidekiq
::
Worker
include
CronjobQueue
include
CronjobQueue
BUILD_STUCK_TIMEOUT
=
1
.
day
BUILD_RUNNING_OUTDATED_TIMEOUT
=
1
.
hour
BUILD_PENDING_OUTDATED_TIMEOUT
=
1
.
day
BUILD_PENDING_STUCK_TIMEOUT
=
1
.
hour
def
perform
def
perform
Rails
.
logger
.
info
'Cleaning stuck builds'
Rails
.
logger
.
info
'Cleaning stuck builds'
builds
=
Ci
::
Build
.
joins
(
:project
).
running_or_pending
.
where
(
'ci_builds.updated_at < ?'
,
BUILD_STUCK_TIMEOUT
.
ago
)
drop
:running
,
BUILD_RUNNING_OUTDATED_TIMEOUT
builds
.
find_each
(
batch_size:
50
).
each
do
|
build
|
drop
:pending
,
BUILD_PENDING_OUTDATED_TIMEOUT
Rails
.
logger
.
debug
"Dropping stuck
#{
build
.
status
}
build
#{
build
.
id
}
for runner
#{
build
.
runner_id
}
"
drop_stuck
:pending
,
BUILD_PENDING_STUCK_TIMEOUT
build
.
drop
end
private
def
drop
(
status
,
timeout
)
search
(
status
,
timeout
)
do
|
build
|
drop_build
:outdated
,
build
,
status
,
timeout
end
end
def
drop_stuck
(
status
,
timeout
)
search
(
status
,
timeout
)
do
|
build
|
return
unless
build
.
stuck?
drop_build
:stuck
,
build
,
status
,
timeout
end
end
def
search
(
status
,
timeout
)
builds
=
Ci
::
Build
.
where
(
status:
status
).
where
(
'ci_builds.updated_at < ?'
,
timeout
.
ago
)
builds
.
joins
(
:project
).
find_each
(
batch_size:
50
).
each
do
|
build
|
yield
(
build
)
end
end
end
# Update builds that failed to drop
def
drop_build
(
type
,
build
,
status
,
timeout
)
builds
.
update_all
(
status:
'failed'
)
Rails
.
logger
.
info
"
#{
self
.
class
}
: Dropping
#{
type
.
to_s
}
build
#{
build
.
id
}
for runner
#{
build
.
runner_id
}
(status:
#{
status
}
, timeout:
#{
timeout
}
)"
build
.
drop
end
end
end
end
spec/workers/stuck_ci_builds_worker_spec.rb
View file @
8d1cdc94
require
"spec_helper"
require
'spec_helper'
describe
StuckCiBuildsWorker
do
describe
StuckCiBuildsWorker
do
let!
(
:build
)
{
create
:ci_build
}
let!
(
:runner
)
{
create
:ci_runner
}
let!
(
:build
)
{
create
:ci_build
,
runner:
runner
}
let
(
:worker
)
{
described_class
.
new
}
let
(
:worker
)
{
described_class
.
new
}
subject
do
subject
do
...
@@ -9,47 +10,88 @@ describe StuckCiBuildsWorker do
...
@@ -9,47 +10,88 @@ describe StuckCiBuildsWorker do
build
.
status
build
.
status
end
end
%w(pending running)
.
each
do
|
status
|
before
{
build
.
update!
(
status:
status
,
updated_at:
updated_at
)
}
context
"
#{
status
}
build"
do
before
do
build
.
update!
(
status:
status
)
end
it
'gets dropped if it was updated over 2 days ago
'
do
shared_examples
'build is dropped
'
do
build
.
update!
(
updated_at:
2
.
days
.
ago
)
it
'changes status'
do
worker
.
perform
worker
.
perform
is_expected
.
to
eq
(
'failed'
)
is_expected
.
to
eq
(
'failed'
)
end
end
end
it
"is still
#{
status
}
"
do
shared_examples
'build is unchanged'
do
build
.
update!
(
updated_at:
1
.
minute
.
ago
)
it
"doesn't change status"
do
worker
.
perform
worker
.
perform
is_expected
.
to
eq
(
status
)
is_expected
.
to
eq
(
status
)
end
end
end
end
context
'when build is pending'
do
let
(
:status
)
{
'pending'
}
context
'when build is not stuck'
do
before
{
allow_any_instance_of
(
Ci
::
Build
).
to
receive
(
:stuck?
).
and_return
(
false
)
}
context
'when build was not updated for more than 1 day ago'
do
let
(
:updated_at
)
{
2
.
days
.
ago
}
it_behaves_like
'build is dropped'
end
end
%w(success failed canceled)
.
each
do
|
status
|
context
'when build was updated in less than 1 day ago'
do
context
"
#{
status
}
build"
do
let
(
:updated_at
)
{
6
.
hours
.
ago
}
before
do
it_behaves_like
'build is unchanged'
build
.
update!
(
status:
status
)
end
end
it
"is still
#{
status
}
"
do
context
'when build was not updated for more than 1 hour ago'
do
build
.
update!
(
updated_at:
2
.
days
.
ago
)
let
(
:updated_at
)
{
2
.
hours
.
ago
}
worker
.
perform
it_behaves_like
'build is unchanged'
is_expected
.
to
eq
(
status
)
end
end
context
'when build is stuck'
do
before
{
allow_any_instance_of
(
Ci
::
Build
).
to
receive
(
:stuck?
).
and_return
(
true
)
}
context
'when build was not updated for more than 1 hour ago'
do
let
(
:updated_at
)
{
2
.
hours
.
ago
}
it_behaves_like
'build is dropped'
end
context
'when build was updated in less than 1 hour ago'
do
let
(
:updated_at
)
{
30
.
minutes
.
ago
}
it_behaves_like
'build is unchanged'
end
end
end
end
end
context
'when build is running'
do
let
(
:status
)
{
'running'
}
context
'when build was not updated for more than 1 hour ago'
do
let
(
:updated_at
)
{
2
.
hours
.
ago
}
it_behaves_like
'build is dropped'
end
end
context
"for deleted project"
do
context
'when build was updated in less than 1 hour ago'
do
before
do
let
(
:updated_at
)
{
30
.
minutes
.
ago
}
build
.
update!
(
status: :running
,
updated_at:
2
.
days
.
ago
)
it_behaves_like
'build is unchanged'
build
.
project
.
update
(
pending_delete:
true
)
end
end
end
it
"does not drop build"
do
%w(success skipped failed canceled)
.
each
do
|
status
|
context
"when build is
#{
status
}
"
do
let
(
:status
)
{
status
}
let
(
:updated_at
)
{
2
.
days
.
ago
}
it_behaves_like
'build is unchanged'
end
end
context
'for deleted project'
do
let
(
:status
)
{
'running'
}
let
(
:updated_at
)
{
2
.
days
.
ago
}
before
{
build
.
project
.
update
(
pending_delete:
true
)
}
it
'does not drop build'
do
expect_any_instance_of
(
Ci
::
Build
).
not_to
receive
(
:drop
)
expect_any_instance_of
(
Ci
::
Build
).
not_to
receive
(
:drop
)
worker
.
perform
worker
.
perform
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