Mercurial > public > mercurial-scm > hg-stable
diff mercurial/subrepo.py @ 51879:3b8d92f71d92
archive: defer opening the output until a file is matched
Before, if no file is matched, an error is thrown, but the archive is
created anyway. When using hgweb, an error 500 is returned as the
response body already exists when the error is seen.
Afterwards, the archive is created before the first match is emitted.
If no match is found, no archive is created. This is more consistent
behavior as an empty archive is not a representable in all output
formats, e.g. tar archives.
author | Joerg Sonnenberger <joerg@bec.de> |
---|---|
date | Wed, 15 Nov 2023 22:11:34 +0100 |
parents | a1e4fa9330d8 |
children | f4733654f144 |
line wrap: on
line diff
--- a/mercurial/subrepo.py Thu Sep 05 13:37:24 2024 +0200 +++ b/mercurial/subrepo.py Wed Nov 15 22:11:34 2023 +0100 @@ -363,9 +363,7 @@ """handle the files command for this subrepo""" return 1 - def archive( - self, archiver, prefix, match: matchmod.basematcher, decode=True - ): + def archive(self, opener, prefix, match: matchmod.basematcher, decode=True): files = [f for f in self.files() if match(f)] total = len(files) relpath = subrelpath(self) @@ -373,10 +371,13 @@ _(b'archiving (%s)') % relpath, unit=_(b'files'), total=total ) progress.update(0) + archiver = None for name in files: flags = self.fileflags(name) mode = b'x' in flags and 0o755 or 0o644 symlink = b'l' in flags + if archiver is None: + archiver = opener() archiver.addfile( prefix + name, mode, symlink, self.filedata(name, decode) ) @@ -651,9 +652,7 @@ ) @annotatesubrepoerror - def archive( - self, archiver, prefix, match: matchmod.basematcher, decode=True - ): + def archive(self, opener, prefix, match: matchmod.basematcher, decode=True): self._get(self._state + (b'hg',)) files = [f for f in self.files() if match(f)] rev = self._state[1] @@ -661,12 +660,12 @@ scmutil.prefetchfiles( self._repo, [(ctx.rev(), scmutil.matchfiles(self._repo, files))] ) - total = abstractsubrepo.archive(self, archiver, prefix, match) + total = abstractsubrepo.archive(self, opener, prefix, match) for subpath in ctx.substate: s = subrepo(ctx, subpath, True) submatch = matchmod.subdirmatcher(subpath, match) subprefix = prefix + subpath + b'/' - total += s.archive(archiver, subprefix, submatch, decode) + total += s.archive(opener, subprefix, submatch, decode) return total @annotatesubrepoerror @@ -1910,9 +1909,7 @@ else: self.wvfs.unlink(f) - def archive( - self, archiver, prefix, match: matchmod.basematcher, decode=True - ): + def archive(self, opener, prefix, match: matchmod.basematcher, decode=True): total = 0 source, revision = self._state if not revision: @@ -1928,6 +1925,7 @@ progress = self.ui.makeprogress( _(b'archiving (%s)') % relpath, unit=_(b'files') ) + archiver = None progress.update(0) for info in tar: if info.isdir(): @@ -1944,6 +1942,8 @@ else: self.ui.warn(_(b'skipping "%s" (unknown type)') % bname) continue + if archiver is None: + archiver = opener() archiver.addfile(prefix + bname, info.mode, info.issym(), data) total += 1 progress.increment()