archival.py
210 lines
| 7.0 KiB
| text/x-python
|
PythonLexer
/ mercurial / archival.py
Vadim Gelfer
|
r2112 | # archival.py - revision archival for mercurial | ||
# | ||||
# Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com> | ||||
# | ||||
# This software may be used and distributed according to the terms of | ||||
# the GNU General Public License, incorporated herein by reference. | ||||
Matt Mackall
|
r3891 | from i18n import _ | ||
Vadim Gelfer
|
r2112 | from node import * | ||
Matt Mackall
|
r3877 | import cStringIO, os, stat, tarfile, time, util, zipfile | ||
csaba.henk@creo.hu
|
r4652 | import zlib, gzip | ||
Vadim Gelfer
|
r2112 | |||
def tidyprefix(dest, prefix, suffixes): | ||||
'''choose prefix to use for names in archive. make sure prefix is | ||||
safe for consumers.''' | ||||
if prefix: | ||||
prefix = prefix.replace('\\', '/') | ||||
else: | ||||
if not isinstance(dest, str): | ||||
raise ValueError('dest must be string if no prefix') | ||||
prefix = os.path.basename(dest) | ||||
lower = prefix.lower() | ||||
for sfx in suffixes: | ||||
if lower.endswith(sfx): | ||||
prefix = prefix[:-len(sfx)] | ||||
break | ||||
lpfx = os.path.normpath(util.localpath(prefix)) | ||||
prefix = util.pconvert(lpfx) | ||||
if not prefix.endswith('/'): | ||||
prefix += '/' | ||||
if prefix.startswith('../') or os.path.isabs(lpfx) or '/../' in prefix: | ||||
raise util.Abort(_('archive prefix contains illegal components')) | ||||
return prefix | ||||
class tarit: | ||||
'''write archive to tar file or stream. can write uncompressed, | ||||
or compress with gzip or bzip2.''' | ||||
csaba.henk@creo.hu
|
r4652 | class GzipFileWithTime(gzip.GzipFile): | ||
def __init__(self, *args, **kw): | ||||
timestamp = None | ||||
if 'timestamp' in kw: | ||||
timestamp = kw.pop('timestamp') | ||||
if timestamp == None: | ||||
self.timestamp = time.time() | ||||
else: | ||||
self.timestamp = timestamp | ||||
gzip.GzipFile.__init__(self, *args, **kw) | ||||
def _write_gzip_header(self): | ||||
self.fileobj.write('\037\213') # magic header | ||||
self.fileobj.write('\010') # compression method | ||||
fname = self.filename[:-3] | ||||
flags = 0 | ||||
if fname: | ||||
flags = gzip.FNAME | ||||
self.fileobj.write(chr(flags)) | ||||
gzip.write32u(self.fileobj, long(self.timestamp)) | ||||
self.fileobj.write('\002') | ||||
self.fileobj.write('\377') | ||||
if fname: | ||||
self.fileobj.write(fname + '\000') | ||||
Vadim Gelfer
|
r2477 | def __init__(self, dest, prefix, mtime, kind=''): | ||
Vadim Gelfer
|
r2112 | self.prefix = tidyprefix(dest, prefix, ['.tar', '.tar.bz2', '.tar.gz', | ||
Thomas Arendsen Hein
|
r3615 | '.tgz', '.tbz2']) | ||
Vadim Gelfer
|
r2477 | self.mtime = mtime | ||
csaba.henk@creo.hu
|
r4652 | |||
def taropen(name, mode, fileobj=None): | ||||
if kind == 'gz': | ||||
mode = mode[0] | ||||
if not fileobj: | ||||
csaba.henk@creo.hu
|
r4731 | fileobj = open(name, mode + 'b') | ||
csaba.henk@creo.hu
|
r4652 | gzfileobj = self.GzipFileWithTime(name, mode + 'b', | ||
zlib.Z_BEST_COMPRESSION, | ||||
fileobj, timestamp=mtime) | ||||
return tarfile.TarFile.taropen(name, mode, gzfileobj) | ||||
else: | ||||
return tarfile.open(name, mode + kind, fileobj) | ||||
Vadim Gelfer
|
r2112 | if isinstance(dest, str): | ||
csaba.henk@creo.hu
|
r4652 | self.z = taropen(dest, mode='w:') | ||
Vadim Gelfer
|
r2112 | else: | ||
Brendan Cully
|
r4357 | # Python 2.5-2.5.1 have a regression that requires a name arg | ||
csaba.henk@creo.hu
|
r4652 | self.z = taropen(name='', mode='w|', fileobj=dest) | ||
Vadim Gelfer
|
r2112 | |||
def addfile(self, name, mode, data): | ||||
i = tarfile.TarInfo(self.prefix + name) | ||||
i.mtime = self.mtime | ||||
i.size = len(data) | ||||
i.mode = mode | ||||
self.z.addfile(i, cStringIO.StringIO(data)) | ||||
def done(self): | ||||
self.z.close() | ||||
class tellable: | ||||
'''provide tell method for zipfile.ZipFile when writing to http | ||||
response file object.''' | ||||
def __init__(self, fp): | ||||
self.fp = fp | ||||
self.offset = 0 | ||||
def __getattr__(self, key): | ||||
return getattr(self.fp, key) | ||||
def write(self, s): | ||||
self.fp.write(s) | ||||
self.offset += len(s) | ||||
def tell(self): | ||||
return self.offset | ||||
class zipit: | ||||
'''write archive to zip file or stream. can write uncompressed, | ||||
or compressed with deflate.''' | ||||
Vadim Gelfer
|
r2477 | def __init__(self, dest, prefix, mtime, compress=True): | ||
Vadim Gelfer
|
r2112 | self.prefix = tidyprefix(dest, prefix, ('.zip',)) | ||
Colin McMillen
|
r2168 | if not isinstance(dest, str): | ||
try: | ||||
dest.tell() | ||||
Thomas Arendsen Hein
|
r2169 | except (AttributeError, IOError): | ||
Colin McMillen
|
r2168 | dest = tellable(dest) | ||
Vadim Gelfer
|
r2112 | self.z = zipfile.ZipFile(dest, 'w', | ||
compress and zipfile.ZIP_DEFLATED or | ||||
zipfile.ZIP_STORED) | ||||
Vadim Gelfer
|
r2477 | self.date_time = time.gmtime(mtime)[:6] | ||
Vadim Gelfer
|
r2112 | |||
def addfile(self, name, mode, data): | ||||
i = zipfile.ZipInfo(self.prefix + name, self.date_time) | ||||
i.compress_type = self.z.compression | ||||
# unzip will not honor unix file modes unless file creator is | ||||
# set to unix (id 3). | ||||
i.create_system = 3 | ||||
i.external_attr = (mode | stat.S_IFREG) << 16L | ||||
self.z.writestr(i, data) | ||||
def done(self): | ||||
self.z.close() | ||||
class fileit: | ||||
'''write archive as files in directory.''' | ||||
Vadim Gelfer
|
r2477 | def __init__(self, name, prefix, mtime): | ||
Vadim Gelfer
|
r2112 | if prefix: | ||
raise util.Abort(_('cannot give prefix when archiving to files')) | ||||
self.basedir = name | ||||
self.dirs = {} | ||||
self.oflags = (os.O_CREAT | os.O_EXCL | os.O_WRONLY | | ||||
getattr(os, 'O_BINARY', 0) | | ||||
getattr(os, 'O_NOFOLLOW', 0)) | ||||
def addfile(self, name, mode, data): | ||||
destfile = os.path.join(self.basedir, name) | ||||
destdir = os.path.dirname(destfile) | ||||
if destdir not in self.dirs: | ||||
if not os.path.isdir(destdir): | ||||
os.makedirs(destdir) | ||||
self.dirs[destdir] = 1 | ||||
os.fdopen(os.open(destfile, self.oflags, mode), 'wb').write(data) | ||||
def done(self): | ||||
pass | ||||
archivers = { | ||||
'files': fileit, | ||||
'tar': tarit, | ||||
Vadim Gelfer
|
r2477 | 'tbz2': lambda name, prefix, mtime: tarit(name, prefix, mtime, 'bz2'), | ||
'tgz': lambda name, prefix, mtime: tarit(name, prefix, mtime, 'gz'), | ||||
'uzip': lambda name, prefix, mtime: zipit(name, prefix, mtime, False), | ||||
Vadim Gelfer
|
r2112 | 'zip': zipit, | ||
} | ||||
def archive(repo, dest, node, kind, decode=True, matchfn=None, | ||||
Vadim Gelfer
|
r2477 | prefix=None, mtime=None): | ||
Vadim Gelfer
|
r2112 | '''create archive of repo as it was at node. | ||
dest can be name of directory, name of archive file, or file | ||||
object to write archive to. | ||||
kind is type of archive to create. | ||||
decode tells whether to put files through decode filters from | ||||
hgrc. | ||||
matchfn is function to filter names of files to write to archive. | ||||
prefix is name of path to put before every archive member.''' | ||||
def write(name, mode, data): | ||||
if matchfn and not matchfn(name): return | ||||
if decode: | ||||
Matt Mackall
|
r4005 | data = repo.wwritedata(name, data) | ||
Vadim Gelfer
|
r2112 | archiver.addfile(name, mode, data) | ||
Benoit Boissinot
|
r3968 | ctx = repo.changectx(node) | ||
archiver = archivers[kind](dest, prefix, mtime or ctx.date()[0]) | ||||
m = ctx.manifest() | ||||
Alexis S. L. Carvalho
|
r2857 | items = m.items() | ||
items.sort() | ||||
Vadim Gelfer
|
r2112 | write('.hg_archival.txt', 0644, | ||
'repo: %s\nnode: %s\n' % (hex(repo.changelog.node(0)), hex(node))) | ||||
Alexis S. L. Carvalho
|
r2857 | for filename, filenode in items: | ||
write(filename, m.execf(filename) and 0755 or 0644, | ||||
Vadim Gelfer
|
r2112 | repo.file(filename).read(filenode)) | ||
archiver.done() | ||||