1""" 2Autotest tempfile wrapper for mkstemp (known as tempfile here) and 3mkdtemp (known as tempdir). 4 5This wrapper provides a mechanism to clean up temporary files/dirs once they 6are no longer need. 7 8Files/Dirs will have a unique_id prepended to the suffix and a 9_autotmp_ tag appended to the prefix. 10 11It is required that the unique_id param is supplied when a temp dir/file is 12created. 13""" 14 15import shutil, os, logging 16import tempfile as module_tempfile 17 18_TEMPLATE = '_autotmp_' 19 20 21class tempfile(object): 22 """ 23 A wrapper for tempfile.mkstemp 24 25 @param unique_id: required, a unique string to help identify what 26 part of code created the tempfile. 27 @var name: The name of the temporary file. 28 @var fd: the file descriptor of the temporary file that was created. 29 @return a tempfile object 30 example usage: 31 t = autotemp.tempfile(unique_id='fig') 32 t.name # name of file 33 t.fd # file descriptor 34 t.fo # file object 35 t.clean() # clean up after yourself 36 """ 37 def __init__(self, unique_id, suffix='', prefix='', dir=None, 38 text=False): 39 suffix = unique_id + suffix 40 prefix = prefix + _TEMPLATE 41 self.fd, self.name = module_tempfile.mkstemp(suffix=suffix, 42 prefix=prefix, 43 dir=dir, text=text) 44 self.fo = os.fdopen(self.fd) 45 46 47 def clean(self): 48 """ 49 Remove the temporary file that was created. 50 This is also called by the destructor. 51 """ 52 if self.fo: 53 self.fo.close() 54 if self.name and os.path.exists(self.name): 55 os.remove(self.name) 56 57 self.fd = self.fo = self.name = None 58 59 60 def __del__(self): 61 try: 62 if self.name is not None: 63 logging.debug('Clean was not called for ' + self.name) 64 self.clean() 65 except: 66 try: 67 msg = 'An exception occurred while calling the destructor' 68 logging.exception(msg) 69 except: 70 pass 71 72 73class tempdir(object): 74 """ 75 A wrapper for tempfile.mkdtemp 76 77 @var name: The name of the temporary dir. 78 @return A tempdir object 79 example usage: 80 b = autotemp.tempdir(unique_id='exemdir') 81 b.name # your directory 82 b.clean() # clean up after yourself 83 """ 84 def __init__(self, suffix='', unique_id='', prefix='', dir=None): 85 """ 86 Initialize temp directory. 87 88 @param suffix: suffix for dir. 89 @param prefix: prefix for dir. Defaults to '_autotmp'. 90 @param unique_id: unique id of tempdir. 91 @param dir: parent directory of the tempdir. Defaults to /tmp. 92 93 eg: autotemp.tempdir(suffix='suffix', unique_id='123', prefix='prefix') 94 creates a dir like '/tmp/prefix_autotmp_<random hash>123suffix' 95 """ 96 suffix = unique_id + suffix 97 prefix = prefix + _TEMPLATE 98 self.name = module_tempfile.mkdtemp(suffix=suffix, 99 prefix=prefix, dir=dir) 100 101 102 def clean(self): 103 """ 104 Remove the temporary dir that was created. 105 This is also called by the destructor. 106 """ 107 if self.name and os.path.exists(self.name): 108 shutil.rmtree(self.name) 109 110 self.name = None 111 112 113 def __del__(self): 114 try: 115 if self.name: 116 logging.debug('Clean was not called for ' + self.name) 117 self.clean() 118 except: 119 try: 120 msg = 'An exception occurred while calling the destructor' 121 logging.exception(msg) 122 except: 123 pass 124 125 126class dummy_dir(object): 127 """A dummy object representing a directory with a name. 128 129 Only used for compat with the tmpdir, in cases where we wish to 130 reuse a dir with the same interface but not to delete it after 131 we're done using it. 132 """ 133 134 def __init__(self, name): 135 """Initialize the dummy_dir object. 136 137 @param name: Path the the directory. 138 """ 139 self.name = name 140