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
96f050fa
Commit
96f050fa
authored
Aug 11, 2016
by
tiagonbotelho
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Backups do not fail anymore when using tar on annex and custom_hooks
only.
parent
198ae21e
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
150 additions
and
17 deletions
+150
-17
CHANGELOG.md
CHANGELOG.md
+1
-0
lib/backup/repository.rb
lib/backup/repository.rb
+76
-16
spec/tasks/gitlab/backup_rake_spec.rb
spec/tasks/gitlab/backup_rake_spec.rb
+73
-1
No files found.
CHANGELOG.md
View file @
96f050fa
Please view this file on the master branch, on stable branches it's out of date.
Please view this file on the master branch, on stable branches it's out of date.
## 8.14.0 (2016-11-22)
## 8.14.0 (2016-11-22)
-
Backups do not fail anymore when using tar on annex and custom_hooks only. !5814
-
Adds user project membership expired event to clarify why user was removed (Callum Dryden)
-
Adds user project membership expired event to clarify why user was removed (Callum Dryden)
-
Trim leading and trailing whitespace on project_path (Linus Thiel)
-
Trim leading and trailing whitespace on project_path (Linus Thiel)
-
Prevent award emoji via notes for issues/MRs authored by user (barthc)
-
Prevent award emoji via notes for issues/MRs authored by user (barthc)
...
...
lib/backup/repository.rb
View file @
96f050fa
...
@@ -2,11 +2,14 @@ require 'yaml'
...
@@ -2,11 +2,14 @@ require 'yaml'
module
Backup
module
Backup
class
Repository
class
Repository
def
dump
def
dump
prepare
prepare
Project
.
find_each
(
batch_size:
1000
)
do
|
project
|
Project
.
find_each
(
batch_size:
1000
)
do
|
project
|
$progress
.
print
" *
#{
project
.
path_with_namespace
}
... "
$progress
.
print
" *
#{
project
.
path_with_namespace
}
... "
path_to_project_repo
=
path_to_repo
(
project
)
path_to_project_bundle
=
path_to_bundle
(
project
)
# Create namespace dir if missing
# Create namespace dir if missing
FileUtils
.
mkdir_p
(
File
.
join
(
backup_repos_path
,
project
.
namespace
.
path
))
if
project
.
namespace
FileUtils
.
mkdir_p
(
File
.
join
(
backup_repos_path
,
project
.
namespace
.
path
))
if
project
.
namespace
...
@@ -14,8 +17,22 @@ module Backup
...
@@ -14,8 +17,22 @@ module Backup
if
project
.
empty_repo?
if
project
.
empty_repo?
$progress
.
puts
"[SKIPPED]"
.
color
(
:cyan
)
$progress
.
puts
"[SKIPPED]"
.
color
(
:cyan
)
else
else
cmd
=
%W(tar -cf
#{
path_to_bundle
(
project
)
}
-C
#{
path_to_repo
(
project
)
}
.)
in_path
(
path_to_project_repo
)
do
|
dir
|
FileUtils
.
mkdir_p
(
path_to_tars
(
project
))
cmd
=
%W(tar -cf
#{
path_to_tars
(
project
,
dir
)
}
-C
#{
path_to_project_repo
}
#{
dir
}
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
unless
status
.
zero?
puts
"[FAILED]"
.
color
(
:red
)
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
output
abort
'Backup failed'
end
end
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
--git-dir=
#{
path_to_project_repo
}
bundle create
#{
path_to_project_bundle
}
--all)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
if
status
.
zero?
if
status
.
zero?
$progress
.
puts
"[DONE]"
.
color
(
:green
)
$progress
.
puts
"[DONE]"
.
color
(
:green
)
else
else
...
@@ -27,19 +44,22 @@ module Backup
...
@@ -27,19 +44,22 @@ module Backup
end
end
wiki
=
ProjectWiki
.
new
(
project
)
wiki
=
ProjectWiki
.
new
(
project
)
path_to_wiki_repo
=
path_to_repo
(
wiki
)
path_to_wiki_bundle
=
path_to_bundle
(
wiki
)
if
File
.
exist?
(
path_to_
repo
(
wiki
)
)
if
File
.
exist?
(
path_to_
wiki_repo
)
$progress
.
print
" *
#{
wiki
.
path_with_namespace
}
... "
$progress
.
print
" *
#{
wiki
.
path_with_namespace
}
... "
if
wiki
.
repository
.
empty?
if
wiki
.
repository
.
empty?
$progress
.
puts
" [SKIPPED]"
.
color
(
:cyan
)
$progress
.
puts
" [SKIPPED]"
.
color
(
:cyan
)
else
else
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
--git-dir=
#{
path_to_
repo
(
wiki
)
}
bundle create
#{
path_to_bundle
(
wiki
)
}
--all)
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
--git-dir=
#{
path_to_
wiki_repo
}
bundle create
#{
path_to_wiki_bundle
}
--all)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
if
status
.
zero?
if
status
.
zero?
$progress
.
puts
" [DONE]"
.
color
(
:green
)
$progress
.
puts
" [DONE]"
.
color
(
:green
)
else
else
puts
" [FAILED]"
.
color
(
:red
)
puts
" [FAILED]"
.
color
(
:red
)
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
output
abort
'Backup failed'
abort
'Backup failed'
end
end
end
end
...
@@ -60,40 +80,59 @@ module Backup
...
@@ -60,40 +80,59 @@ module Backup
Project
.
find_each
(
batch_size:
1000
)
do
|
project
|
Project
.
find_each
(
batch_size:
1000
)
do
|
project
|
$progress
.
print
" *
#{
project
.
path_with_namespace
}
... "
$progress
.
print
" *
#{
project
.
path_with_namespace
}
... "
path_to_project_repo
=
path_to_repo
(
project
)
path_to_project_bundle
=
path_to_bundle
(
project
)
project
.
ensure_dir_exist
project
.
ensure_dir_exist
if
File
.
exist?
(
path_to_bundle
(
project
))
if
File
.
exists?
(
path_to_project_bundle
)
FileUtils
.
mkdir_p
(
path_to_repo
(
project
))
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
clone --bare
#{
path_to_project_bundle
}
#{
path_to_project_repo
}
)
cmd
=
%W(tar -xf
#{
path_to_bundle
(
project
)
}
-C
#{
path_to_repo
(
project
)
}
)
else
else
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
init --bare
#{
path_to_
repo
(
project
)
}
)
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
init --bare
#{
path_to_
project_repo
}
)
end
end
if
system
(
*
cmd
,
silent
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
if
status
.
zero?
$progress
.
puts
"[DONE]"
.
color
(
:green
)
$progress
.
puts
"[DONE]"
.
color
(
:green
)
else
else
puts
"[FAILED]"
.
color
(
:red
)
puts
"[FAILED]"
.
color
(
:red
)
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
output
abort
'Restore failed'
abort
'Restore failed'
end
end
in_path
(
path_to_tars
(
project
))
do
|
dir
|
cmd
=
%W(tar -xf
#{
path_to_tars
(
project
,
dir
)
}
-C
#{
path_to_project_repo
}
#{
dir
}
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
unless
status
.
zero?
puts
"[FAILED]"
.
color
(
:red
)
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
output
abort
'Restore failed'
end
end
wiki
=
ProjectWiki
.
new
(
project
)
wiki
=
ProjectWiki
.
new
(
project
)
path_to_wiki_repo
=
path_to_repo
(
wiki
)
path_to_wiki_bundle
=
path_to_bundle
(
wiki
)
if
File
.
exist?
(
path_to_
bundle
(
wiki
)
)
if
File
.
exist?
(
path_to_
wiki_bundle
)
$progress
.
print
" *
#{
wiki
.
path_with_namespace
}
... "
$progress
.
print
" *
#{
wiki
.
path_with_namespace
}
... "
# If a wiki bundle exists, first remove the empty repo
# If a wiki bundle exists, first remove the empty repo
# that was initialized with ProjectWiki.new() and then
# that was initialized with ProjectWiki.new() and then
# try to restore with 'git clone --bare'.
# try to restore with 'git clone --bare'.
FileUtils
.
rm_rf
(
path_to_
repo
(
wiki
)
)
FileUtils
.
rm_rf
(
path_to_
wiki_repo
)
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
clone --bare
#{
path_to_
bundle
(
wiki
)
}
#{
path_to_repo
(
wiki
)
}
)
cmd
=
%W(
#{
Gitlab
.
config
.
git
.
bin_path
}
clone --bare
#{
path_to_
wiki_bundle
}
#{
path_to_wiki_repo
}
)
if
system
(
*
cmd
,
silent
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
if
status
.
zero?
$progress
.
puts
" [DONE]"
.
color
(
:green
)
$progress
.
puts
" [DONE]"
.
color
(
:green
)
else
else
puts
" [FAILED]"
.
color
(
:red
)
puts
" [FAILED]"
.
color
(
:red
)
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
"failed:
#{
cmd
.
join
(
' '
)
}
"
puts
output
abort
'Restore failed'
abort
'Restore failed'
end
end
end
end
...
@@ -101,13 +140,15 @@ module Backup
...
@@ -101,13 +140,15 @@ module Backup
$progress
.
print
'Put GitLab hooks in repositories dirs'
.
color
(
:yellow
)
$progress
.
print
'Put GitLab hooks in repositories dirs'
.
color
(
:yellow
)
cmd
=
%W(
#{
Gitlab
.
config
.
gitlab_shell
.
path
}
/bin/create-hooks)
+
repository_storage_paths_args
cmd
=
%W(
#{
Gitlab
.
config
.
gitlab_shell
.
path
}
/bin/create-hooks)
+
repository_storage_paths_args
if
system
(
*
cmd
)
output
,
status
=
Gitlab
::
Popen
.
popen
(
cmd
)
if
status
.
zero?
$progress
.
puts
" [DONE]"
.
color
(
:green
)
$progress
.
puts
" [DONE]"
.
color
(
:green
)
else
else
puts
" [FAILED]"
.
color
(
:red
)
puts
" [FAILED]"
.
color
(
:red
)
puts
"failed:
#{
cmd
}
"
puts
"failed:
#{
cmd
}
"
puts
output
end
end
end
end
protected
protected
...
@@ -117,11 +158,30 @@ module Backup
...
@@ -117,11 +158,30 @@ module Backup
end
end
def
path_to_bundle
(
project
)
def
path_to_bundle
(
project
)
File
.
join
(
backup_repos_path
,
project
.
path_with_namespace
+
".bundle"
)
File
.
join
(
backup_repos_path
,
project
.
path_with_namespace
+
'.bundle'
)
end
def
path_to_tars
(
project
,
dir
=
nil
)
path
=
File
.
join
(
backup_repos_path
,
project
.
path_with_namespace
)
if
dir
File
.
join
(
path
,
"
#{
dir
}
.tar"
)
else
path
end
end
end
def
backup_repos_path
def
backup_repos_path
File
.
join
(
Gitlab
.
config
.
backup
.
path
,
"repositories"
)
File
.
join
(
Gitlab
.
config
.
backup
.
path
,
'repositories'
)
end
def
in_path
(
path
)
return
unless
Dir
.
exist?
(
path
)
dir_entries
=
Dir
.
entries
(
path
)
%w[annex custom_hooks]
.
each
do
|
entry
|
yield
(
entry
)
if
dir_entries
.
include?
(
entry
)
end
end
end
def
prepare
def
prepare
...
...
spec/tasks/gitlab/backup_rake_spec.rb
View file @
96f050fa
...
@@ -79,7 +79,7 @@ describe 'gitlab:app namespace rake task' do
...
@@ -79,7 +79,7 @@ describe 'gitlab:app namespace rake task' do
end
end
end
# backup_restore task
end
# backup_restore task
describe
'backup
_create
'
do
describe
'backup'
do
def
tars_glob
def
tars_glob
Dir
.
glob
(
File
.
join
(
Gitlab
.
config
.
backup
.
path
,
'*_gitlab_backup.tar'
))
Dir
.
glob
(
File
.
join
(
Gitlab
.
config
.
backup
.
path
,
'*_gitlab_backup.tar'
))
end
end
...
@@ -98,6 +98,78 @@ describe 'gitlab:app namespace rake task' do
...
@@ -98,6 +98,78 @@ describe 'gitlab:app namespace rake task' do
@backup_tar
=
tars_glob
.
first
@backup_tar
=
tars_glob
.
first
end
end
def
restore_backup
orig_stdout
=
$stdout
$stdout
=
StringIO
.
new
reenable_backup_sub_tasks
run_rake_task
(
'gitlab:backup:restore'
)
reenable_backup_sub_tasks
$stdout
=
orig_stdout
end
describe
'backup creation and deletion using annex and custom_hooks'
do
let
(
:project
)
{
create
(
:project
)
}
let
(
:user_backup_path
)
{
"repositories/
#{
project
.
path_with_namespace
}
"
}
before
(
:each
)
do
@origin_cd
=
Dir
.
pwd
path
=
File
.
join
(
project
.
repository
.
path_to_repo
,
filename
)
FileUtils
.
mkdir_p
(
path
)
FileUtils
.
touch
(
File
.
join
(
path
,
"dummy.txt"
))
# We need to use the full path instead of the relative one
allow
(
Gitlab
.
config
.
gitlab_shell
).
to
receive
(
:path
).
and_return
(
File
.
expand_path
(
Gitlab
.
config
.
gitlab_shell
.
path
,
Rails
.
root
.
to_s
))
ENV
[
"SKIP"
]
=
"db"
create_backup
end
after
(
:each
)
do
ENV
[
"SKIP"
]
=
""
FileUtils
.
rm
(
@backup_tar
)
Dir
.
chdir
(
@origin_cd
)
end
context
'project uses git-annex and successfully creates backup'
do
let
(
:filename
)
{
"annex"
}
it
'creates annex.tar and project bundle'
do
tar_contents
,
exit_status
=
Gitlab
::
Popen
.
popen
(
%W{tar -tvf
#{
@backup_tar
}
}
)
expect
(
exit_status
).
to
eq
(
0
)
expect
(
tar_contents
).
to
match
(
user_backup_path
)
expect
(
tar_contents
).
to
match
(
"
#{
user_backup_path
}
/annex.tar"
)
expect
(
tar_contents
).
to
match
(
"
#{
user_backup_path
}
.bundle"
)
end
it
'restores files correctly'
do
restore_backup
expect
(
Dir
.
entries
(
File
.
join
(
project
.
repository
.
path
,
"annex"
))).
to
include
(
"dummy.txt"
)
end
end
context
'project uses custom_hooks and successfully creates backup'
do
let
(
:filename
)
{
"custom_hooks"
}
it
'creates custom_hooks.tar and project bundle'
do
tar_contents
,
exit_status
=
Gitlab
::
Popen
.
popen
(
%W{tar -tvf
#{
@backup_tar
}
}
)
expect
(
exit_status
).
to
eq
(
0
)
expect
(
tar_contents
).
to
match
(
user_backup_path
)
expect
(
tar_contents
).
to
match
(
"
#{
user_backup_path
}
/custom_hooks.tar"
)
expect
(
tar_contents
).
to
match
(
"
#{
user_backup_path
}
.bundle"
)
end
it
'restores files correctly'
do
restore_backup
expect
(
Dir
.
entries
(
File
.
join
(
project
.
repository
.
path
,
"custom_hooks"
))).
to
include
(
"dummy.txt"
)
end
end
end
context
'tar creation'
do
context
'tar creation'
do
before
do
before
do
create_backup
create_backup
...
...
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