repair.py
167 lines
| 5.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / repair.py
Matt Mackall
|
r4702 | # repair.py - functions for repository repair for mercurial | ||
# | ||||
# Copyright 2005, 2006 Chris Mason <mason@suse.com> | ||||
# Copyright 2007 Matt Mackall | ||||
# | ||||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Matt Mackall
|
r4702 | |||
Simon Heimberg
|
r8312 | import changegroup | ||
Joel Rosdahl
|
r6211 | from node import nullrev, short | ||
Martin Geisler
|
r6953 | from i18n import _ | ||
Simon Heimberg
|
r8312 | import os | ||
Matt Mackall
|
r4702 | |||
Nicolas Dumazet
|
r11791 | def _bundle(repo, bases, heads, node, suffix, extranodes=None, compress=True): | ||
Alexis S. L. Carvalho
|
r5905 | """create a bundle with the specified revisions as a backup""" | ||
Alexis S. L. Carvalho
|
r5909 | cg = repo.changegroupsubset(bases, heads, 'strip', extranodes) | ||
Alexis S. L. Carvalho
|
r5905 | backupdir = repo.join("strip-backup") | ||
if not os.path.isdir(backupdir): | ||||
os.mkdir(backupdir) | ||||
Matt Mackall
|
r11333 | name = os.path.join(backupdir, "%s-%s.hg" % (short(node), suffix)) | ||
Nicolas Dumazet
|
r11791 | if compress: | ||
bundletype = "HG10BZ" | ||||
else: | ||||
bundletype = "HG10UN" | ||||
return changegroup.writebundle(cg, name, bundletype) | ||||
Matt Mackall
|
r4702 | |||
Alexis S. L. Carvalho
|
r5910 | def _collectfiles(repo, striprev): | ||
"""find out the filelogs affected by the strip""" | ||||
Benoit Boissinot
|
r8462 | files = set() | ||
Matt Mackall
|
r4702 | |||
Matt Mackall
|
r6750 | for x in xrange(striprev, len(repo)): | ||
Martin Geisler
|
r8479 | files.update(repo[x].files()) | ||
Alexis S. L. Carvalho
|
r5902 | |||
Benoit Boissinot
|
r8462 | return sorted(files) | ||
Alexis S. L. Carvalho
|
r5902 | |||
Alexis S. L. Carvalho
|
r5909 | def _collectextranodes(repo, files, link): | ||
"""return the nodes that have to be saved before the strip""" | ||||
Patrick Mezard
|
r12057 | def collectone(cl, revlog): | ||
Alexis S. L. Carvalho
|
r5909 | extra = [] | ||
Matt Mackall
|
r6750 | startrev = count = len(revlog) | ||
Alexis S. L. Carvalho
|
r5909 | # find the truncation point of the revlog | ||
Martin Geisler
|
r8624 | for i in xrange(count): | ||
Matt Mackall
|
r7361 | lrev = revlog.linkrev(i) | ||
Alexis S. L. Carvalho
|
r5909 | if lrev >= link: | ||
startrev = i + 1 | ||||
break | ||||
# see if any revision after that point has a linkrev less than link | ||||
# (we have to manually save these guys) | ||||
for i in xrange(startrev, count): | ||||
node = revlog.node(i) | ||||
Matt Mackall
|
r7361 | lrev = revlog.linkrev(i) | ||
Alexis S. L. Carvalho
|
r5909 | if lrev < link: | ||
extra.append((node, cl.node(lrev))) | ||||
return extra | ||||
extranodes = {} | ||||
cl = repo.changelog | ||||
Patrick Mezard
|
r12057 | extra = collectone(cl, repo.manifest) | ||
Alexis S. L. Carvalho
|
r5909 | if extra: | ||
extranodes[1] = extra | ||||
for fname in files: | ||||
f = repo.file(fname) | ||||
Patrick Mezard
|
r12057 | extra = collectone(cl, f) | ||
Alexis S. L. Carvalho
|
r5909 | if extra: | ||
extranodes[fname] = extra | ||||
return extranodes | ||||
Alexis S. L. Carvalho
|
r5905 | def strip(ui, repo, node, backup="all"): | ||
Alexis S. L. Carvalho
|
r5901 | cl = repo.changelog | ||
Matt Mackall
|
r4702 | # TODO delete the undo files, and handle undo of merge sets | ||
Alexis S. L. Carvalho
|
r5901 | striprev = cl.rev(node) | ||
Matt Mackall
|
r4702 | |||
Nicolas Dumazet
|
r11791 | keeppartialbundle = backup == 'strip' | ||
Alexis S. L. Carvalho
|
r6147 | # Some revisions with rev > striprev may not be descendants of striprev. | ||
# We have to find these revisions and put them in a bundle, so that | ||||
# we can restore them after the truncations. | ||||
# To create the bundle we use repo.changegroupsubset which requires | ||||
# the list of heads and bases of the set of interesting revisions. | ||||
# (head = revision in the set that has no descendant in the set; | ||||
# base = revision in the set that has no ancestor in the set) | ||||
Benoit Boissinot
|
r8462 | tostrip = set((striprev,)) | ||
saveheads = set() | ||||
Alexis S. L. Carvalho
|
r6147 | savebases = [] | ||
Matt Mackall
|
r6750 | for r in xrange(striprev + 1, len(cl)): | ||
Alexis S. L. Carvalho
|
r6147 | parents = cl.parentrevs(r) | ||
if parents[0] in tostrip or parents[1] in tostrip: | ||||
# r is a descendant of striprev | ||||
Benoit Boissinot
|
r8462 | tostrip.add(r) | ||
Alexis S. L. Carvalho
|
r6147 | # if this is a merge and one of the parents does not descend | ||
# from striprev, mark that parent as a savehead. | ||||
if parents[1] != nullrev: | ||||
for p in parents: | ||||
if p not in tostrip and p > striprev: | ||||
Benoit Boissinot
|
r8462 | saveheads.add(p) | ||
Alexis S. L. Carvalho
|
r6147 | else: | ||
# if no parents of this revision will be stripped, mark it as | ||||
# a savebase | ||||
if parents[0] < striprev and parents[1] < striprev: | ||||
savebases.append(cl.node(r)) | ||||
Matt Mackall
|
r4702 | |||
Martin Geisler
|
r8479 | saveheads.difference_update(parents) | ||
Benoit Boissinot
|
r8462 | saveheads.add(r) | ||
Matt Mackall
|
r4702 | |||
Alexis S. L. Carvalho
|
r6147 | saveheads = [cl.node(r) for r in saveheads] | ||
Alexis S. L. Carvalho
|
r5910 | files = _collectfiles(repo, striprev) | ||
Alexis S. L. Carvalho
|
r5909 | |||
Alexis S. L. Carvalho
|
r5910 | extranodes = _collectextranodes(repo, files, striprev) | ||
Alexis S. L. Carvalho
|
r5909 | |||
Matt Mackall
|
r4702 | # create a changegroup for all the branches we need to keep | ||
Matt Mackall
|
r11197 | backupfile = None | ||
Matt Mackall
|
r4702 | if backup == "all": | ||
Matt Mackall
|
r11197 | backupfile = _bundle(repo, [node], cl.heads(), node, 'backup') | ||
Matt Mackall
|
r11200 | repo.ui.status(_("saved backup bundle to %s\n") % backupfile) | ||
Alexis S. L. Carvalho
|
r5909 | if saveheads or extranodes: | ||
Nicolas Dumazet
|
r11791 | # do not compress partial bundle if we remove it from disk later | ||
Alexis S. L. Carvalho
|
r6147 | chgrpfile = _bundle(repo, savebases, saveheads, node, 'temp', | ||
Nicolas Dumazet
|
r11791 | extranodes=extranodes, compress=keeppartialbundle) | ||
Matt Mackall
|
r4702 | |||
Henrik Stuart
|
r8073 | mfst = repo.manifest | ||
Steve Borho
|
r10881 | tr = repo.transaction("strip") | ||
Henrik Stuart
|
r8073 | offset = len(tr.entries) | ||
Matt Mackall
|
r11197 | try: | ||
tr.startgroup() | ||||
cl.strip(striprev, tr) | ||||
mfst.strip(striprev, tr) | ||||
for fn in files: | ||||
repo.file(fn).strip(striprev, tr) | ||||
tr.endgroup() | ||||
Henrik Stuart
|
r8073 | |||
Matt Mackall
|
r11197 | try: | ||
for i in xrange(offset, len(tr.entries)): | ||||
file, troffset, ignore = tr.entries[i] | ||||
repo.sopener(file, 'a').truncate(troffset) | ||||
tr.close() | ||||
except: | ||||
tr.abort() | ||||
raise | ||||
if saveheads or extranodes: | ||||
Matt Mackall
|
r11202 | ui.note(_("adding branch\n")) | ||
Matt Mackall
|
r11197 | f = open(chgrpfile, "rb") | ||
gen = changegroup.readbundle(f, chgrpfile) | ||||
Matt Mackall
|
r11202 | if not repo.ui.verbose: | ||
# silence internal shuffling chatter | ||||
repo.ui.pushbuffer() | ||||
Matt Mackall
|
r11197 | repo.addchangegroup(gen, 'strip', 'bundle:' + chgrpfile, True) | ||
Matt Mackall
|
r11202 | if not repo.ui.verbose: | ||
repo.ui.popbuffer() | ||||
Matt Mackall
|
r11197 | f.close() | ||
Nicolas Dumazet
|
r11791 | if not keeppartialbundle: | ||
Matt Mackall
|
r11197 | os.unlink(chgrpfile) | ||
Henrik Stuart
|
r8073 | except: | ||
Matt Mackall
|
r11197 | if backupfile: | ||
Martin Geisler
|
r11600 | ui.warn(_("strip failed, full bundle stored in '%s'\n") | ||
% backupfile) | ||||
Matt Mackall
|
r11197 | elif saveheads: | ||
Martin Geisler
|
r11600 | ui.warn(_("strip failed, partial bundle stored in '%s'\n") | ||
Matt Mackall
|
r11197 | % chgrpfile) | ||
Henrik Stuart
|
r8073 | raise | ||
Matt Mackall
|
r4702 | |||
Greg Ward
|
r9150 | repo.destroyed() | ||