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
1d53918b
Commit
1d53918b
authored
May 01, 2018
by
Shinya Maeda
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Introduces `FastDestroyAll` module
parent
b9cc1188
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
89 additions
and
36 deletions
+89
-36
app/models/ci/build_trace_chunk.rb
app/models/ci/build_trace_chunk.rb
+18
-24
app/models/concerns/fast_destroy_all.rb
app/models/concerns/fast_destroy_all.rb
+61
-0
app/models/project.rb
app/models/project.rb
+6
-8
app/workers/build_finished_worker.rb
app/workers/build_finished_worker.rb
+1
-1
lib/gitlab/ci/trace/chunked_io.rb
lib/gitlab/ci/trace/chunked_io.rb
+1
-1
spec/models/ci/build_trace_chunk_spec.rb
spec/models/ci/build_trace_chunk_spec.rb
+2
-2
No files found.
app/models/ci/build_trace_chunk.rb
View file @
1d53918b
module
Ci
class
BuildTraceChunk
<
ActiveRecord
::
Base
include
FastDestroyAll
extend
Gitlab
::
Ci
::
Model
belongs_to
:build
,
class_name:
"Ci::Build"
,
foreign_key: :build_id
default_value_for
:data_store
,
:redis
fast_destroy_all_with
:delete_all_redis_data
,
:redis_all_data_keys
WriteError
=
Class
.
new
(
StandardError
)
...
...
@@ -19,27 +21,23 @@ module Ci
db:
2
}
def
self
.
delayed_cleanup_blk
ids
=
all
.
redis
.
pluck
(
:build_id
,
:chunk_index
).
map
do
|
data
|
"gitlab:ci:trace:
#{
data
.
first
}
:chunks:
#{
data
.
second
}
"
class
<<
self
def
redis_data_key
(
build_id
,
chunk_index
)
"gitlab:ci:trace:
#{
build_id
}
:chunks:
#{
chunk_index
}
"
end
puts
"before cleanup:
#{
ids
.
count
}
"
Proc
.
new
do
puts
"after cleanup:
#{
ids
.
count
}
"
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
del
(
ids
)
end
unless
ids
.
empty?
true
def
redis_all_data_keys
redis
.
pluck
(
:build_id
,
:chunk_index
).
map
do
|
data
|
redis_data_key
(
data
.
first
,
data
.
second
)
end
end
end
def
self
.
fast_destroy_all
delayed_cleanup_blk
.
tap
do
|
cleanup
|
delete_all
cleanup
.
call
def
delete_all_redis_data
(
redis_keys
)
if
redis_keys
.
any?
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
del
(
redis_keys
)
end
end
end
end
...
...
@@ -130,26 +128,22 @@ module Ci
def
redis_data
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
get
(
redis_data_key
)
redis
.
get
(
self
.
class
.
redis_data_key
(
build_id
,
chunk_index
)
)
end
end
def
redis_set_data
(
data
)
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
set
(
redis_data_key
,
data
,
ex:
CHUNK_REDIS_TTL
)
redis
.
set
(
self
.
class
.
redis_data_key
(
build_id
,
chunk_index
)
,
data
,
ex:
CHUNK_REDIS_TTL
)
end
end
def
redis_delete_data
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
redis
.
del
(
redis_data_key
)
redis
.
del
(
self
.
class
.
redis_data_key
(
build_id
,
chunk_index
)
)
end
end
def
redis_data_key
"gitlab:ci:trace:
#{
build_id
}
:chunks:
#{
chunk_index
}
"
end
def
redis_lock_key
"trace_write:
#{
build_id
}
:chunks:
#{
chunk_index
}
"
end
...
...
app/models/concerns/fast_destroy_all.rb
0 → 100644
View file @
1d53918b
##
# This module is for replacing `dependent: :destroy` and `before_destroy` hooks.
#
# In general, `destroy_all` is inefficient because it calls each callback with `DELETE` queries i.e. O(n), whereas,
# `delete_all` is efficient as it deletes all rows with a single `DELETE` query.
#
# It's better to use `delete_all` as much as possible, however, in the real cases, it's hard to adopt because
# in some cases, external data (in ObjectStorage/FileStorage/Redis) is assosiated with rows.
#
# This module introduces a protocol to support the adoption with easy way.
# You can use in the following scnenes
# - When calling `destroy_all` explicitly
# e.g. `job.trace_chunks.fast_destroy_all``
# - When a parent record is deleted and all children are deleted subsequently (cascade delete)
# e.g. `before_destroy -> { run_after_commit(&build_trace_chunks.delete_all_external_data_proc) }``
module
FastDestroyAll
extend
ActiveSupport
::
Concern
included
do
class_attribute
:_delete_method
,
:_delete_params_generator
end
class_methods
do
##
# This method is for registering :delete_method and :delete_params_generator
# You have to define the method if you want to use `FastDestroyAll`` module.
#
# e.g. fast_destroy_all_with :delete_all_redis_data, :redis_all_data_keys
def
fast_destroy_all_with
(
delete_method
,
delete_params_generator
)
self
.
_delete_method
=
delete_method
self
.
_delete_params_generator
=
delete_params_generator
end
##
# This method generates a proc to delete external data.
# It's useful especially when you want to hook parent record's deletion event.
#
# e.g. before_destroy -> { run_after_commit(&build_trace_chunks.delete_all_external_data_proc) }
def
delete_all_external_data_proc
params
=
send
self
.
_delete_params_generator
callee
=
self
# Preserve the callee class, otherwise `proc` uses a different class
proc
do
callee
.
send
callee
.
_delete_method
,
params
true
end
end
##
# This method deletes all rows at first and delete all external data at second.
# Before deleting the rows, it generates a proc to delete external data.
# So it won't lose the track of deleting external data, even if it happened after rows had been deleted.
def
fast_destroy_all
delete_all_external_data_proc
.
tap
do
|
delete_all_external_data
|
delete_all
delete_all_external_data
.
call
end
end
end
end
app/models/project.rb
View file @
1d53918b
...
...
@@ -78,6 +78,12 @@ class Project < ActiveRecord::Base
after_update
:update_forks_visibility_level
before_destroy
:remove_private_deploy_keys
# This has to be defined before `has_many :builds, depenedent: :destroy`,
# otherwise we will not delete any data, due to trace chunks
# going through :builds
before_destroy
->
{
run_after_commit
(
&
build_trace_chunks
.
delete_all_external_data_proc
)
}
after_destroy
->
{
run_after_commit
{
remove_pages
}
}
after_destroy
:remove_exports
...
...
@@ -214,14 +220,6 @@ class Project < ActiveRecord::Base
has_many
:commit_statuses
has_many
:pipelines
,
class_name:
'Ci::Pipeline'
,
inverse_of: :project
# This has to be defined before `has_many :builds, depenedent: :destroy`,
# otherwise we will not delete any data, due to trace chunks
# going through :builds
before_destroy
do
puts
"destroying all chunks"
self
.
run_after_commit
(
&
build_trace_chunks
.
delayed_cleanup_blk
)
end
# Ci::Build objects store data on the file system such as artifact files and
# build traces. Currently there's no efficient way of removing this data in
# bulk that doesn't involve loading the rows into memory. As a result we're
...
...
app/workers/build_finished_worker.rb
View file @
1d53918b
...
...
@@ -6,7 +6,7 @@ class BuildFinishedWorker
def
perform
(
build_id
)
Ci
::
Build
.
find_by
(
id:
build_id
).
try
do
|
build
|
# We execute that in sync as this access the files in order to access local
data
, and reduce IO
# We execute that in sync as this access the files in order to access local
file
, and reduce IO
BuildTraceSectionsWorker
.
new
.
perform
(
build
.
id
)
BuildCoverageWorker
.
new
.
perform
(
build
.
id
)
...
...
lib/gitlab/ci/trace/chunked_io.rb
View file @
1d53918b
...
...
@@ -134,7 +134,7 @@ module Gitlab
end
def
truncate
(
offset
)
r
aise
ArgumentError
,
'Outside of file'
if
offset
>
size
r
eturn
unless
offset
<
size
&&
offset
>=
0
@tell
=
offset
@size
=
offset
...
...
spec/models/ci/build_trace_chunk_spec.rb
View file @
1d53918b
...
...
@@ -345,7 +345,7 @@ describe Ci::BuildTraceChunk, :clean_gitlab_redis_shared_state do
shared_examples_for
'deletes all build_trace_chunk and data in redis'
do
it
do
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
expect
(
redis
.
scan_each
(
match:
"gitlab:ci:trace:?:chunks:?"
).
to_a
.
count
).
to
eq
(
3
)
expect
(
redis
.
scan_each
(
match:
"gitlab:ci:trace:?:chunks:?"
).
to_a
.
size
).
to
eq
(
3
)
end
expect
(
described_class
.
count
).
to
eq
(
3
)
...
...
@@ -355,7 +355,7 @@ describe Ci::BuildTraceChunk, :clean_gitlab_redis_shared_state do
expect
(
described_class
.
count
).
to
eq
(
0
)
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
expect
(
redis
.
scan_each
(
match:
"gitlab:ci:trace:?:chunks:?"
).
to_a
.
count
).
to
eq
(
0
)
expect
(
redis
.
scan_each
(
match:
"gitlab:ci:trace:?:chunks:?"
).
to_a
.
size
).
to
eq
(
0
)
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