##// END OF EJS Templates
Merge pull request #6269 from takluyver/atomic-save...
Min RK -
r17754:15ecb841 merge
parent child Browse files
Show More
@@ -13,6 +13,7 b' from tornado import web'
13
13
14 from .manager import ContentsManager
14 from .manager import ContentsManager
15 from IPython.nbformat import current
15 from IPython.nbformat import current
16 from IPython.utils.io import atomic_writing
16 from IPython.utils.path import ensure_dir_exists
17 from IPython.utils.path import ensure_dir_exists
17 from IPython.utils.traitlets import Unicode, Bool, TraitError
18 from IPython.utils.traitlets import Unicode, Bool, TraitError
18 from IPython.utils.py3compat import getcwd
19 from IPython.utils.py3compat import getcwd
@@ -295,7 +296,7 b' class FileContentsManager(ContentsManager):'
295 if 'name' in nb['metadata']:
296 if 'name' in nb['metadata']:
296 nb['metadata']['name'] = u''
297 nb['metadata']['name'] = u''
297
298
298 with io.open(os_path, 'w', encoding='utf-8') as f:
299 with atomic_writing(os_path, encoding='utf-8') as f:
299 current.write(nb, f, u'json')
300 current.write(nb, f, u'json')
300
301
301 def _save_file(self, os_path, model, name='', path=''):
302 def _save_file(self, os_path, model, name='', path=''):
@@ -312,7 +313,7 b' class FileContentsManager(ContentsManager):'
312 bcontent = base64.decodestring(b64_bytes)
313 bcontent = base64.decodestring(b64_bytes)
313 except Exception as e:
314 except Exception as e:
314 raise web.HTTPError(400, u'Encoding error saving %s: %s' % (os_path, e))
315 raise web.HTTPError(400, u'Encoding error saving %s: %s' % (os_path, e))
315 with io.open(os_path, 'wb') as f:
316 with atomic_writing(os_path, text=False) as f:
316 f.write(bcontent)
317 f.write(bcontent)
317
318
318 def _save_directory(self, os_path, model, name='', path=''):
319 def _save_directory(self, os_path, model, name='', path=''):
@@ -16,6 +16,8 b' from __future__ import absolute_import'
16 # Imports
16 # Imports
17 #-----------------------------------------------------------------------------
17 #-----------------------------------------------------------------------------
18 import codecs
18 import codecs
19 from contextlib import contextmanager
20 import io
19 import os
21 import os
20 import sys
22 import sys
21 import tempfile
23 import tempfile
@@ -217,6 +219,61 b" def temp_pyfile(src, ext='.py'):"
217 f.flush()
219 f.flush()
218 return fname, f
220 return fname, f
219
221
222 @contextmanager
223 def atomic_writing(path, text=True, encoding='utf-8', **kwargs):
224 """Context manager to write to a file only if the entire write is successful.
225
226 This works by creating a temporary file in the same directory, and renaming
227 it over the old file if the context is exited without an error. If the
228 target file is a symlink or a hardlink, this will not be preserved: it will
229 be replaced by a new regular file.
230
231 On Windows, there is a small chink in the atomicity: the target file is
232 deleted before renaming the temporary file over it. This appears to be
233 unavoidable.
234
235 Parameters
236 ----------
237 path : str
238 The target file to write to.
239
240 text : bool, optional
241 Whether to open the file in text mode (i.e. to write unicode). Default is
242 True.
243
244 encoding : str, optional
245 The encoding to use for files opened in text mode. Default is UTF-8.
246
247 **kwargs
248 Passed to :func:`io.open`.
249 """
250 dirname, basename = os.path.split(path)
251 handle, tmp_path = tempfile.mkstemp(prefix=basename, dir=dirname, text=text)
252 if text:
253 fileobj = io.open(handle, 'w', encoding=encoding, **kwargs)
254 else:
255 fileobj = io.open(handle, 'wb', **kwargs)
256
257 try:
258 yield fileobj
259 except:
260 fileobj.close()
261 os.remove(tmp_path)
262 raise
263
264 # Flush to disk
265 fileobj.flush()
266 os.fsync(fileobj.fileno())
267
268 # Written successfully, now rename it
269 fileobj.close()
270
271 if os.name == 'nt' and os.path.exists(path):
272 # Rename over existing file doesn't work on Windows
273 os.remove(path)
274
275 os.rename(tmp_path, path)
276
220
277
221 def raw_print(*args, **kw):
278 def raw_print(*args, **kw):
222 """Raw print to sys.__stdout__, otherwise identical interface to print()."""
279 """Raw print to sys.__stdout__, otherwise identical interface to print()."""
@@ -15,6 +15,7 b' from __future__ import print_function'
15 from __future__ import absolute_import
15 from __future__ import absolute_import
16
16
17 import io as stdlib_io
17 import io as stdlib_io
18 import os.path
18 import sys
19 import sys
19
20
20 from subprocess import Popen, PIPE
21 from subprocess import Popen, PIPE
@@ -23,8 +24,11 b' import unittest'
23 import nose.tools as nt
24 import nose.tools as nt
24
25
25 from IPython.testing.decorators import skipif
26 from IPython.testing.decorators import skipif
26 from IPython.utils.io import Tee, capture_output, unicode_std_stream
27 from IPython.utils.io import (Tee, capture_output, unicode_std_stream,
28 atomic_writing,
29 )
27 from IPython.utils.py3compat import doctest_refactor_print, PY3
30 from IPython.utils.py3compat import doctest_refactor_print, PY3
31 from IPython.utils.tempdir import TemporaryDirectory
28
32
29 if PY3:
33 if PY3:
30 from io import StringIO
34 from io import StringIO
@@ -121,4 +125,27 b' def test_UnicodeStdStream_nowrap():'
121 nt.assert_is(unicode_std_stream(), sys.stdout)
125 nt.assert_is(unicode_std_stream(), sys.stdout)
122 assert not sys.stdout.closed
126 assert not sys.stdout.closed
123 finally:
127 finally:
124 sys.stdout = orig_stdout No newline at end of file
128 sys.stdout = orig_stdout
129
130 def test_atomic_writing():
131 class CustomExc(Exception): pass
132
133 with TemporaryDirectory() as td:
134 f1 = os.path.join(td, 'penguin')
135 with stdlib_io.open(f1, 'w') as f:
136 f.write(u'Before')
137
138 with nt.assert_raises(CustomExc):
139 with atomic_writing(f1) as f:
140 f.write(u'Failing write')
141 raise CustomExc
142
143 # Because of the exception, the file should not have been modified
144 with stdlib_io.open(f1, 'r') as f:
145 nt.assert_equal(f.read(), u'Before')
146
147 with atomic_writing(f1) as f:
148 f.write(u'Overwritten')
149
150 with stdlib_io.open(f1, 'r') as f:
151 nt.assert_equal(f.read(), u'Overwritten')
General Comments 0
You need to be logged in to leave comments. Login now