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
6e8d0b78
Commit
6e8d0b78
authored
Aug 15, 2017
by
Nick Thomas
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use event-based waiting in Gitlab::JobWaiter
parent
6509833c
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
94 additions
and
33 deletions
+94
-33
app/workers/authorized_projects_worker.rb
app/workers/authorized_projects_worker.rb
+10
-3
lib/gitlab/job_waiter.rb
lib/gitlab/job_waiter.rb
+48
-9
spec/lib/gitlab/job_waiter_spec.rb
spec/lib/gitlab/job_waiter_spec.rb
+25
-16
spec/workers/authorized_projects_worker_spec.rb
spec/workers/authorized_projects_worker_spec.rb
+11
-5
No files found.
app/workers/authorized_projects_worker.rb
View file @
6e8d0b78
...
@@ -4,18 +4,25 @@ class AuthorizedProjectsWorker
...
@@ -4,18 +4,25 @@ class AuthorizedProjectsWorker
# Schedules multiple jobs and waits for them to be completed.
# Schedules multiple jobs and waits for them to be completed.
def
self
.
bulk_perform_and_wait
(
args_list
)
def
self
.
bulk_perform_and_wait
(
args_list
)
job_ids
=
bulk_perform_async
(
args_list
)
waiter
=
Gitlab
::
JobWaiter
.
new
(
args_list
.
size
)
Gitlab
::
JobWaiter
.
new
(
job_ids
).
wait
# Point all the bulk jobs at the same JobWaiter. Converts, [[1], [2], [3]]
# into [[1, "key"], [2, "key"], [3, "key"]]
waiting_args_list
=
args_list
.
map
{
|
args
|
args
<<
waiter
.
key
}
bulk_perform_async
(
waiting_args_list
)
waiter
.
wait
end
end
def
self
.
bulk_perform_async
(
args_list
)
def
self
.
bulk_perform_async
(
args_list
)
Sidekiq
::
Client
.
push_bulk
(
'class'
=>
self
,
'queue'
=>
sidekiq_options
[
'queue'
],
'args'
=>
args_list
)
Sidekiq
::
Client
.
push_bulk
(
'class'
=>
self
,
'queue'
=>
sidekiq_options
[
'queue'
],
'args'
=>
args_list
)
end
end
def
perform
(
user_id
)
def
perform
(
user_id
,
notify_key
=
nil
)
user
=
User
.
find_by
(
id:
user_id
)
user
=
User
.
find_by
(
id:
user_id
)
user
&
.
refresh_authorized_projects
user
&
.
refresh_authorized_projects
ensure
Gitlab
::
JobWaiter
.
notify
(
notify_key
,
jid
)
if
notify_key
end
end
end
end
lib/gitlab/job_waiter.rb
View file @
6e8d0b78
module
Gitlab
module
Gitlab
# JobWaiter can be used to wait for a number of Sidekiq jobs to complete.
# JobWaiter can be used to wait for a number of Sidekiq jobs to complete.
#
# Its use requires the cooperation of the sidekiq jobs themselves. Set up the
# waiter, then start the jobs, passing them its `key`. Their `perform` methods
# should look like:
#
# def perform(args, notify_key)
# # do work
# ensure
# ::Gitlab::JobWaiter.notify(notify_key, jid)
# end
#
# The JobWaiter blocks popping items from a Redis array. All the sidekiq jobs
# push to that array when done. Once the waiter has popped `count` items, it
# knows all the jobs are done.
class
JobWaiter
class
JobWaiter
# The sleep interval between checking keys, in seconds.
def
self
.
notify
(
key
,
jid
)
INTERVAL
=
0.1
Gitlab
::
Redis
::
SharedState
.
with
{
|
redis
|
redis
.
lpush
(
key
,
jid
)
}
end
attr_reader
:key
,
:jobs_remaining
,
:finished
# jobs - The job IDs to wait for.
# jobs_remaining - the number of jobs left to wait for
def
initialize
(
jobs
)
def
initialize
(
jobs_remaining
)
@jobs
=
jobs
@key
=
"gitlab:job_waiter:
#{
SecureRandom
.
uuid
}
"
@jobs_remaining
=
jobs_remaining
@finished
=
[]
end
end
# Waits for all the jobs to be completed.
# Waits for all the jobs to be completed.
...
@@ -15,13 +34,33 @@ module Gitlab
...
@@ -15,13 +34,33 @@ module Gitlab
# ensures we don't indefinitely block a caller in case a job takes
# ensures we don't indefinitely block a caller in case a job takes
# long to process, or is never processed.
# long to process, or is never processed.
def
wait
(
timeout
=
10
)
def
wait
(
timeout
=
10
)
start
=
Time
.
current
deadline
=
Time
.
now
.
utc
+
timeout
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
# Fallback key expiry: allow a long grace period to reduce the chance of
# a job pushing to an expired key and recreating it
redis
.
expire
(
key
,
[
timeout
*
2
,
10
.
minutes
.
to_i
].
max
)
while
jobs_remaining
>
0
# Redis will not take fractional seconds. Prefer waiting too long over
# not waiting long enough
seconds_left
=
(
deadline
-
Time
.
now
.
utc
).
ceil
while
(
Time
.
current
-
start
)
<=
timeout
# Redis interprets 0 as "wait forever", so skip the final `blpop` call
break
if
SidekiqStatus
.
all_completed?
(
@jobs
)
break
if
seconds_left
<=
0
sleep
(
INTERVAL
)
# to not overload Redis too much.
list
,
jid
=
redis
.
blpop
(
key
,
timeout:
seconds_left
)
break
unless
list
&&
jid
# timed out
@finished
<<
jid
@jobs_remaining
-=
1
end
# All jobs have finished, so expire the key immediately
redis
.
expire
(
key
,
0
)
if
jobs_remaining
==
0
end
end
finished
end
end
end
end
end
end
spec/lib/gitlab/job_waiter_spec.rb
View file @
6e8d0b78
require
'spec_helper'
require
'spec_helper'
describe
Gitlab
::
JobWaiter
do
describe
Gitlab
::
JobWaiter
do
describe
'#wait'
do
describe
'.notify'
do
let
(
:waiter
)
{
described_class
.
new
(
%w(a)
)
}
it
'pushes the jid to the named queue'
do
it
'returns when all jobs have been completed'
do
key
=
'gitlab:job_waiter:foo'
expect
(
Gitlab
::
SidekiqStatus
).
to
receive
(
:all_completed?
).
with
(
%w(a)
)
jid
=
1
.
and_return
(
true
)
expect
(
waiter
).
not_to
receive
(
:sleep
)
redis
=
double
(
'redis'
)
expect
(
Gitlab
::
Redis
::
SharedState
).
to
receive
(
:with
).
and_yield
(
redis
)
expect
(
redis
).
to
receive
(
:lpush
).
with
(
key
,
jid
)
waiter
.
wait
described_class
.
notify
(
key
,
jid
)
end
end
end
describe
'#wait'
do
let
(
:waiter
)
{
described_class
.
new
(
2
)
}
it
'sleeps between checking the job statuses'
do
it
'returns when all jobs have been completed'
do
expect
(
Gitlab
::
SidekiqStatus
).
to
receive
(
:all_completed?
)
described_class
.
notify
(
waiter
.
key
,
'a'
)
.
with
(
%w(a)
)
described_class
.
notify
(
waiter
.
key
,
'b'
)
.
and_return
(
false
,
true
)
expect
(
waiter
).
to
receive
(
:sleep
).
with
(
described_class
::
INTERVAL
)
result
=
nil
expect
{
Timeout
.
timeout
(
1
)
{
result
=
waiter
.
wait
(
2
)
}
}.
not_to
raise_error
waiter
.
wait
expect
(
result
).
to
contain_exactly
(
'a'
,
'b'
)
end
end
it
'returns when timing out'
do
it
'times out if not all jobs complete'
do
expect
(
waiter
).
not_to
receive
(
:sleep
)
described_class
.
notify
(
waiter
.
key
,
'a'
)
waiter
.
wait
(
0
)
result
=
nil
expect
{
Timeout
.
timeout
(
2
)
{
result
=
waiter
.
wait
(
1
)
}
}.
not_to
raise_error
expect
(
result
).
to
contain_exactly
(
'a'
)
end
end
end
end
end
end
spec/workers/authorized_projects_worker_spec.rb
View file @
6e8d0b78
...
@@ -29,21 +29,27 @@ describe AuthorizedProjectsWorker do
...
@@ -29,21 +29,27 @@ describe AuthorizedProjectsWorker do
end
end
describe
'#perform'
do
describe
'#perform'
do
subject
{
described_class
.
new
}
let
(
:user
)
{
create
(
:user
)
}
it
"refreshes user's authorized projects"
do
subject
(
:job
)
{
described_class
.
new
}
user
=
create
(
:user
)
it
"refreshes user's authorized projects"
do
expect_any_instance_of
(
User
).
to
receive
(
:refresh_authorized_projects
)
expect_any_instance_of
(
User
).
to
receive
(
:refresh_authorized_projects
)
subject
.
perform
(
user
.
id
)
job
.
perform
(
user
.
id
)
end
it
'notifies the JobWaiter when done if the key is provided'
do
expect
(
Gitlab
::
JobWaiter
).
to
receive
(
:notify
).
with
(
'notify-key'
,
job
.
jid
)
job
.
perform
(
user
.
id
,
'notify-key'
)
end
end
context
"when the user is not found"
do
context
"when the user is not found"
do
it
"does nothing"
do
it
"does nothing"
do
expect_any_instance_of
(
User
).
not_to
receive
(
:refresh_authorized_projects
)
expect_any_instance_of
(
User
).
not_to
receive
(
:refresh_authorized_projects
)
subject
.
perform
(
-
1
)
job
.
perform
(
-
1
)
end
end
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