core: add a stopgap detection for simultaneous qubes.xml access

For now simply throw an exception. Proper solution require some locking

QubesOS/qubes-issues#1729
This commit is contained in:
Marek Marczykowski-Górecki 2016-02-10 16:55:25 +01:00 committed by Wojtek Porczyk
parent c531b0eac2
commit f7f1038f57

View File

@ -1184,6 +1184,8 @@ class Qubes(PropertyHolder):
super(Qubes, self).__init__(xml=None, **kwargs) super(Qubes, self).__init__(xml=None, **kwargs)
self.__load_timestamp = None
if load: if load:
self.load() self.load()
@ -1261,6 +1263,9 @@ class Qubes(PropertyHolder):
vm.events_enabled = True vm.events_enabled = True
vm.fire_event('domain-loaded') vm.fire_event('domain-loaded')
# get a file timestamp (before closing it - still holding the lock!),
# to detect whether anyone else have modified it in the meantime
self.__load_timestamp = os.path.getmtime(self._store)
# intentionally do not call explicit unlock # intentionally do not call explicit unlock
fh.close() fh.close()
del fh del fh
@ -1287,7 +1292,7 @@ class Qubes(PropertyHolder):
mitigated: mitigated:
- Running out of disk space. No space left should not result in empty - Running out of disk space. No space left should not result in empty
file. This is done by writing to temporary file and the renaming. file. This is done by writing to temporary file and then renaming.
- Attempts to write two or more files concurrently. This is done by - Attempts to write two or more files concurrently. This is done by
sophisticated locking. sophisticated locking.
@ -1313,6 +1318,13 @@ class Qubes(PropertyHolder):
else: else:
os.close(fd_old) os.close(fd_old)
if self.__load_timestamp:
current_file_timestamp = os.path.getmtime(self._store)
if current_file_timestamp != self.__load_timestamp:
os.close(fd_old)
raise qubes.exc.QubesException(
"Someone else modified qubes.xml in the meantime")
fh_new = tempfile.NamedTemporaryFile(prefix=self._store, delete=False) fh_new = tempfile.NamedTemporaryFile(prefix=self._store, delete=False)
lxml.etree.ElementTree(self.__xml__()).write( lxml.etree.ElementTree(self.__xml__()).write(
fh_new, encoding='utf-8', pretty_print=True) fh_new, encoding='utf-8', pretty_print=True)
@ -1324,6 +1336,9 @@ class Qubes(PropertyHolder):
# intentionally do not call explicit unlock to not unlock the file # intentionally do not call explicit unlock to not unlock the file
# before all buffers are flushed # before all buffers are flushed
fh_new.close() fh_new.close()
# update stored mtime, in case of multiple save() calls without
# loading qubes.xml again
self.__load_timestamp = os.path.getmtime(self._store)
os.close(fd_old) os.close(fd_old)