Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
slapos
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
104
Merge Requests
104
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Jobs
Commits
Open sidebar
nexedi
slapos
Commits
1082d8f3
Commit
1082d8f3
authored
Dec 11, 2024
by
Xavier Thompson
Committed by
Thomas Gambier
Jan 13, 2025
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
software/theia: Include ~/.netrc in resiliency
See merge request
nexedi/slapos!1705
parent
fb1aafad
Changes
7
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
78 additions
and
18 deletions
+78
-18
software/theia/buildout.hash.cfg
software/theia/buildout.hash.cfg
+4
-4
software/theia/instance-export.cfg.jinja.in
software/theia/instance-export.cfg.jinja.in
+2
-0
software/theia/instance-import.cfg.jinja.in
software/theia/instance-import.cfg.jinja.in
+2
-0
software/theia/test/test_resiliency.py
software/theia/test/test_resiliency.py
+39
-5
software/theia/theia_export.py
software/theia/theia_export.py
+15
-4
software/theia/theia_import.py
software/theia/theia_import.py
+15
-4
stack/slapos.cfg
stack/slapos.cfg
+1
-1
No files found.
software/theia/buildout.hash.cfg
View file @
1082d8f3
...
@@ -23,11 +23,11 @@ md5sum = 4f752dd5444a6f9e7c617ec7ccfe62d6
...
@@ -23,11 +23,11 @@ md5sum = 4f752dd5444a6f9e7c617ec7ccfe62d6
[instance-import]
[instance-import]
_update_hash_filename_ = instance-import.cfg.jinja.in
_update_hash_filename_ = instance-import.cfg.jinja.in
md5sum =
647d99aa6f96b2515ac28013145fe81f
md5sum =
854c9234d719ddf855330f9cb5c6a00b
[instance-export]
[instance-export]
_update_hash_filename_ = instance-export.cfg.jinja.in
_update_hash_filename_ = instance-export.cfg.jinja.in
md5sum =
b982e83fa42103b7391d97eb36591174
md5sum =
a723e9ba1d27c4c988e3193eb54e428f
[instance-resilient]
[instance-resilient]
_update_hash_filename_ = instance-resilient.cfg.jinja
_update_hash_filename_ = instance-resilient.cfg.jinja
...
@@ -43,11 +43,11 @@ md5sum = 6a25c6a7f1beb27232a3c9acd8a76500
...
@@ -43,11 +43,11 @@ md5sum = 6a25c6a7f1beb27232a3c9acd8a76500
[theia-export]
[theia-export]
_update_hash_filename_ = theia_export.py
_update_hash_filename_ = theia_export.py
md5sum =
e2f6c483cce09f87ab1e63ae8be0daf4
md5sum =
bc58b7191243bce2bbc95696abce9ef8
[theia-import]
[theia-import]
_update_hash_filename_ = theia_import.py
_update_hash_filename_ = theia_import.py
md5sum = 4
5e757f216374d22f0a92d5334dc00f0
md5sum = 4
3720e0bf8487ae6bd7e7c462ea5c9f6
[slapos.css.in]
[slapos.css.in]
_update_hash_filename_ = slapos.css.in
_update_hash_filename_ = slapos.css.in
...
...
software/theia/instance-export.cfg.jinja.in
View file @
1082d8f3
...
@@ -47,6 +47,7 @@ context =
...
@@ -47,6 +47,7 @@ context =
raw project_path $${directory:project}
raw project_path $${directory:project}
raw public_path $${directory:frontend-static-public}
raw public_path $${directory:frontend-static-public}
raw statefiles_path $${directory:statefiles}
raw statefiles_path $${directory:statefiles}
raw netrc_path_if_exists $${buildout:directory}/.netrc
key exitfile :exitcode-file
key exitfile :exitcode-file
key errorfile :error-file
key errorfile :error-file
{%- raw %}
{%- raw %}
...
@@ -61,6 +62,7 @@ inline =
...
@@ -61,6 +62,7 @@ inline =
--dirs {{ project_path }} \
--dirs {{ project_path }} \
--dirs {{ public_path }} \
--dirs {{ public_path }} \
--dirs {{ statefiles_path }} \
--dirs {{ statefiles_path }} \
--files {{ netrc_path_if_exists }} \
--exitfile {{ exitfile }} \
--exitfile {{ exitfile }} \
--errorfile {{ errorfile }}
--errorfile {{ errorfile }}
{%- endraw %}
{%- endraw %}
...
...
software/theia/instance-import.cfg.jinja.in
View file @
1082d8f3
...
@@ -111,6 +111,7 @@ context =
...
@@ -111,6 +111,7 @@ context =
raw project_path $${directory:project}
raw project_path $${directory:project}
raw public_path $${directory:frontend-static-public}
raw public_path $${directory:frontend-static-public}
raw statefiles_path $${directory:statefiles}
raw statefiles_path $${directory:statefiles}
raw netrc_path_if_exists $${buildout:directory}/.netrc
key exitfile :exitcode-file
key exitfile :exitcode-file
key errorfile :error-file
key errorfile :error-file
{%- raw %}
{%- raw %}
...
@@ -132,6 +133,7 @@ inline =
...
@@ -132,6 +133,7 @@ inline =
--dirs {{ project_path }} \
--dirs {{ project_path }} \
--dirs {{ public_path }} \
--dirs {{ public_path }} \
--dirs {{ statefiles_path }} \
--dirs {{ statefiles_path }} \
--files {{ netrc_path_if_exists }} \
--exitfile {{ exitfile }} \
--exitfile {{ exitfile }} \
--errorfile {{ errorfile }}
--errorfile {{ errorfile }}
{%- endraw %}
{%- endraw %}
...
...
software/theia/test/test_resiliency.py
View file @
1082d8f3
...
@@ -247,7 +247,7 @@ class ExportAndImportMixin(object):
...
@@ -247,7 +247,7 @@ class ExportAndImportMixin(object):
self
.
assertPromiseSucess
()
self
.
assertPromiseSucess
()
class
TestTheiaExportAndImport
Failures
(
ExportAndImportMixin
,
ResilientTheiaTestCase
):
class
TestTheiaExportAndImport
(
ExportAndImportMixin
,
ResilientTheiaTestCase
):
script_relpath
=
os
.
path
.
join
(
script_relpath
=
os
.
path
.
join
(
'srv'
,
'runner'
,
'instance'
,
'slappart0'
,
'srv'
,
'runner'
,
'instance'
,
'slappart0'
,
'srv'
,
'.backup_identity_script'
)
'srv'
,
'.backup_identity_script'
)
...
@@ -327,15 +327,15 @@ class TestTheiaExportAndImportFailures(ExportAndImportMixin, ResilientTheiaTestC
...
@@ -327,15 +327,15 @@ class TestTheiaExportAndImportFailures(ExportAndImportMixin, ResilientTheiaTestC
except
OSError
:
except
OSError
:
pass
pass
def
test_export_promise
(
self
):
def
test_export_promise
_error
(
self
):
self
.
writeFile
(
self
.
getExportExitfile
(),
'1'
)
self
.
writeFile
(
self
.
getExportExitfile
(),
'1'
)
self
.
assertPromiseFailure
(
'ERROR export script failed'
)
self
.
assertPromiseFailure
(
'ERROR export script failed'
)
def
test_import_promise
(
self
):
def
test_import_promise
_error
(
self
):
self
.
writeFile
(
self
.
getImportExitfile
(),
'1'
)
self
.
writeFile
(
self
.
getImportExitfile
(),
'1'
)
self
.
assertPromiseFailure
(
'ERROR import script failed'
)
self
.
assertPromiseFailure
(
'ERROR import script failed'
)
def
test_custom_hash_script
(
self
):
def
test_custom_hash_script
_error
(
self
):
errmsg
=
'Bye bye'
errmsg
=
'Bye bye'
self
.
customSignatureScript
(
content
=
'>&2 echo "%s"
\
n
exit 1'
%
errmsg
)
self
.
customSignatureScript
(
content
=
'>&2 echo "%s"
\
n
exit 1'
%
errmsg
)
custom_script
=
self
.
getPartitionPath
(
'export'
,
self
.
script_relpath
)
custom_script
=
self
.
getPartitionPath
(
'export'
,
self
.
script_relpath
)
...
@@ -354,8 +354,42 @@ class TestTheiaExportAndImportFailures(ExportAndImportMixin, ResilientTheiaTestC
...
@@ -354,8 +354,42 @@ class TestTheiaExportAndImportFailures(ExportAndImportMixin, ResilientTheiaTestC
restore_script
=
self
.
customRestoreScript
(
'exit 1'
)
restore_script
=
self
.
customRestoreScript
(
'exit 1'
)
self
.
assertImportFailure
(
'Run custom restore script %s
\
n
... ERROR !'
%
restore_script
)
self
.
assertImportFailure
(
'Run custom restore script %s
\
n
... ERROR !'
%
restore_script
)
def
assertFileState
(
self
,
filepath
,
expect_content
):
if
expect_content
is
None
:
self
.
assertFalse
(
os
.
path
.
exists
(
filepath
))
else
:
self
.
assertTrue
(
os
.
path
.
exists
(
filepath
))
with
open
(
filepath
)
as
f
:
self
.
assertEqual
(
f
.
read
(),
expect_content
)
def
test_export_import_netrc
(
self
):
netrc_content
=
'machine example.com login someone password secret'
backup_relpath
=
os
.
path
.
join
(
'srv'
,
'backup'
,
'theia'
,
'.netrc'
)
# Propagate .netrc to import theia and check each step
self
.
writeFile
(
self
.
getPartitionPath
(
'export'
,
'.netrc'
),
netrc_content
)
self
.
_doExport
()
self
.
assertFileState
(
self
.
getPartitionPath
(
'export'
,
backup_relpath
),
netrc_content
)
self
.
_doTransfer
()
self
.
assertFileState
(
self
.
getPartitionPath
(
'import'
,
backup_relpath
),
netrc_content
)
self
.
_doImport
()
self
.
assertFileState
(
self
.
getPartitionPath
(
'import'
,
'.netrc'
),
netrc_content
)
# Propagate deletion of .netrc to import theia and check each step
os
.
remove
(
self
.
getPartitionPath
(
'export'
,
'.netrc'
))
self
.
_doExport
()
self
.
assertFileState
(
self
.
getPartitionPath
(
'export'
,
backup_relpath
),
None
)
self
.
_doTransfer
()
self
.
assertFileState
(
self
.
getPartitionPath
(
'import'
,
backup_relpath
),
None
)
self
.
_doImport
()
self
.
assertFileState
(
self
.
getPartitionPath
(
'import'
,
'.netrc'
),
None
)
class
TestTheia
ExportAndIm
port
(
ResilienceMixin
,
ExportAndImportMixin
,
ResilientTheiaTestCase
):
class
TestTheia
ResilienceImportAndEx
port
(
ResilienceMixin
,
ExportAndImportMixin
,
ResilientTheiaTestCase
):
def
test_twice
(
self
):
def
test_twice
(
self
):
# Run two synchronisations on the same instances
# Run two synchronisations on the same instances
# to make sure everything still works the second time
# to make sure everything still works the second time
...
...
software/theia/theia_export.py
View file @
1082d8f3
...
@@ -28,6 +28,7 @@ def main():
...
@@ -28,6 +28,7 @@ def main():
parser
.
add_argument
(
'--backup'
,
required
=
True
)
parser
.
add_argument
(
'--backup'
,
required
=
True
)
parser
.
add_argument
(
'--cfg'
,
required
=
True
)
parser
.
add_argument
(
'--cfg'
,
required
=
True
)
parser
.
add_argument
(
'--dirs'
,
action
=
'append'
)
parser
.
add_argument
(
'--dirs'
,
action
=
'append'
)
parser
.
add_argument
(
'--files'
,
action
=
'append'
)
parser
.
add_argument
(
'--exitfile'
,
required
=
True
)
parser
.
add_argument
(
'--exitfile'
,
required
=
True
)
parser
.
add_argument
(
'--errorfile'
,
required
=
True
)
parser
.
add_argument
(
'--errorfile'
,
required
=
True
)
args
=
parser
.
parse_args
()
args
=
parser
.
parse_args
()
...
@@ -44,6 +45,7 @@ class TheiaExport(object):
...
@@ -44,6 +45,7 @@ class TheiaExport(object):
self
.
backup_dir
=
args
.
backup
self
.
backup_dir
=
args
.
backup
self
.
slapos_cfg
=
cfg
=
args
.
cfg
self
.
slapos_cfg
=
cfg
=
args
.
cfg
self
.
dirs
=
args
.
dirs
self
.
dirs
=
args
.
dirs
self
.
files
=
args
.
files
self
.
exit_file
=
args
.
exitfile
self
.
exit_file
=
args
.
exitfile
self
.
error_file
=
args
.
errorfile
self
.
error_file
=
args
.
errorfile
configp
=
configparser
.
SafeConfigParser
()
configp
=
configparser
.
SafeConfigParser
()
...
@@ -62,8 +64,15 @@ class TheiaExport(object):
...
@@ -62,8 +64,15 @@ class TheiaExport(object):
def
backup_tree
(
self
,
src
):
def
backup_tree
(
self
,
src
):
return
copytree
(
self
.
rsync_bin
,
src
,
self
.
mirror_path
(
src
))
return
copytree
(
self
.
rsync_bin
,
src
,
self
.
mirror_path
(
src
))
def
backup_file
(
self
,
src
):
def
backup_file
(
self
,
src
,
fail_if_missing
=
False
):
return
copyfile
(
src
,
self
.
mirror_path
(
src
))
if
os
.
path
.
exists
(
src
):
self
.
log
(
'Backup file '
+
src
)
copyfile
(
src
,
self
.
mirror_path
(
src
))
elif
fail_if_missing
:
raise
Exception
(
'File %s is missing'
%
src
)
else
:
self
.
log
(
'Delete file from backup '
+
src
)
remove
(
self
.
mirror_path
(
src
))
def
backup_db
(
self
):
def
backup_db
(
self
):
copydb
(
self
.
sqlite3_bin
,
self
.
proxy_db
,
self
.
mirror_path
(
self
.
proxy_db
))
copydb
(
self
.
sqlite3_bin
,
self
.
proxy_db
,
self
.
mirror_path
(
self
.
proxy_db
))
...
@@ -153,13 +162,15 @@ class TheiaExport(object):
...
@@ -153,13 +162,15 @@ class TheiaExport(object):
self
.
remove_signatures
()
self
.
remove_signatures
()
self
.
log
(
'Backup resilient timestamp '
+
timestamp
)
self
.
backup_file
(
timestamp
,
fail_if_missing
=
True
)
self
.
backup_file
(
timestamp
)
for
d
in
self
.
dirs
:
for
d
in
self
.
dirs
:
self
.
log
(
'Backup directory '
+
d
)
self
.
log
(
'Backup directory '
+
d
)
self
.
backup_tree
(
d
)
self
.
backup_tree
(
d
)
for
f
in
self
.
files
:
self
.
backup_file
(
f
)
self
.
log
(
'Backup slapproxy database'
)
self
.
log
(
'Backup slapproxy database'
)
self
.
backup_db
()
self
.
backup_db
()
...
...
software/theia/theia_import.py
View file @
1082d8f3
...
@@ -34,6 +34,7 @@ def main():
...
@@ -34,6 +34,7 @@ def main():
parser
.
add_argument
(
'--backup'
,
required
=
True
)
parser
.
add_argument
(
'--backup'
,
required
=
True
)
parser
.
add_argument
(
'--cfg'
,
required
=
True
)
parser
.
add_argument
(
'--cfg'
,
required
=
True
)
parser
.
add_argument
(
'--dirs'
,
action
=
'append'
)
parser
.
add_argument
(
'--dirs'
,
action
=
'append'
)
parser
.
add_argument
(
'--files'
,
action
=
'append'
)
parser
.
add_argument
(
'--exitfile'
,
required
=
True
)
parser
.
add_argument
(
'--exitfile'
,
required
=
True
)
parser
.
add_argument
(
'--errorfile'
,
required
=
True
)
parser
.
add_argument
(
'--errorfile'
,
required
=
True
)
args
=
parser
.
parse_args
()
args
=
parser
.
parse_args
()
...
@@ -55,6 +56,7 @@ class TheiaImport(object):
...
@@ -55,6 +56,7 @@ class TheiaImport(object):
self
.
backup_dir
=
args
.
backup
self
.
backup_dir
=
args
.
backup
self
.
slapos_cfg
=
cfg
=
args
.
cfg
self
.
slapos_cfg
=
cfg
=
args
.
cfg
self
.
dirs
=
args
.
dirs
self
.
dirs
=
args
.
dirs
self
.
files
=
args
.
files
self
.
exit_file
=
args
.
exitfile
self
.
exit_file
=
args
.
exitfile
self
.
error_file
=
args
.
errorfile
self
.
error_file
=
args
.
errorfile
configp
=
configparser
.
SafeConfigParser
()
configp
=
configparser
.
SafeConfigParser
()
...
@@ -79,9 +81,16 @@ class TheiaImport(object):
...
@@ -79,9 +81,16 @@ class TheiaImport(object):
src
=
self
.
mirror_path
(
dst
)
src
=
self
.
mirror_path
(
dst
)
return
copytree
(
self
.
rsync_bin
,
src
,
dst
,
exclude
,
extrargs
,
verbosity
)
return
copytree
(
self
.
rsync_bin
,
src
,
dst
,
exclude
,
extrargs
,
verbosity
)
def
restore_file
(
self
,
dst
):
def
restore_file
(
self
,
dst
,
fail_if_missing
=
False
):
src
=
self
.
mirror_path
(
dst
)
src
=
self
.
mirror_path
(
dst
)
return
copyfile
(
src
,
dst
)
if
os
.
path
.
exists
(
src
):
self
.
log
(
'Restore file '
+
dst
)
copyfile
(
src
,
dst
)
elif
fail_if_missing
:
raise
Exception
(
'File %s is missing from backup'
%
dst
)
else
:
self
.
log
(
'Remove deleted file '
+
dst
)
remove
(
os
.
path
.
abspath
(
dst
))
def
restore_db
(
self
):
def
restore_db
(
self
):
copydb
(
self
.
sqlite3_bin
,
self
.
mirror_path
(
self
.
proxy_db
),
self
.
proxy_db
)
copydb
(
self
.
sqlite3_bin
,
self
.
mirror_path
(
self
.
proxy_db
),
self
.
proxy_db
)
...
@@ -224,12 +233,14 @@ class TheiaImport(object):
...
@@ -224,12 +233,14 @@ class TheiaImport(object):
self
.
log
(
'Restore directory '
+
d
)
self
.
log
(
'Restore directory '
+
d
)
self
.
restore_tree
(
d
)
self
.
restore_tree
(
d
)
for
f
in
self
.
files
:
self
.
restore_file
(
f
)
self
.
log
(
'Restore slapproxy database'
)
self
.
log
(
'Restore slapproxy database'
)
self
.
restore_db
()
self
.
restore_db
()
timestamp
=
os
.
path
.
join
(
self
.
root_dir
,
'etc'
,
'.resilient_timestamp'
)
timestamp
=
os
.
path
.
join
(
self
.
root_dir
,
'etc'
,
'.resilient_timestamp'
)
self
.
log
(
'Restore resilient timestamp '
+
timestamp
)
self
.
restore_file
(
timestamp
,
fail_if_missing
=
True
)
self
.
restore_file
(
timestamp
)
custom_script
=
os
.
path
.
join
(
self
.
root_dir
,
'srv'
,
'runner-import-restore'
)
custom_script
=
os
.
path
.
join
(
self
.
root_dir
,
'srv'
,
'runner-import-restore'
)
if
os
.
path
.
exists
(
custom_script
):
if
os
.
path
.
exists
(
custom_script
):
...
...
stack/slapos.cfg
View file @
1082d8f3
...
@@ -359,7 +359,7 @@ simplegeneric = 0.8.1
...
@@ -359,7 +359,7 @@ simplegeneric = 0.8.1
singledispatch = 3.4.0.3
singledispatch = 3.4.0.3
six = 1.16.0
six = 1.16.0
slapos.cookbook = 1.0.373
slapos.cookbook = 1.0.373
slapos.core = 1.14.
0
slapos.core = 1.14.
1
slapos.extension.shared = 1.0
slapos.extension.shared = 1.0
slapos.libnetworkcache = 0.25
slapos.libnetworkcache = 0.25
slapos.rebootstrap = 4.7
slapos.rebootstrap = 4.7
...
...
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