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
cdcfc3b6
Commit
cdcfc3b6
authored
Nov 17, 2017
by
Douglas Barbosa Alexandre
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Does not inherits Geo::DeletedProject from Project
parent
7807f07a
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
104 additions
and
53 deletions
+104
-53
app/models/geo/deleted_project.rb
app/models/geo/deleted_project.rb
+39
-9
app/services/geo/repository_destroy_service.rb
app/services/geo/repository_destroy_service.rb
+6
-6
ee/app/services/ee/projects/destroy_service.rb
ee/app/services/ee/projects/destroy_service.rb
+0
-26
spec/models/geo/deleted_project_spec.rb
spec/models/geo/deleted_project_spec.rb
+57
-12
spec/services/geo/repository_destroy_service_spec.rb
spec/services/geo/repository_destroy_service_spec.rb
+2
-0
No files found.
app/models/geo/deleted_project.rb
View file @
cdcfc3b6
class
Geo::DeletedProject
<
::
Project
class
Geo::DeletedProject
after_initialize
:readonly!
include
Gitlab
::
CurrentSettings
def
initialize
(
id
:,
name
:,
full_path
:,
repository_storage
:)
attr_reader
:id
,
:name
,
:disk_path
repository_storage
||=
current_application_settings
.
pick_repository_storage
super
(
id:
id
,
name:
name
,
repository_storage:
repository_storage
)
def
initialize
(
id
:,
name
:,
disk_path
:,
repository_storage
:)
@full_path
=
full_path
@id
=
id
@name
=
name
@disk_path
=
disk_path
@repository_storage
=
repository_storage
end
end
def
full_path
alias_method
:full_path
,
:disk_path
@full_path
def
repository
@repository
||=
Repository
.
new
(
disk_path
,
self
)
end
def
repository_storage
@repository_storage
||=
current_application_settings
.
pick_repository_storage
end
def
repository_storage_path
Gitlab
.
config
.
repositories
.
storages
[
repository_storage
].
try
(
:[]
,
'path'
)
end
def
wiki
@wiki
||=
ProjectWiki
.
new
(
self
,
nil
)
end
def
wiki_path
wiki
.
disk_path
end
# When we remove project we move the repository to path+deleted.git then
# outside the transaction we schedule removal of path+deleted with Sidekiq
# through the run_after_commit callback. In a Geo secondary node, we don't
# attempt to remove the repositories inside a transaction because we don't
# have access to the original model anymore, we just need to perform some
# cleanup. This method will run the given block to remove repositories
# immediately otherwise will leave us with stalled repositories on disk.
def
run_after_commit
(
&
block
)
instance_eval
(
&
block
)
end
end
alias_method
:path_with_namespace
,
:full_path
end
end
app/services/geo/repository_destroy_service.rb
View file @
cdcfc3b6
module
Geo
module
Geo
class
RepositoryDestroyService
class
RepositoryDestroyService
attr_reader
:id
,
:name
,
:disk_path
,
:
storage_nam
e
attr_reader
:id
,
:name
,
:disk_path
,
:
repository_storag
e
def
initialize
(
id
,
name
,
disk_path
,
storage_nam
e
)
def
initialize
(
id
,
name
,
disk_path
,
repository_storag
e
)
@id
=
id
@id
=
id
@name
=
name
@name
=
name
@disk_path
=
disk_path
@disk_path
=
disk_path
@
storage_name
=
storage_nam
e
@
repository_storage
=
repository_storag
e
end
end
def
async_execute
def
async_execute
GeoRepositoryDestroyWorker
.
perform_async
(
id
,
name
,
disk_path
,
storage_nam
e
)
GeoRepositoryDestroyWorker
.
perform_async
(
id
,
name
,
disk_path
,
repository_storag
e
)
end
end
def
execute
def
execute
...
@@ -24,8 +24,8 @@ module Geo
...
@@ -24,8 +24,8 @@ module Geo
# rebuilding only what our service class requires
# rebuilding only what our service class requires
::
Geo
::
DeletedProject
.
new
(
id:
id
,
::
Geo
::
DeletedProject
.
new
(
id:
id
,
name:
name
,
name:
name
,
full
_path:
disk_path
,
disk
_path:
disk_path
,
repository_storage:
storage_nam
e
)
repository_storage:
repository_storag
e
)
end
end
end
end
end
end
ee/app/services/ee/projects/destroy_service.rb
View file @
cdcfc3b6
...
@@ -41,38 +41,12 @@ module EE
...
@@ -41,38 +41,12 @@ module EE
flush_caches
(
project
)
flush_caches
(
project
)
trash_repositories!
trash_repositories!
trash_repositories_cleanup!
log_info
(
"Project
\"
#{
project
.
name
}
\"
was removed"
)
log_info
(
"Project
\"
#{
project
.
name
}
\"
was removed"
)
end
end
private
private
# When we remove project we move the repository to path+deleted.git
# then outside the transaction we schedule removal of path+deleted
# with Sidekiq through the after_commit callback. In a Geo secondary
# node we don't have access to the original model anymore then we
# rebuild a Geo::DeletedProject model. Since this model is read-only,
# this callback will not be triggered leaving us with stalled
# repositories on disk.
def
trash_repositories_cleanup!
repo_removed_path
=
removal_path
(
repo_path
)
if
gitlab_shell
.
exists?
(
repository_storage_path
,
repo_removed_path
+
'.git'
)
GitlabShellWorker
.
perform_in
(
5
.
minutes
,
:remove_repository
,
repository_storage_path
,
repo_removed_path
)
end
wiki_removed_path
=
removal_path
(
wiki_path
)
if
gitlab_shell
.
exists?
(
repository_storage_path
,
wiki_removed_path
+
'.git'
)
GitlabShellWorker
.
perform_in
(
5
.
minutes
,
:remove_repository
,
repository_storage_path
,
wiki_removed_path
)
end
end
def
repository_storage_path
project
.
repository_storage_path
end
def
log_audit_event
(
project
)
def
log_audit_event
(
project
)
::
AuditEventService
.
new
(
::
AuditEventService
.
new
(
current_user
,
current_user
,
...
...
spec/models/geo/deleted_project_spec.rb
View file @
cdcfc3b6
require
'spec_helper'
require
'spec_helper'
RSpec
.
describe
Geo
::
DeletedProject
,
type: :model
do
RSpec
.
describe
Geo
::
DeletedProject
,
type: :model
do
subject
{
described_class
.
new
(
id:
1
,
name:
'sample'
,
full_path:
'root/sample'
,
repository_storage:
nil
)
}
include
StubConfiguration
it
{
expect
(
subject
).
to
be_kind_of
(
Project
)
}
before
do
storages
=
{
'foo'
=>
{
'path'
=>
'tmp/tests/storage_foo'
},
'bar'
=>
{
'path'
=>
'tmp/tests/storage_bar'
}
}
describe
'#full_path'
do
stub_storage_settings
(
storages
)
it
'returns the initialized value'
do
expect
(
subject
.
full_path
).
to
eq
'root/sample'
end
end
end
describe
'#path_with_namespace'
do
subject
{
described_class
.
new
(
id:
1
,
name:
'sample'
,
disk_path:
'root/sample'
,
repository_storage:
'foo'
)
}
it
'is an alias for full_path'
do
full_path
=
described_class
.
instance_method
(
:full_path
)
it
{
is_expected
.
to
respond_to
(
:id
)
}
path_with_namespace
=
described_class
.
instance_method
(
:path_with_namespace
)
it
{
is_expected
.
to
respond_to
(
:name
)
}
it
{
is_expected
.
to
respond_to
(
:disk_path
)
}
expect
(
path_with_namespace
).
to
eq
(
full_path
)
describe
'#full_path'
do
it
'is an alias for disk_path'
do
expect
(
subject
.
full_path
).
to
eq
'root/sample'
end
end
end
end
describe
'#repository'
do
describe
'#repository'
do
it
'returns a valid repository'
do
it
'returns a valid repository'
do
expect
(
subject
.
repository
).
to
be_kind_of
(
Repository
)
expect
(
subject
.
repository
).
to
be_kind_of
(
Repository
)
expect
(
subject
.
repository
.
full_path
).
to
eq
(
'root/sample'
)
expect
(
subject
.
repository
.
disk_path
).
to
eq
(
'root/sample'
)
end
end
describe
'#repository_storage'
do
it
'returns the initialized value when set'
do
expect
(
subject
.
repository_storage
).
to
eq
'foo'
end
it
'picks storage from ApplicationSetting when value is not initialized'
do
allow_any_instance_of
(
ApplicationSetting
).
to
receive
(
:pick_repository_storage
).
and_return
(
'bar'
)
subject
=
described_class
.
new
(
id:
1
,
name:
'sample'
,
disk_path:
'root/sample'
,
repository_storage:
nil
)
expect
(
subject
.
repository_storage
).
to
eq
(
'bar'
)
end
end
describe
'#repository_storage_path'
do
it
'returns the repository storage path'
do
expect
(
subject
.
repository_storage_path
).
to
eq
(
'tmp/tests/storage_foo'
)
end
end
describe
'#wiki'
do
it
'returns a valid wiki repository'
do
expect
(
subject
.
wiki
).
to
be_kind_of
(
ProjectWiki
)
expect
(
subject
.
wiki
.
disk_path
).
to
eq
(
'root/sample.wiki'
)
end
end
describe
'#wiki_path'
do
it
'returns the wiki repository path on disk'
do
expect
(
subject
.
wiki_path
).
to
eq
(
'root/sample.wiki'
)
end
end
describe
'#run_after_commit'
do
it
'runs the given block changing self to the caller'
do
expect
(
subject
).
to
receive
(
:repository_storage_path
).
once
subject
.
run_after_commit
{
self
.
repository_storage_path
}
end
end
end
end
end
end
spec/services/geo/repository_destroy_service_spec.rb
View file @
cdcfc3b6
...
@@ -43,6 +43,7 @@ describe Geo::RepositoryDestroyService do
...
@@ -43,6 +43,7 @@ describe Geo::RepositoryDestroyService do
expect
(
::
GitlabShellWorker
).
to
receive
(
:perform_in
)
expect
(
::
GitlabShellWorker
).
to
receive
(
:perform_in
)
.
with
(
5
.
minutes
,
:remove_repository
,
project
.
repository_storage_path
,
"
#{
project
.
disk_path
}
+
#{
project
.
id
}
+deleted"
)
.
with
(
5
.
minutes
,
:remove_repository
,
project
.
repository_storage_path
,
"
#{
project
.
disk_path
}
+
#{
project
.
id
}
+deleted"
)
.
and_return
(
true
)
service
.
execute
service
.
execute
end
end
...
@@ -66,6 +67,7 @@ describe Geo::RepositoryDestroyService do
...
@@ -66,6 +67,7 @@ describe Geo::RepositoryDestroyService do
expect
(
::
GitlabShellWorker
).
to
receive
(
:perform_in
)
expect
(
::
GitlabShellWorker
).
to
receive
(
:perform_in
)
.
with
(
5
.
minutes
,
:remove_repository
,
project
.
repository_storage_path
,
"
#{
project
.
disk_path
}
+
#{
project
.
id
}
+deleted"
)
.
with
(
5
.
minutes
,
:remove_repository
,
project
.
repository_storage_path
,
"
#{
project
.
disk_path
}
+
#{
project
.
id
}
+deleted"
)
.
and_return
(
true
)
service
.
execute
service
.
execute
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