Commit 1082d8f3 authored by Xavier Thompson's avatar Xavier Thompson Committed by Thomas Gambier

software/theia: Include ~/.netrc in resiliency

See merge request nexedi/slapos!1705
parent fb1aafad
...@@ -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 = 45e757f216374d22f0a92d5334dc00f0 md5sum = 43720e0bf8487ae6bd7e7c462ea5c9f6
[slapos.css.in] [slapos.css.in]
_update_hash_filename_ = slapos.css.in _update_hash_filename_ = slapos.css.in
......
...@@ -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 %}
......
...@@ -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 %}
......
...@@ -247,7 +247,7 @@ class ExportAndImportMixin(object): ...@@ -247,7 +247,7 @@ class ExportAndImportMixin(object):
self.assertPromiseSucess() self.assertPromiseSucess()
class TestTheiaExportAndImportFailures(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"\nexit 1' % errmsg) self.customSignatureScript(content='>&2 echo "%s"\nexit 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 TestTheiaExportAndImport(ResilienceMixin, ExportAndImportMixin, ResilientTheiaTestCase): class TestTheiaResilienceImportAndExport(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
......
...@@ -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()
......
...@@ -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):
......
...@@ -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
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment