28
29
from bzrlib.export import _export_iter_entries
29
from bzrlib.filters import (
31
filtered_output_bytes,
33
30
from bzrlib.trace import mutter
36
# Windows expects this bit to be set in the 'external_attr' section
37
# Or it won't consider the entry a directory
33
# Windows expects this bit to be set in the 'external_attr' section,
34
# or it won't consider the entry a directory.
38
35
ZIP_DIRECTORY_BIT = (1 << 4)
39
36
FILE_PERMISSIONS = (0644 << 16)
37
DIR_PERMISSIONS = (0755 << 16)
41
39
_FILE_ATTR = stat.S_IFREG | FILE_PERMISSIONS
42
_DIR_ATTR = stat.S_IFDIR | ZIP_DIRECTORY_BIT
45
def zip_exporter(tree, dest, root, subdir, filtered=False,
46
per_file_timestamps=False):
40
_DIR_ATTR = stat.S_IFDIR | ZIP_DIRECTORY_BIT | DIR_PERMISSIONS
43
def zip_exporter_generator(tree, dest, root, subdir=None,
44
force_mtime=None, fileobj=None):
47
45
""" Export this tree to a new zip file.
49
47
`dest` will be created holding the contents of this tree; if it
50
48
already exists, it will be overwritten".
52
mutter('export version %r', tree)
54
now = time.localtime()[:6]
56
51
compression = zipfile.ZIP_DEFLATED
52
if fileobj is not None:
57
56
zipf = zipfile.ZipFile(dest, "w", compression)
60
for dp, ie in _export_iter_entries(tree, subdir):
58
for dp, tp, ie in _export_iter_entries(tree, subdir):
61
59
file_id = ie.file_id
62
60
mutter(" export {%s} kind %s to %s", file_id, ie.kind, dest)
64
62
# zipfile.ZipFile switches all paths to forward
65
63
# slashes anyway, so just stick with that.
66
if per_file_timestamps:
67
mtime = tree.get_file_mtime(ie.file_id, dp)
64
if force_mtime is not None:
67
mtime = tree.get_file_mtime(ie.file_id, tp)
68
date_time = time.localtime(mtime)[:6]
70
69
filename = osutils.pathjoin(root, dp).encode('utf8')
71
70
if ie.kind == "file":
72
71
zinfo = zipfile.ZipInfo(
75
74
zinfo.compress_type = compression
76
75
zinfo.external_attr = _FILE_ATTR
78
chunks = tree.get_file_lines(file_id)
79
filters = tree._content_filter_stack(dp)
80
context = ContentFilterContext(dp, tree, ie)
81
contents = filtered_output_bytes(chunks, filters, context)
82
content = ''.join(contents)
84
content = tree.get_file_text(file_id)
76
content = tree.get_file_text(file_id, tp)
85
77
zipf.writestr(zinfo, content)
86
78
elif ie.kind == "directory":
87
79
# Directories must contain a trailing slash, to indicate
89
81
# not just empty files.
90
82
zinfo = zipfile.ZipInfo(
91
83
filename=filename + '/',
93
85
zinfo.compress_type = compression
94
86
zinfo.external_attr = _DIR_ATTR
95
zipf.writestr(zinfo,'')
87
zipf.writestr(zinfo, '')
96
88
elif ie.kind == "symlink":
97
89
zinfo = zipfile.ZipInfo(
98
90
filename=(filename + '.lnk'),
100
92
zinfo.compress_type = compression
101
93
zinfo.external_attr = _FILE_ATTR
102
zipf.writestr(zinfo, ie.symlink_target)
94
zipf.writestr(zinfo, tree.get_symlink_target(file_id, tp))