13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
"""Export a Tree to a zip file.
17
"""Export a Tree to a non-versioned directory.
20
from __future__ import absolute_import
32
29
from bzrlib.trace import mutter
35
# Windows expects this bit to be set in the 'external_attr' section,
36
# or it won't consider the entry a directory.
32
# Windows expects this bit to be set in the 'external_attr' section
33
# Or it won't consider the entry a directory
37
34
ZIP_DIRECTORY_BIT = (1 << 4)
38
FILE_PERMISSIONS = (0644 << 16)
39
DIR_PERMISSIONS = (0755 << 16)
41
_FILE_ATTR = stat.S_IFREG | FILE_PERMISSIONS
42
_DIR_ATTR = stat.S_IFDIR | ZIP_DIRECTORY_BIT | DIR_PERMISSIONS
45
def zip_exporter_generator(tree, dest, root, subdir=None,
46
force_mtime=None, fileobj=None):
36
_FILE_ATTR = stat.S_IFREG
37
_DIR_ATTR = stat.S_IFDIR | ZIP_DIRECTORY_BIT
40
def zip_exporter(tree, dest, root, subdir):
47
41
""" Export this tree to a new zip file.
49
43
`dest` will be created holding the contents of this tree; if it
50
44
already exists, it will be overwritten".
46
mutter('export version %r', tree)
48
now = time.localtime()[:6]
53
50
compression = zipfile.ZIP_DEFLATED
54
if fileobj is not None:
58
51
zipf = zipfile.ZipFile(dest, "w", compression)
60
for dp, tp, ie in _export_iter_entries(tree, subdir):
54
for dp, ie in _export_iter_entries(tree, subdir):
61
55
file_id = ie.file_id
62
56
mutter(" export {%s} kind %s to %s", file_id, ie.kind, dest)
64
58
# zipfile.ZipFile switches all paths to forward
65
59
# slashes anyway, so just stick with that.
66
if force_mtime is not None:
69
mtime = tree.get_file_mtime(ie.file_id, tp)
70
date_time = time.localtime(mtime)[:6]
71
60
filename = osutils.pathjoin(root, dp).encode('utf8')
72
61
if ie.kind == "file":
73
62
zinfo = zipfile.ZipInfo(
76
65
zinfo.compress_type = compression
77
66
zinfo.external_attr = _FILE_ATTR
78
content = tree.get_file_text(file_id, tp)
79
zipf.writestr(zinfo, content)
67
zipf.writestr(zinfo, tree.get_file_text(file_id))
80
68
elif ie.kind == "directory":
81
69
# Directories must contain a trailing slash, to indicate
82
70
# to the zip routine that they are really directories and
83
71
# not just empty files.
84
72
zinfo = zipfile.ZipInfo(
85
73
filename=filename + '/',
87
75
zinfo.compress_type = compression
88
76
zinfo.external_attr = _DIR_ATTR
89
zipf.writestr(zinfo, '')
77
zipf.writestr(zinfo,'')
90
78
elif ie.kind == "symlink":
91
79
zinfo = zipfile.ZipInfo(
92
80
filename=(filename + '.lnk'),
94
82
zinfo.compress_type = compression
95
83
zinfo.external_attr = _FILE_ATTR
96
zipf.writestr(zinfo, tree.get_symlink_target(file_id, tp))
84
zipf.writestr(zinfo, ie.symlink_target)