From c6cebb6e621a245c9c2d5bff0df59689b0140373 Mon Sep 17 00:00:00 2001 From: Florian Hines Date: Wed, 26 Mar 2014 13:39:46 -0700 Subject: [PATCH] Write out ring.gz's in a safer fashion Write the ring file to a temporary location first and then move it into place. This should help prevent issues like reading a partial ring file while a new instances is being written out. Change-Id: I28357a2f038a51cb9d823822b92f736dff033a9e --- swift/common/ring/ring.py | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/swift/common/ring/ring.py b/swift/common/ring/ring.py index 8cbcaea675..5b315285ac 100644 --- a/swift/common/ring/ring.py +++ b/swift/common/ring/ring.py @@ -24,6 +24,7 @@ import os from io import BufferedReader from hashlib import md5 from itertools import chain +from tempfile import NamedTemporaryFile from swift.common.utils import hash_path, validate_configuration, json from swift.common.ring.utils import tiers_for_dev @@ -108,12 +109,18 @@ class RingData(object): # # This only works on Python 2.7; on 2.6, we always get the # current time in the gzip output. + tempf = NamedTemporaryFile(dir=".", prefix=filename, delete=False) try: - gz_file = GzipFile(filename, 'wb', mtime=1300507380.0) + gz_file = GzipFile(filename, mode='wb', fileobj=tempf, + mtime=1300507380.0) except TypeError: - gz_file = GzipFile(filename, 'wb') + gz_file = GzipFile(filename, mode='wb', fileobj=tempf) self.serialize_v1(gz_file) gz_file.close() + tempf.flush() + os.fsync(tempf.fileno()) + tempf.close() + os.rename(tempf.name, filename) def to_dict(self): return {'devs': self.devs,