-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathzipimager.py
141 lines (131 loc) · 4.31 KB
/
zipimager.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
__app_name__ = 'ZipImager'
__author__ = 'Markus Thilo'
__version__ = '0.6.0_2025-03-30'
__license__ = 'GPL-3'
__email__ = '[email protected]'
__status__ = 'Testing'
__description__ = '''
Using the Python library zipfile this module generates an ZIP archive from a source file structure.
'''
from pathlib import Path
from zipfile import ZipFile, ZIP_DEFLATED
from argparse import ArgumentParser
from lib.pathutils import PathUtils, Progressor
from lib.timestamp import TimeStamp
from lib.logger import Logger
from lib.hashes import FileHash
class ZipImager:
'''Imager using ZipFile'''
def __init__(self, echo=print):
'''Create object'''
self.available = True
self.echo = echo
def create(self, root, filename=None, outdir=None, hashes=['md5'], log=None):
'''Build zip file'''
self.root_path = Path(root)
self.filename = TimeStamp.now_or(filename)
self.outdir = PathUtils.mkdir(outdir)
self.log = log if log else Logger(
filename=self.filename, outdir=self.outdir, head='zipimager.ZipImager', echo=self.echo)
self.echo('Creating Zip file')
self.image_path = self.outdir / f'{self.filename}.zip'
self.tsv_path = self.outdir / f'{self.filename}.tsv'
file_cnt = 0
dir_cnt = 0
other_cnt = 0
file_error_cnt = 0
dir_error_cnt = 0
progress = Progressor(self.root_path, echo=self.echo)
with (
ZipFile(self.image_path, 'w', ZIP_DEFLATED) as zf,
self.tsv_path.open('w', encoding='utf-8') as tsv_fh
):
print('Path\tType\tCopied', file=tsv_fh)
for path, relative, tp in PathUtils.walk(self.root_path):
if tp == 'file':
try:
zf.write(path, relative)
print(f'"{relative}"\tFile\tyes', file=tsv_fh)
file_cnt += 1
except:
print(f'"{relative}"\tFile\tno', file=tsv_fh)
file_error_cnt += 1
elif tp == 'dir':
try:
zf.mkdir(f'{relative}')
print(f'"{relative}"\tDir\tyes', file=tsv_fh)
dir_cnt += 1
except:
print(f'"{relative}"\tDir\tno', file=tsv_fh)
dir_error_cnt += 1
else:
print(f'"{relative}"\tOther\tno', file=tsv_fh)
other_cnt += 1
progress.inc()
msg = f'Created {self.image_path.name} '
msg += f'(Files: {file_cnt} / Directories: {dir_cnt})'
self.log.info(msg, echo=True)
msg = ''
if file_error_cnt > 0:
msg += f'{file_error_cnt} missing file(s)'
if dir_error_cnt > 0:
if msg:
msg += ' and '
msg += f'{dir_error_cnt} missing dir(s)'
if other_cnt:
if msg:
msg += ' and '
msg += f'{other_cnt} other object(s) not included'
if msg:
self.log.warning(msg)
if not hashes:
return
self.log.info('Calculating hash(es)', echo=True)
self.txt_path = self.outdir / f'{self.filename}_hash.txt'
with self.txt_path.open('w', encoding='utf-8') as fh:
for alg in hashes:
hash = FileHash.hashsum(self.image_path, algorithm=alg)
self.log.info(f'Calculated {alg} hash: {hash}', echo=True)
print(f'{alg}\t{hash}', file=fh)
self.log.info('Finished calculating hashes', echo=True)
class ZipImagerCli(ArgumentParser):
'''CLI for the imager'''
def __init__(self, echo=print):
'''Define CLI using argparser'''
super().__init__(description=__description__.strip(), prog=__app_name__.lower())
self.add_argument('-a', '--algorithms',
help=f'''Algorithms to hash seperated by colon (e.g. "md5,sha256", no hashing: "none", default: "md5",
available algorithms: {', '.join(FileHash.get_algorithms())})''', metavar='STRING'
)
self.add_argument('-f', '--filename', type=str,
help='Filename to generated (without extension)', metavar='STRING'
)
self.add_argument('-o', '--outdir', type=Path,
help='Directory to write generated files (default: current)', metavar='DIRECTORY'
)
self.add_argument('root', nargs=1, type=Path,
help='Source root', metavar='DIRECTORY'
)
self.echo = echo
def parse(self, *cmd):
'''Parse arguments'''
args = super().parse_args(*cmd)
self.root = args.root[0]
self.filename = args.filename
self.outdir = args.outdir
self.algorithms = FileHash.parse_algorithms(args.algorithms)
def run(self):
'''Run the imager'''
imager = ZipImager(echo=self.echo)
imager.create(self.root,
filename = self.filename,
outdir = self.outdir,
hashes = self.algorithms
)
imager.log.close()
if __name__ == '__main__': # start here if called as application
app = ZipImagerCli()
app.parse()
app.run()