9
9
import random
10
10
import getpass
11
11
from typing import IO
12
- import securetar
13
12
import tempfile
14
13
import platform
15
14
from pathlib import Path
15
+ from .hacked_secure_tar_file import HackedSecureTarFile
16
+
17
+ class FailureError (Exception ):
18
+ """Indicates a failure with a user readable message attached"""
19
+
20
+ def __init__ (self , message : str ) -> None :
21
+ """Initialize failure error."""
22
+ super ().__init__ (message )
23
+ self .message = message
24
+
25
+ def __str__ (self ) -> str :
26
+ """Return string representation of failure error."""
27
+ return self .message
16
28
17
- #PATH = "EncryptedFolders.tar"
18
- PATH = "EncryptedFolders.tar"
19
- PASSWORD = "orcsorcs"
20
29
21
30
def password_to_key (password : str ) -> bytes :
22
31
"""Generate a AES Key from password."""
23
- key : bytes = password .encode ()
32
+ key : bytes = password .encode ("utf-8" )
24
33
for _ in range (100 ):
25
34
key = hashlib .sha256 (key ).digest ()
26
35
return key [:16 ]
@@ -31,15 +40,8 @@ def key_to_iv(key: bytes) -> bytes:
31
40
key = hashlib .sha256 (key ).digest ()
32
41
return key [:16 ]
33
42
34
- def _generate_iv (key : bytes , salt : bytes ) -> bytes :
35
- """Generate an iv from data."""
36
- temp_iv = key + salt
37
- for _ in range (100 ):
38
- temp_iv = hashlib .sha256 (temp_iv ).digest ()
39
- return temp_iv [:16 ]
40
-
41
43
def overwrite (line : str ):
42
- sys .stdout .write (f"\r { line } \033 [K" )
44
+ sys .stdout .write (f"\r { line . encode ( 'utf-8' , 'replace' ). decode () } \033 [K" )
43
45
44
46
def readTarMembers (tar : tarfile .TarFile ):
45
47
while (True ):
@@ -57,6 +59,8 @@ def __init__(self, tarfile: tarfile.TarFile):
57
59
except KeyError :
58
60
self ._configMember = self ._tarfile .getmember ("./backup.json" )
59
61
json_file = self ._tarfile .extractfile (self ._configMember )
62
+ if not json_file :
63
+ raise FailureError ("Backup doesn't contain a metadata file named 'snapshot.json' or 'backup.json'" )
60
64
self ._config = json .loads (json_file .read ())
61
65
json_file .close ()
62
66
self ._items = [BackupItem (entry ['slug' ], entry ['name' ], self ) for entry in self ._config .get ("addons" )]
@@ -118,6 +122,8 @@ def __init__(self, slug, name, backup: Backup):
118
122
self ._name = name
119
123
self ._backup = backup
120
124
self ._info = self ._backup ._tarfile .getmember (self .fileName )
125
+ if not self ._info :
126
+ raise FailureError (f"Backup file doesn't contain a file for { self ._name } with the name '{ self .fileName } '" )
121
127
122
128
@property
123
129
def fileName (self ):
@@ -141,7 +147,10 @@ def size(self):
141
147
return self .info .size
142
148
143
149
def _open (self ):
144
- return self ._backup ._tarfile .extractfile (self .info )
150
+ data = self ._backup ._tarfile .extractfile (self .info )
151
+ if not data :
152
+ raise FailureError (f"Backup file doesn't contain a file named { self .info .name } " )
153
+ return data
145
154
146
155
def _extractTo (self , file : IO ):
147
156
progress = 0
@@ -154,9 +163,7 @@ def _extractTo(self, file: IO):
154
163
file .write (data )
155
164
overwrite (f"Extracting '{ self .name } ' { round (100 * progress / self .size , 1 )} %" )
156
165
progress += len (data )
157
- file .flush ()
158
166
overwrite (f"Extracting '{ self .name } ' { round (100 * progress / self .size , 1 )} %" )
159
- file .seek (0 )
160
167
print ()
161
168
162
169
def _copyTar (self , source : tarfile .TarFile , dest : tarfile .TarFile ):
@@ -167,12 +174,13 @@ def _copyTar(self, source: tarfile.TarFile, dest: tarfile.TarFile):
167
174
else :
168
175
dest .addfile (member , source .extractfile (member ))
169
176
170
- def addTo (self , output : tarfile , key : bytes ):
171
- with tempfile .NamedTemporaryFile () as extracted :
172
- self ._extractTo (extracted )
173
- overwrite (f"Decrypting '{ self .name } '" )
174
- extracted .seek (0 )
175
- with securetar .SecureTarFile (Path (extracted .name ), "r" , key = key , gzip = self ._backup .compressed ) as decrypted :
177
+ def addTo (self , temp_folder : str , output : tarfile .TarFile , key : bytes ):
178
+ temp_file = os .path .join (temp_folder , os .urandom (24 ).hex ())
179
+ try :
180
+ with open (temp_file , "wb" ) as f :
181
+ self ._extractTo (f )
182
+ overwrite (f"Decrypting '{ self .name } '" )
183
+ with HackedSecureTarFile (Path (temp_file ), key = key , gzip = self ._backup .compressed ) as decrypted :
176
184
with tempfile .NamedTemporaryFile () as processed :
177
185
tarmode = "w|" + ("gz" if self ._backup .compressed else "" )
178
186
with tarfile .open (f"{ self .slug } .tar" , tarmode , fileobj = processed ) as archivetar :
@@ -187,6 +195,10 @@ def addTo(self, output: tarfile, key: bytes):
187
195
output .addfile (info , processed )
188
196
overwrite (f"Saving '{ self .name } ' done" )
189
197
print ()
198
+ finally :
199
+ if os .path .isfile (temp_file ):
200
+ os .remove (temp_file )
201
+ pass
190
202
191
203
192
204
def main ():
@@ -209,12 +221,13 @@ def main():
209
221
resp = input (f"The output file '{ args .output_file } ' already exists, do you want to overwrite it [y/n]?" )
210
222
if not resp .startswith ("y" ):
211
223
print ("Aborted" )
212
- exit ()
224
+ exit (1 )
213
225
214
226
if args .password is None :
215
227
# ask fro a password
216
228
args .password = getpass .getpass ("Backup Password:" )
217
229
230
+ temp_dir = tempfile .gettempdir ()
218
231
try :
219
232
with tarfile .open (Path (args .backup_file ), "r:" ) as backup_file :
220
233
backup = Backup (backup_file )
@@ -226,10 +239,11 @@ def main():
226
239
return
227
240
228
241
_key = password_to_key (args .password )
242
+ print (_key )
229
243
230
244
with tarfile .open (args .output_file , "w:" ) as output :
231
245
for archive in backup .items :
232
- archive .addTo (output , _key )
246
+ archive .addTo (temp_dir , output , _key )
233
247
234
248
# Add the modified backup config
235
249
backup .addModifiedConfig (output )
@@ -238,11 +252,15 @@ def main():
238
252
except tarfile .ReadError as e :
239
253
if "not a gzip file" in str (e ):
240
254
print ("The file could not be read as a gzip file. Please ensure your password is correct." )
255
+ else :
256
+ raise
257
+ except FailureError as e :
258
+ print (e )
259
+ exit (1 )
241
260
242
261
243
262
if __name__ == '__main__' :
244
263
if platform .system () == 'Windows' :
245
264
from ctypes import windll
246
265
windll .kernel32 .SetConsoleMode (windll .kernel32 .GetStdHandle (- 11 ), 7 )
247
-
248
266
main ()
0 commit comments