Commit d8381176 authored by Tim Peters's avatar Tim Peters

repozo improvement suggested by Toby Dickenson.

Write backups to a temp file first; flush and fsync when done;
only then rename; in case of a crash, no relevant damaged file
will be left behind then.  And on Windows, Python's os.fsync()
is necessary to convince the OS to write anything to disk.
parent 2c9e11d6
...@@ -35,6 +35,14 @@ investigate to find and repair the true cause). ...@@ -35,6 +35,14 @@ investigate to find and repair the true cause).
Tools Tools
----- -----
repozo.py: Thanks to a suggestion from Toby Dickenson, backups
(whether incremental or full) are first written to a temp file now,
which is fsync'ed at the end, and only after that succeeds is the
file renamed to YYYY-MM-DD-HH-MM-SS.ext form. In case of a system
crash during a repozo backup, this at least makes it much less
likely that a backup file with incomplete or incorrect data will be
left behind.
fsrefs.py: Fleshed out the module docstring, and repaired a bug fsrefs.py: Fleshed out the module docstring, and repaired a bug
wherein spurious error msgs could be produced after reporting a wherein spurious error msgs could be produced after reporting a
problem with an unloadable object. problem with an unloadable object.
......
...@@ -177,6 +177,14 @@ def parseargs(): ...@@ -177,6 +177,14 @@ def parseargs():
return options return options
# afile is a Python file object, or created by gzip.open(). The latter
# doesn't have a fileno() method, so to fsync it we need to reach into
# its underlying file object.
def fsync(afile):
afile.flush()
fileobject = getattr(afile, 'fileobj', afile)
os.fsync(fileobject.fileno())
# Read bytes (no more than n, or to EOF if n is None) in chunks from the # Read bytes (no more than n, or to EOF if n is None) in chunks from the
# current position in file fp. Pass each chunk as an argument to func(). # current position in file fp. Pass each chunk as an argument to func().
# Return the total number of bytes read == the total number of bytes # Return the total number of bytes read == the total number of bytes
...@@ -211,21 +219,28 @@ def checksum(fp, n): ...@@ -211,21 +219,28 @@ def checksum(fp, n):
def copyfile(options, dst, start, n): def copyfile(options, dst, start, n):
# Copy bytes from file src, to file dst, starting at offset start, for n # Copy bytes from file src, to file dst, starting at offset start, for n
# length of bytes # length of bytes. For robustness, we first write, flush and fsync
# to a temp file, then rename the temp file at the end.
sum = md5.new() sum = md5.new()
ifp = open(options.file, 'rb') ifp = open(options.file, 'rb')
ifp.seek(start) ifp.seek(start)
tempname = os.path.join(os.path.dirname(dst), 'tmp.tmp')
if options.gzip: if options.gzip:
ofp = gzip.open(dst, 'wb') ofp = gzip.open(tempname, 'wb')
else: else:
ofp = open(dst, 'wb') ofp = open(tempname, 'wb')
def func(data): def func(data):
sum.update(data) sum.update(data)
ofp.write(data) ofp.write(data)
ndone = dofile(func, ifp, n) ndone = dofile(func, ifp, n)
ofp.close()
ifp.close()
assert ndone == n assert ndone == n
ifp.close()
fsync(ofp)
ofp.close()
os.rename(tempname, dst)
return sum.hexdigest() return sum.hexdigest()
...@@ -353,6 +368,8 @@ def do_full_backup(options): ...@@ -353,6 +368,8 @@ def do_full_backup(options):
datfile = os.path.splitext(dest)[0] + '.dat' datfile = os.path.splitext(dest)[0] + '.dat'
fp = open(datfile, 'w') fp = open(datfile, 'w')
print >> fp, dest, 0, pos, sum print >> fp, dest, 0, pos, sum
fp.flush()
os.fsync(fp.fileno())
fp.close() fp.close()
...@@ -381,6 +398,8 @@ def do_incremental_backup(options, reposz, repofiles): ...@@ -381,6 +398,8 @@ def do_incremental_backup(options, reposz, repofiles):
# This .dat file better exist. Let the exception percolate if not. # This .dat file better exist. Let the exception percolate if not.
fp = open(datfile, 'a') fp = open(datfile, 'a')
print >> fp, dest, reposz, pos, sum print >> fp, dest, reposz, pos, sum
fp.flush()
os.fsync(fp.fileno())
fp.close() fp.close()
......
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