##// END OF EJS Templates
tests: don't reimplement enumerate() in run-tests...
tests: don't reimplement enumerate() in run-tests Differential Revision: https://phab.mercurial-scm.org/D3877

File last commit:

r38400:1a2ff11e default
r38568:29664832 default
Show More
archival.py
344 lines | 10.8 KiB | text/x-python | PythonLexer
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 # archival.py - revision archival for mercurial
#
# Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
#
Martin Geisler
updated license to be explicit about GPL version 2
r8225 # This software may be used and distributed according to the terms of the
Matt Mackall
Update license to GPLv2+
r10263 # GNU General Public License version 2 or any later version.
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Gregory Szorc
archival: use absolute_import
r25916 from __future__ import absolute_import
import gzip
import os
FUJIWARA Katsunori
archival: add "extended-timestamp" extra block for zip archives (issue3600)...
r17628 import struct
Gregory Szorc
archival: use absolute_import
r25916 import tarfile
import time
import zipfile
import zlib
from .i18n import _
from . import (
error,
Matt Harbison
archive: use a templater to build the metadata file...
r33544 formatter,
Gregory Szorc
archival: use absolute_import
r25916 match as matchmod,
Augie Fackler
archival: fsdecode paths before passing to tar or zip objects...
r36724 pycompat,
Matt Harbison
archive: migrate to the fileprefetch callback mechanism
r36156 scmutil,
Gregory Szorc
archival: use absolute_import
r25916 util,
Pierre-Yves David
vfs: use 'vfs' module directly in 'mercurial.archival'...
r31235 vfs as vfsmod,
Gregory Szorc
archival: use absolute_import
r25916 )
timeless
pycompat: switch to util.stringio for py3 compat
r28861 stringio = util.stringio
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Mads Kiilerich
declare local constants instead of using magic values and comments
r17429 # from unzip source code:
_UNX_IFREG = 0x8000
_UNX_IFLNK = 0xa000
Martin Geisler
archival: remove prefix argument from archivers...
r11558 def tidyprefix(dest, kind, prefix):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 '''choose prefix to use for names in archive. make sure prefix is
safe for consumers.'''
if prefix:
Shun-ichi GOTO
Use util.normpath() instead of direct path string operation....
r5842 prefix = util.normpath(prefix)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 else:
Pulkit Goyal
py3: use bytes instead of str in isinstance...
r36455 if not isinstance(dest, bytes):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 raise ValueError('dest must be string if no prefix')
prefix = os.path.basename(dest)
lower = prefix.lower()
Martin Geisler
archival: remove prefix argument from archivers...
r11558 for sfx in exts.get(kind, []):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 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 += '/'
Matt Harbison
archive: drop the leading '.' path component from the prefix (issue4634)...
r24953 # Drop the leading '.' path component if present, so Windows can read the
# zip files (issue4634)
if prefix.startswith('./'):
prefix = prefix[2:]
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 if prefix.startswith('../') or os.path.isabs(lpfx) or '/../' in prefix:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('archive prefix contains illegal components'))
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 return prefix
Martin Geisler
archival: move commands.archive.guess_type to archival.guesskind...
r11557 exts = {
'tar': ['.tar'],
'tbz2': ['.tbz2', '.tar.bz2'],
'tgz': ['.tgz', '.tar.gz'],
'zip': ['.zip'],
}
def guesskind(dest):
for kind, extensions in exts.iteritems():
Augie Fackler
cleanup: use __builtins__.any instead of util.any...
r25149 if any(dest.endswith(ext) for ext in extensions):
Martin Geisler
archival: move commands.archive.guess_type to archival.guesskind...
r11557 return kind
return None
Yuya Nishihara
archive: look for first visible revision to build repo identity (issue4591)...
r24681 def _rootctx(repo):
# repo[0] may be hidden
for rev in repo:
return repo[rev]
return repo['null']
Yuya Nishihara
archive: rewrite default metadata template as a multi-line bytes literal...
r35923 # {tags} on ctx includes local tags and 'tip', with no current way to limit
# that to global tags. Therefore, use {latesttag} as a substitute when
# the distance is 0, since that will be the list of global tags on ctx.
_defaultmetatemplate = br'''
repo: {root}
node: {ifcontains(rev, revset("wdir()"), "{p1node}{dirty}", "{node}")}
branch: {branch|utf8}
{ifeq(latesttagdistance, 0, join(latesttag % "tag: {tag}", "\n"),
separate("\n",
join(latesttag % "latesttag: {tag}", "\n"),
"latesttagdistance: {latesttagdistance}",
"changessincelatesttag: {changessincelatesttag}"))}
'''[1:] # drop leading '\n'
Yuya Nishihara
archive: extract metadata() closure to module-level function...
r24678 def buildmetadata(ctx):
'''build content of .hg_archival.txt'''
repo = ctx.repo()
Matt Harbison
archive: use a templater to build the metadata file...
r33544
opts = {
Matt Harbison
archive: add an experimental config to control the metadata file template...
r33545 'template': repo.ui.config('experimental', 'archivemetatemplate',
Yuya Nishihara
archive: rewrite default metadata template as a multi-line bytes literal...
r35923 _defaultmetatemplate)
Matt Harbison
archive: use a templater to build the metadata file...
r33544 }
out = util.stringio()
Yuya Nishihara
archive: extract metadata() closure to module-level function...
r24678
Matt Harbison
archive: use a templater to build the metadata file...
r33544 fm = formatter.formatter(repo.ui, out, 'archive', opts)
fm.startitem()
fm.context(ctx=ctx)
fm.data(root=_rootctx(repo).hex())
if ctx.rev() is None:
dirty = ''
if ctx.dirty(missing=True):
dirty = '+'
fm.data(dirty=dirty)
fm.end()
return out.getvalue()
Martin Geisler
archival: move commands.archive.guess_type to archival.guesskind...
r11557
Benoit Boissinot
use new style classes
r8778 class tarit(object):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 '''write archive to tar file or stream. can write uncompressed,
or compress with gzip or bzip2.'''
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 class GzipFileWithTime(gzip.GzipFile):
def __init__(self, *args, **kw):
timestamp = None
Augie Fackler
archival: fix a missing r'' on a kwargs check...
r36744 if r'timestamp' in kw:
Pulkit Goyal
py3: handle keyword arguments correctly in archival.py...
r35350 timestamp = kw.pop(r'timestamp')
Martin Geisler
use 'x is None' instead of 'x == None'...
r8527 if timestamp is None:
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 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
timeless@mozdev.org
archival: drop self.filename - deprecated in py2.6
r26198 fname = self.name
Brodie Rao
archival: don't set gzip filename header when there's no filename...
r13102 if fname and fname.endswith('.gz'):
fname = fname[:-3]
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 flags = 0
if fname:
flags = gzip.FNAME
Augie Fackler
archival: use py3 friendly replacements for chr() and long()...
r36746 self.fileobj.write(pycompat.bytechr(flags))
gzip.write32u(self.fileobj, int(self.timestamp))
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 self.fileobj.write('\002')
self.fileobj.write('\377')
if fname:
self.fileobj.write(fname + '\000')
Martin Geisler
archival: remove prefix argument from archivers...
r11558 def __init__(self, dest, mtime, kind=''):
Vadim Gelfer
use commit time as mtime for file archives....
r2477 self.mtime = mtime
Dan Villiom Podlaski Christiansen
explicitly close files...
r13400 self.fileobj = None
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652
Augie Fackler
archival: simplify code and drop message about Python 2.5
r30479 def taropen(mode, name='', fileobj=None):
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 if kind == 'gz':
Pulkit Goyal
py3: slice over bytes or use .startswith() to prevent getting ascii values...
r36465 mode = mode[0:1]
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 if not fileobj:
csaba.henk@creo.hu
Fix tgz archival on Windows....
r4731 fileobj = open(name, mode + 'b')
Augie Fackler
archival: ensure file mode for gzipfile is sysstr...
r36745 gzfileobj = self.GzipFileWithTime(name,
pycompat.sysstr(mode + 'b'),
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 zlib.Z_BEST_COMPRESSION,
fileobj, timestamp=mtime)
Dan Villiom Podlaski Christiansen
explicitly close files...
r13400 self.fileobj = gzfileobj
Augie Fackler
archival: tar file modes need to be sysstrs...
r36725 return tarfile.TarFile.taropen(
name, pycompat.sysstr(mode), gzfileobj)
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652 else:
Augie Fackler
archival: tar file modes need to be sysstrs...
r36725 return tarfile.open(
name, pycompat.sysstr(mode + kind), fileobj)
csaba.henk@creo.hu
timestamp of gzip archives taken from changeset context
r4652
Augie Fackler
archival: our filenames are bytes, not strs...
r36726 if isinstance(dest, bytes):
Augie Fackler
archival: simplify code and drop message about Python 2.5
r30479 self.z = taropen('w:', name=dest)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 else:
Augie Fackler
archival: simplify code and drop message about Python 2.5
r30479 self.z = taropen('w|', fileobj=dest)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Alexis S. L. Carvalho
archive: add symlink support
r4831 def addfile(self, name, mode, islink, data):
Augie Fackler
archival: fsdecode paths before passing to tar or zip objects...
r36724 name = pycompat.fsdecode(name)
Martin Geisler
archival: remove prefix argument from archivers...
r11558 i = tarfile.TarInfo(name)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 i.mtime = self.mtime
i.size = len(data)
Alexis S. L. Carvalho
archive: add symlink support
r4831 if islink:
i.type = tarfile.SYMTYPE
Gregory Szorc
global: mass rewrite to use modern octal syntax...
r25658 i.mode = 0o777
Augie Fackler
archival: fsdecode paths before passing to tar or zip objects...
r36724 i.linkname = pycompat.fsdecode(data)
Alexis S. L. Carvalho
archive: add symlink support
r4831 data = None
Peter van Dijk
fix disappearing symlinks [issue1509]
r7770 i.size = 0
Alexis S. L. Carvalho
archive: add symlink support
r4831 else:
i.mode = mode
timeless
pycompat: switch to util.stringio for py3 compat
r28861 data = stringio(data)
Alexis S. L. Carvalho
archive: add symlink support
r4831 self.z.addfile(i, data)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
def done(self):
self.z.close()
Dan Villiom Podlaski Christiansen
explicitly close files...
r13400 if self.fileobj:
self.fileobj.close()
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Benoit Boissinot
use new style classes
r8778 class zipit(object):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 '''write archive to zip file or stream. can write uncompressed,
or compressed with deflate.'''
Martin Geisler
archival: remove prefix argument from archivers...
r11558 def __init__(self, dest, mtime, compress=True):
Augie Fackler
archival: fsdecode paths before passing to tar or zip objects...
r36724 self.z = zipfile.ZipFile(pycompat.fsdecode(dest), r'w',
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 compress and zipfile.ZIP_DEFLATED or
zipfile.ZIP_STORED)
Martin Geisler
archive: set date to 1980 for very old zip files...
r12319
# Python's zipfile module emits deprecation warnings if we try
# to store files with a date before 1980.
epoch = 315532800 # calendar.timegm((1980, 1, 1, 0, 0, 0, 1, 1, 0))
if mtime < epoch:
mtime = epoch
FUJIWARA Katsunori
archival: add "extended-timestamp" extra block for zip archives (issue3600)...
r17628 self.mtime = mtime
Vadim Gelfer
use commit time as mtime for file archives....
r2477 self.date_time = time.gmtime(mtime)[:6]
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Alexis S. L. Carvalho
archive: add symlink support
r4831 def addfile(self, name, mode, islink, data):
Augie Fackler
archival: fsdecode paths before passing to tar or zip objects...
r36724 i = zipfile.ZipInfo(pycompat.fsdecode(name), self.date_time)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 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
Mads Kiilerich
declare local constants instead of using magic values and comments
r17429 ftype = _UNX_IFREG
Alexis S. L. Carvalho
archive: add symlink support
r4831 if islink:
Gregory Szorc
global: mass rewrite to use modern octal syntax...
r25658 mode = 0o777
Mads Kiilerich
declare local constants instead of using magic values and comments
r17429 ftype = _UNX_IFLNK
Pulkit Goyal
py3: remove use of *L syntax...
r29890 i.external_attr = (mode | ftype) << 16
FUJIWARA Katsunori
archival: add "extended-timestamp" extra block for zip archives (issue3600)...
r17628 # add "extended-timestamp" extra block, because zip archives
# without this will be extracted with unexpected timestamp,
# if TZ is not configured as GMT
i.extra += struct.pack('<hhBl',
0x5455, # block type: "extended-timestamp"
1 + 4, # size of this block
1, # "modification time is present"
Mads Kiilerich
archival: pass integer to struct.pack int field instead of float...
r18301 int(self.mtime)) # last modification (UTC)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 self.z.writestr(i, data)
def done(self):
self.z.close()
Benoit Boissinot
use new style classes
r8778 class fileit(object):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 '''write archive as files in directory.'''
Martin Geisler
archival: remove prefix argument from archivers...
r11558 def __init__(self, name, mtime):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 self.basedir = name
Pierre-Yves David
vfs: use 'vfs' module directly in 'mercurial.archival'...
r31235 self.opener = vfsmod.vfs(self.basedir)
James May
archive: pass thru mtime for directory archives, like other archive types do...
r35203 self.mtime = mtime
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Alexis S. L. Carvalho
archive: add symlink support
r4831 def addfile(self, name, mode, islink, data):
if islink:
self.opener.symlink(data, name)
return
Vincent Parrett
archival: fileit should not use atomictemp, causes performance regression...
r36785 f = self.opener(name, "w", atomictemp=False)
Alexis S. L. Carvalho
archive: use util.opener when archiving files....
r4830 f.write(data)
Greg Ward
atomictempfile: make close() consistent with other file-like objects....
r15057 f.close()
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 destfile = os.path.join(self.basedir, name)
Alexis S. L. Carvalho
archive: use util.opener when archiving files....
r4830 os.chmod(destfile, mode)
James May
archive: pass thru mtime for directory archives, like other archive types do...
r35203 if self.mtime is not None:
os.utime(destfile, (self.mtime, self.mtime))
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
def done(self):
pass
archivers = {
'files': fileit,
'tar': tarit,
Martin Geisler
archival: remove prefix argument from archivers...
r11558 'tbz2': lambda name, mtime: tarit(name, mtime, 'bz2'),
'tgz': lambda name, mtime: tarit(name, mtime, 'gz'),
'uzip': lambda name, mtime: zipit(name, mtime, False),
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 'zip': zipit,
}
def archive(repo, dest, node, kind, decode=True, matchfn=None,
Matt Harbison
archive: change the default prefix to '' from None...
r24172 prefix='', mtime=None, subrepos=False):
Vadim Gelfer
add "archive" command, like "cvs export" only better....
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.
James May
archive: pass thru mtime for directory archives, like other archive types do...
r35203 prefix is name of path to put before every archive member.
mtime is the modified time, in seconds, or None to use the changeset time.
subrepos tells whether to include subrepos.
'''
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Martin Geisler
archival: remove prefix argument from archivers...
r11558 if kind == 'files':
if prefix:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('cannot give prefix when archiving to files'))
Martin Geisler
archival: remove prefix argument from archivers...
r11558 else:
prefix = tidyprefix(dest, kind, prefix)
Alexis S. L. Carvalho
archive: delay extraction of file revisions...
r4951 def write(name, mode, islink, getdata):
data = getdata()
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 if decode:
Matt Mackall
replace filehandle version of wwrite with wwritedata
r4005 data = repo.wwritedata(name, data)
Martin Geisler
archival: remove prefix argument from archivers...
r11558 archiver.addfile(prefix + name, mode, islink, data)
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112
Dirkjan Ochtman
cleanly abort on unknown archive type (issue966)
r6019 if kind not in archivers:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("unknown archive type '%s'") % kind)
Matt Mackall
manifest: remove execf/linkf methods
r6749
ctx = repo[node]
Martin Geisler
archival: remove prefix argument from archivers...
r11558 archiver = archivers[kind](dest, mtime or ctx.date()[0])
Matt Mackall
manifest: remove execf/linkf methods
r6749
Jun Wu
codemod: register core configitems using a script...
r33499 if repo.ui.configbool("ui", "archivemeta"):
Thomas Arendsen Hein
archive: make progress only show files that are actually archived...
r16919 name = '.hg_archival.txt'
if not matchfn or matchfn(name):
Gregory Szorc
global: mass rewrite to use modern octal syntax...
r25658 write(name, 0o644, False, lambda: buildmetadata(ctx))
Gilles Moris
archive: add branch and tag informations to the .hg_archival.txt file...
r9614
Thomas Arendsen Hein
archive: make progress only show files that are actually archived...
r16919 if matchfn:
files = [f for f in ctx.manifest().keys() if matchfn(f)]
else:
files = ctx.manifest().keys()
total = len(files)
Angel Ezquerra
archive: raise error.Abort if the file pattern matches no files...
r18967 if total:
files.sort()
Matt Harbison
scmutil: teach the file prefetch hook to handle multiple commits...
r37780 scmutil.prefetchfiles(repo, [ctx.rev()],
scmutil.matchfiles(repo, files))
Martin von Zweigbergk
archival: use progress helper...
r38400 progress = scmutil.progress(repo.ui, _('archiving'), unit=_('files'),
total=total)
progress.update(0)
for f in files:
Angel Ezquerra
archive: raise error.Abort if the file pattern matches no files...
r18967 ff = ctx.flags(f)
Gregory Szorc
global: mass rewrite to use modern octal syntax...
r25658 write(f, 'x' in ff and 0o755 or 0o644, 'l' in ff, ctx[f].data)
Martin von Zweigbergk
archival: use progress helper...
r38400 progress.increment(item=f)
progress.complete()
Martin Geisler
subrepo: add support for 'hg archive'
r12323
if subrepos:
Mads Kiilerich
subrepos: process subrepos in sorted order...
r18364 for subpath in sorted(ctx.substate):
Matt Harbison
archive: support 'wdir()'...
r25601 sub = ctx.workingsub(subpath)
Martin von Zweigbergk
match: rename "narrowmatcher" to "subdirmatcher" (API)...
r28017 submatch = matchmod.subdirmatcher(subpath, matchfn)
Matt Harbison
subrepo: run the repo decoders when archiving...
r31099 total += sub.archive(archiver, prefix, submatch, decode)
Angel Ezquerra
archive: raise error.Abort if the file pattern matches no files...
r18967
if total == 0:
raise error.Abort(_('no files match the archive pattern'))
Martin Geisler
subrepo: add support for 'hg archive'
r12323
Vadim Gelfer
add "archive" command, like "cvs export" only better....
r2112 archiver.done()
Angel Ezquerra
archive: raise error.Abort if the file pattern matches no files...
r18967 return total