[postprocessor:zip] add 'mode' option (#355)
This commit is contained in:
@@ -1465,6 +1465,22 @@ Default ``false``
|
|||||||
Description Keep the actual files after writing them to a ZIP archive.
|
Description Keep the actual files after writing them to a ZIP archive.
|
||||||
=========== =====
|
=========== =====
|
||||||
|
|
||||||
|
zip.mode
|
||||||
|
--------
|
||||||
|
=========== =====
|
||||||
|
Type ``string``
|
||||||
|
Default ``"default"``
|
||||||
|
Description * ``"default"``: Write the central directory file header
|
||||||
|
once after everything is done or an exception is raised.
|
||||||
|
|
||||||
|
* ``"safe"``: Update the central directory file header
|
||||||
|
each time a file is stored in a ZIP archive.
|
||||||
|
|
||||||
|
This greatly reduces the chance a ZIP archive gets corrupted in
|
||||||
|
case the Python interpreter gets shut down unexpectedly
|
||||||
|
(power outage, SIGKILL) but is also a lot slower.
|
||||||
|
=========== =====
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Miscellaneous Options
|
Miscellaneous Options
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
# Copyright 2018 Mike Fährmann
|
# Copyright 2018-2019 Mike Fährmann
|
||||||
#
|
#
|
||||||
# This program is free software; you can redistribute it and/or modify
|
# This program is free software; you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU General Public License version 2 as
|
# it under the terms of the GNU General Public License version 2 as
|
||||||
@@ -25,7 +25,7 @@ class ZipPP(PostProcessor):
|
|||||||
def __init__(self, pathfmt, options):
|
def __init__(self, pathfmt, options):
|
||||||
PostProcessor.__init__(self)
|
PostProcessor.__init__(self)
|
||||||
self.delete = not options.get("keep-files", False)
|
self.delete = not options.get("keep-files", False)
|
||||||
self.ext = "." + options.get("extension", "zip")
|
ext = "." + options.get("extension", "zip")
|
||||||
algorithm = options.get("compression", "store")
|
algorithm = options.get("compression", "store")
|
||||||
if algorithm not in self.COMPRESSION_ALGORITHMS:
|
if algorithm not in self.COMPRESSION_ALGORITHMS:
|
||||||
self.log.warning(
|
self.log.warning(
|
||||||
@@ -34,29 +34,45 @@ class ZipPP(PostProcessor):
|
|||||||
algorithm = "store"
|
algorithm = "store"
|
||||||
|
|
||||||
self.path = pathfmt.realdirectory
|
self.path = pathfmt.realdirectory
|
||||||
self.zfile = zipfile.ZipFile(
|
args = (self.path + ext, "a",
|
||||||
self.path + self.ext, "a",
|
self.COMPRESSION_ALGORITHMS[algorithm], True)
|
||||||
self.COMPRESSION_ALGORITHMS[algorithm], True)
|
|
||||||
|
|
||||||
def run(self, pathfmt):
|
if options.get("mode") == "safe":
|
||||||
|
self.run = self._write_safe
|
||||||
|
self.zfile = None
|
||||||
|
self.args = args
|
||||||
|
else:
|
||||||
|
self.run = self._write
|
||||||
|
self.zfile = zipfile.ZipFile(*args)
|
||||||
|
|
||||||
|
def _write(self, pathfmt, zfile=None):
|
||||||
# 'NameToInfo' is not officially documented, but it's available
|
# 'NameToInfo' is not officially documented, but it's available
|
||||||
# for all supported Python versions and using it directly is a lot
|
# for all supported Python versions and using it directly is a lot
|
||||||
# better than calling getinfo()
|
# faster than calling getinfo()
|
||||||
if pathfmt.filename not in self.zfile.NameToInfo:
|
if zfile is None:
|
||||||
self.zfile.write(pathfmt.temppath, pathfmt.filename)
|
zfile = self.zfile
|
||||||
|
if pathfmt.filename not in zfile.NameToInfo:
|
||||||
|
zfile.write(pathfmt.temppath, pathfmt.filename)
|
||||||
pathfmt.delete = self.delete
|
pathfmt.delete = self.delete
|
||||||
|
|
||||||
|
def _write_safe(self, pathfmt):
|
||||||
|
with zipfile.ZipFile(*self.args) as zfile:
|
||||||
|
self._write(pathfmt, zfile)
|
||||||
|
|
||||||
def finalize(self):
|
def finalize(self):
|
||||||
self.zfile.close()
|
if self.zfile:
|
||||||
|
self.zfile.close()
|
||||||
|
|
||||||
if self.delete:
|
if self.delete:
|
||||||
try:
|
try:
|
||||||
|
# remove target directory
|
||||||
os.rmdir(self.path)
|
os.rmdir(self.path)
|
||||||
except OSError:
|
except OSError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
if not self.zfile.NameToInfo:
|
if self.zfile and not self.zfile.NameToInfo:
|
||||||
try:
|
try:
|
||||||
|
# delete empty zip archive
|
||||||
os.unlink(self.zfile.filename)
|
os.unlink(self.zfile.filename)
|
||||||
except OSError:
|
except OSError:
|
||||||
pass
|
pass
|
||||||
|
|||||||
Reference in New Issue
Block a user