You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							100 lines
						
					
					
						
							2.5 KiB
						
					
					
				
			
		
		
	
	
							100 lines
						
					
					
						
							2.5 KiB
						
					
					
				| import os
 | |
| import shutil
 | |
| import tempfile
 | |
| from atomicwrites import AtomicWriter
 | |
| 
 | |
| 
 | |
| def mkdirs_exists_ok(path):
 | |
|   if path.startswith('http://') or path.startswith('https://'):
 | |
|     raise ValueError('URL path')
 | |
|   try:
 | |
|     os.makedirs(path)
 | |
|   except OSError:
 | |
|     if not os.path.isdir(path):
 | |
|       raise
 | |
| 
 | |
| 
 | |
| def rm_not_exists_ok(path):
 | |
|   try:
 | |
|     os.remove(path)
 | |
|   except OSError:
 | |
|     if os.path.exists(path):
 | |
|       raise
 | |
| 
 | |
| 
 | |
| def rm_tree_or_link(path):
 | |
|   if os.path.islink(path):
 | |
|     os.unlink(path)
 | |
|   elif os.path.isdir(path):
 | |
|     shutil.rmtree(path)
 | |
| 
 | |
| 
 | |
| def get_tmpdir_on_same_filesystem(path):
 | |
|   normpath = os.path.normpath(path)
 | |
|   parts = normpath.split("/")
 | |
|   if len(parts) > 1 and parts[1] == "scratch":
 | |
|     return "/scratch/tmp"
 | |
|   elif len(parts) > 2 and parts[2] == "runner":
 | |
|     return "/{}/runner/tmp".format(parts[1])
 | |
|   return "/tmp"
 | |
| 
 | |
| 
 | |
| class NamedTemporaryDir():
 | |
|   def __init__(self, temp_dir=None):
 | |
|     self._path = tempfile.mkdtemp(dir=temp_dir)
 | |
| 
 | |
|   @property
 | |
|   def name(self):
 | |
|     return self._path
 | |
| 
 | |
|   def close(self):
 | |
|     shutil.rmtree(self._path)
 | |
| 
 | |
|   def __enter__(self):
 | |
|     return self
 | |
| 
 | |
|   def __exit__(self, exc_type, exc_value, traceback):
 | |
|     self.close()
 | |
| 
 | |
| 
 | |
| class CallbackReader:
 | |
|   """Wraps a file, but overrides the read method to also
 | |
|   call a callback function with the number of bytes read so far."""
 | |
|   def __init__(self, f, callback, *args):
 | |
|     self.f = f
 | |
|     self.callback = callback
 | |
|     self.cb_args = args
 | |
|     self.total_read = 0
 | |
| 
 | |
|   def __getattr__(self, attr):
 | |
|     return getattr(self.f, attr)
 | |
| 
 | |
|   def read(self, *args, **kwargs):
 | |
|     chunk = self.f.read(*args, **kwargs)
 | |
|     self.total_read += len(chunk)
 | |
|     self.callback(*self.cb_args, self.total_read)
 | |
|     return chunk
 | |
| 
 | |
| 
 | |
| def _get_fileobject_func(writer, temp_dir):
 | |
|   def _get_fileobject():
 | |
|     return writer.get_fileobject(dir=temp_dir)
 | |
|   return _get_fileobject
 | |
| 
 | |
| 
 | |
| def atomic_write_on_fs_tmp(path, **kwargs):
 | |
|   """Creates an atomic writer using a temporary file in a temporary directory
 | |
|      on the same filesystem as path.
 | |
|   """
 | |
|   # TODO(mgraczyk): This use of AtomicWriter relies on implementation details to set the temp
 | |
|   #                 directory.
 | |
|   writer = AtomicWriter(path, **kwargs)
 | |
|   return writer._open(_get_fileobject_func(writer, get_tmpdir_on_same_filesystem(path)))
 | |
| 
 | |
| 
 | |
| def atomic_write_in_dir(path, **kwargs):
 | |
|   """Creates an atomic writer using a temporary file in the same directory
 | |
|      as the destination file.
 | |
|   """
 | |
|   writer = AtomicWriter(path, **kwargs)
 | |
|   return writer._open(_get_fileobject_func(writer, os.path.dirname(path)))
 | |
| 
 |