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
76485cbf
Commit
76485cbf
authored
Apr 07, 2018
by
Shinya Maeda
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add ExclusiveLock in Ci::JobTraceChunk
parent
180267d6
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
63 additions
and
20 deletions
+63
-20
app/models/ci/job_trace_chunk.rb
app/models/ci/job_trace_chunk.rb
+52
-20
spec/models/ci/job_trace_chunk_spec.rb
spec/models/ci/job_trace_chunk_spec.rb
+11
-0
No files found.
app/models/ci/job_trace_chunk.rb
View file @
76485cbf
...
@@ -8,8 +8,13 @@ module Ci
...
@@ -8,8 +8,13 @@ module Ci
default_value_for
:data_store
,
:redis
default_value_for
:data_store
,
:redis
WriteError
=
Class
.
new
(
StandardError
)
CHUNK_SIZE
=
128
.
kilobytes
CHUNK_SIZE
=
128
.
kilobytes
CHUNK_REDIS_TTL
=
1
.
week
CHUNK_REDIS_TTL
=
1
.
week
LOCK_RETRY
=
100
LOCK_SLEEP
=
1
LOCK_TTL
=
5
.
minutes
enum
data_store:
{
enum
data_store:
{
redis:
1
,
redis:
1
,
...
@@ -27,18 +32,20 @@ module Ci
...
@@ -27,18 +32,20 @@ module Ci
end
end
def
set_data
(
value
)
def
set_data
(
value
)
raise
ArgumentError
,
'too much data'
if
value
.
bytesize
>
CHUNK_SIZE
in_lock
do
raise
ArgumentError
,
'too much data'
if
value
.
bytesize
>
CHUNK_SIZE
if
redis?
redis_set_data
(
value
)
if
redis?
elsif
db?
redis_set_data
(
value
)
self
.
raw_data
=
value
elsif
db?
else
self
.
raw_data
=
value
raise
'Unsupported data store'
else
raise
'Unsupported data store'
end
save!
if
changed?
schedule_to_db
if
fullfilled?
end
end
save!
if
changed?
schedule_to_db
if
fullfilled?
end
end
def
truncate
(
offset
=
0
)
def
truncate
(
offset
=
0
)
...
@@ -70,11 +77,13 @@ module Ci
...
@@ -70,11 +77,13 @@ module Ci
end
end
def
use_database!
def
use_database!
return
if
db?
in_lock
do
return
unless
size
>
0
return
if
db?
return
unless
size
>
0
self
.
update!
(
raw_data:
data
,
data_store: :db
)
self
.
update!
(
raw_data:
data
,
data_store: :db
)
redis_delete_data
redis_delete_data
end
end
end
private
private
...
@@ -91,24 +100,47 @@ module Ci
...
@@ -91,24 +100,47 @@ module Ci
def
redis_data
def
redis_data
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
get
(
redis_key
)
redis
.
get
(
redis_
data_
key
)
end
end
end
end
def
redis_set_data
(
data
)
def
redis_set_data
(
data
)
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
set
(
redis_key
,
data
,
ex:
CHUNK_REDIS_TTL
)
redis
.
set
(
redis_
data_
key
,
data
,
ex:
CHUNK_REDIS_TTL
)
end
end
end
end
def
redis_delete_data
def
redis_delete_data
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
del
(
redis_key
)
redis
.
del
(
redis_
data_
key
)
end
end
end
end
def
redis_key
def
redis_data_key
"gitlab:ci:trace:
#{
job_id
}
:chunks:
#{
chunk_index
}
"
"gitlab:ci:trace:
#{
job_id
}
:chunks:
#{
chunk_index
}
:data"
end
def
redis_lock_key
"gitlab:ci:trace:
#{
job_id
}
:chunks:
#{
chunk_index
}
:lock"
end
def
in_lock
lease
=
Gitlab
::
ExclusiveLease
.
new
(
redis_lock_key
,
timeout:
LOCK_TTL
)
retry_count
=
0
until
uuid
=
lease
.
try_obtain
# Keep trying until we obtain the lease. To prevent hammering Redis too
# much we'll wait for a bit between retries.
sleep
(
LOCK_SLEEP
)
break
if
LOCK_RETRY
<
(
retry_count
+=
1
)
end
raise
WriteError
,
'Failed to obtain write lock'
unless
uuid
self
.
reload
if
self
.
persisted?
return
yield
ensure
Gitlab
::
ExclusiveLease
.
cancel
(
redis_lock_key
,
uuid
)
end
end
end
end
end
end
spec/models/ci/job_trace_chunk_spec.rb
View file @
76485cbf
...
@@ -317,6 +317,17 @@ describe Ci::JobTraceChunk, :clean_gitlab_redis_shared_state do
...
@@ -317,6 +317,17 @@ describe Ci::JobTraceChunk, :clean_gitlab_redis_shared_state do
end
end
end
end
describe
'ExclusiveLock'
do
before
do
allow_any_instance_of
(
Gitlab
::
ExclusiveLease
).
to
receive
(
:try_obtain
)
{
nil
}
stub_const
(
'Ci::JobTraceChunk::LOCK_RETRY'
,
1
)
end
it
'raise an error'
do
expect
{
job_trace_chunk
.
append
(
'ABC'
,
0
)
}.
to
raise_error
(
'Failed to obtain write lock'
)
end
end
describe
'deletes data in redis after chunk record destroyed'
do
describe
'deletes data in redis after chunk record destroyed'
do
let
(
:project
)
{
create
(
:project
)
}
let
(
:project
)
{
create
(
:project
)
}
...
...
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