Show More
@@ -0,0 +1,61 b'' | |||
|
1 | #!/bin/sh | |
|
2 | ||
|
3 | # test stripping of filelogs where the linkrev doesn't always increase | |
|
4 | ||
|
5 | echo '[extensions]' >> $HGRCPATH | |
|
6 | echo 'hgext.mq =' >> $HGRCPATH | |
|
7 | ||
|
8 | hg init orig | |
|
9 | cd orig | |
|
10 | ||
|
11 | hidefilename() | |
|
12 | { | |
|
13 | sed -e 's/saving bundle to .*strip-backup/saving bundle to strip-backup/' | |
|
14 | } | |
|
15 | ||
|
16 | commit() | |
|
17 | { | |
|
18 | hg up -qC null | |
|
19 | count=1 | |
|
20 | for i in "$@"; do | |
|
21 | for f in $i; do | |
|
22 | echo $count > $f | |
|
23 | done | |
|
24 | count=`expr $count + 1` | |
|
25 | done | |
|
26 | hg commit -qAm "$*" | |
|
27 | } | |
|
28 | ||
|
29 | # 2 1 0 2 0 1 2 | |
|
30 | commit '201 210' | |
|
31 | ||
|
32 | commit '102 120' '210' | |
|
33 | ||
|
34 | commit '021' | |
|
35 | ||
|
36 | commit '201' '021 120' | |
|
37 | ||
|
38 | commit '012 021' '102 201' '120 210' | |
|
39 | ||
|
40 | commit '102 120' '012 210' '021 201' | |
|
41 | ||
|
42 | commit '201 210' '021 120' '012 102' | |
|
43 | ||
|
44 | cd .. | |
|
45 | hg clone -q -U -r -1 -r -2 -r -3 orig crossed | |
|
46 | ||
|
47 | for i in crossed/.hg/store/{00manifest.i,data/*.i}; do | |
|
48 | echo $i | |
|
49 | hg debugindex $i | |
|
50 | echo | |
|
51 | done | |
|
52 | ||
|
53 | for i in 0 1 2; do | |
|
54 | hg clone -q -U --pull crossed $i | |
|
55 | echo "% Trying to strip revision $i" | |
|
56 | hg --cwd $i strip $i 2>&1 | hidefilename | |
|
57 | echo "% Verifying" | |
|
58 | hg --cwd $i verify | |
|
59 | echo | |
|
60 | done | |
|
61 |
@@ -0,0 +1,87 b'' | |||
|
1 | crossed/.hg/store/00manifest.i | |
|
2 | rev offset length base linkrev nodeid p1 p2 | |
|
3 | 0 0 112 0 0 6f105cbb914d 000000000000 000000000000 | |
|
4 | 1 112 123 0 1 8f3d04e263e5 000000000000 000000000000 | |
|
5 | 2 235 122 0 2 f0ef8726ac4f 000000000000 000000000000 | |
|
6 | ||
|
7 | crossed/.hg/store/data/012.i | |
|
8 | rev offset length base linkrev nodeid p1 p2 | |
|
9 | 0 0 3 0 0 b8e02f643373 000000000000 000000000000 | |
|
10 | 1 3 3 1 1 5d9299349fc0 000000000000 000000000000 | |
|
11 | 2 6 3 2 2 2661d26c6496 000000000000 000000000000 | |
|
12 | ||
|
13 | crossed/.hg/store/data/021.i | |
|
14 | rev offset length base linkrev nodeid p1 p2 | |
|
15 | 0 0 3 0 0 b8e02f643373 000000000000 000000000000 | |
|
16 | 1 3 3 1 2 5d9299349fc0 000000000000 000000000000 | |
|
17 | 2 6 3 2 1 2661d26c6496 000000000000 000000000000 | |
|
18 | ||
|
19 | crossed/.hg/store/data/102.i | |
|
20 | rev offset length base linkrev nodeid p1 p2 | |
|
21 | 0 0 3 0 1 b8e02f643373 000000000000 000000000000 | |
|
22 | 1 3 3 1 0 5d9299349fc0 000000000000 000000000000 | |
|
23 | 2 6 3 2 2 2661d26c6496 000000000000 000000000000 | |
|
24 | ||
|
25 | crossed/.hg/store/data/120.i | |
|
26 | rev offset length base linkrev nodeid p1 p2 | |
|
27 | 0 0 3 0 1 b8e02f643373 000000000000 000000000000 | |
|
28 | 1 3 3 1 2 5d9299349fc0 000000000000 000000000000 | |
|
29 | 2 6 3 2 0 2661d26c6496 000000000000 000000000000 | |
|
30 | ||
|
31 | crossed/.hg/store/data/201.i | |
|
32 | rev offset length base linkrev nodeid p1 p2 | |
|
33 | 0 0 3 0 2 b8e02f643373 000000000000 000000000000 | |
|
34 | 1 3 3 1 0 5d9299349fc0 000000000000 000000000000 | |
|
35 | 2 6 3 2 1 2661d26c6496 000000000000 000000000000 | |
|
36 | ||
|
37 | crossed/.hg/store/data/210.i | |
|
38 | rev offset length base linkrev nodeid p1 p2 | |
|
39 | 0 0 3 0 2 b8e02f643373 000000000000 000000000000 | |
|
40 | 1 3 3 1 1 5d9299349fc0 000000000000 000000000000 | |
|
41 | 2 6 3 2 0 2661d26c6496 000000000000 000000000000 | |
|
42 | ||
|
43 | % Trying to strip revision 0 | |
|
44 | saving bundle to strip-backup/cbb8c2f0a2e3-backup | |
|
45 | saving bundle to strip-backup/cbb8c2f0a2e3-temp | |
|
46 | adding branch | |
|
47 | adding changesets | |
|
48 | adding manifests | |
|
49 | adding file changes | |
|
50 | added 2 changesets with 12 changes to 6 files (+1 heads) | |
|
51 | % Verifying | |
|
52 | checking changesets | |
|
53 | checking manifests | |
|
54 | crosschecking files in changesets and manifests | |
|
55 | checking files | |
|
56 | 6 files, 2 changesets, 12 total revisions | |
|
57 | ||
|
58 | % Trying to strip revision 1 | |
|
59 | saving bundle to strip-backup/124ecc0cbec9-backup | |
|
60 | saving bundle to strip-backup/124ecc0cbec9-temp | |
|
61 | adding branch | |
|
62 | adding changesets | |
|
63 | adding manifests | |
|
64 | adding file changes | |
|
65 | added 1 changesets with 10 changes to 6 files (+1 heads) | |
|
66 | % Verifying | |
|
67 | checking changesets | |
|
68 | checking manifests | |
|
69 | crosschecking files in changesets and manifests | |
|
70 | checking files | |
|
71 | 6 files, 2 changesets, 12 total revisions | |
|
72 | ||
|
73 | % Trying to strip revision 2 | |
|
74 | saving bundle to strip-backup/f6439b304a1a-backup | |
|
75 | saving bundle to strip-backup/f6439b304a1a-temp | |
|
76 | adding branch | |
|
77 | adding changesets | |
|
78 | adding manifests | |
|
79 | adding file changes | |
|
80 | added 0 changesets with 6 changes to 4 files | |
|
81 | % Verifying | |
|
82 | checking changesets | |
|
83 | checking manifests | |
|
84 | crosschecking files in changesets and manifests | |
|
85 | checking files | |
|
86 | 6 files, 2 changesets, 12 total revisions | |
|
87 |
@@ -21,9 +21,9 b' def _limitheads(cl, stoprev):' | |||
|
21 | 21 | seen[p] = 1 |
|
22 | 22 | return heads |
|
23 | 23 | |
|
24 | def _bundle(repo, bases, heads, node, suffix): | |
|
24 | def _bundle(repo, bases, heads, node, suffix, extranodes=None): | |
|
25 | 25 | """create a bundle with the specified revisions as a backup""" |
|
26 | cg = repo.changegroupsubset(bases, heads, 'strip') | |
|
26 | cg = repo.changegroupsubset(bases, heads, 'strip', extranodes) | |
|
27 | 27 | backupdir = repo.join("strip-backup") |
|
28 | 28 | if not os.path.isdir(backupdir): |
|
29 | 29 | os.mkdir(backupdir) |
@@ -44,6 +44,42 b' def _collectfilenodes(repo, striprev):' | |||
|
44 | 44 | |
|
45 | 45 | return filenodes |
|
46 | 46 | |
|
47 | def _collectextranodes(repo, files, link): | |
|
48 | """return the nodes that have to be saved before the strip""" | |
|
49 | def collectone(revlog): | |
|
50 | extra = [] | |
|
51 | startrev = count = revlog.count() | |
|
52 | # find the truncation point of the revlog | |
|
53 | for i in xrange(0, count): | |
|
54 | node = revlog.node(i) | |
|
55 | lrev = revlog.linkrev(node) | |
|
56 | if lrev >= link: | |
|
57 | startrev = i + 1 | |
|
58 | break | |
|
59 | ||
|
60 | # see if any revision after that point has a linkrev less than link | |
|
61 | # (we have to manually save these guys) | |
|
62 | for i in xrange(startrev, count): | |
|
63 | node = revlog.node(i) | |
|
64 | lrev = revlog.linkrev(node) | |
|
65 | if lrev < link: | |
|
66 | extra.append((node, cl.node(lrev))) | |
|
67 | ||
|
68 | return extra | |
|
69 | ||
|
70 | extranodes = {} | |
|
71 | cl = repo.changelog | |
|
72 | extra = collectone(repo.manifest) | |
|
73 | if extra: | |
|
74 | extranodes[1] = extra | |
|
75 | for fname in files: | |
|
76 | f = repo.file(fname) | |
|
77 | extra = collectone(f) | |
|
78 | if extra: | |
|
79 | extranodes[fname] = extra | |
|
80 | ||
|
81 | return extranodes | |
|
82 | ||
|
47 | 83 | def _stripall(repo, striprev, filenodes): |
|
48 | 84 | """strip the requested nodes from the filelogs""" |
|
49 | 85 | # we go in two steps here so the strip loop happens in a |
@@ -102,23 +138,27 b' def strip(ui, repo, node, backup="all"):' | |||
|
102 | 138 | if cl.rev(x) > striprev: |
|
103 | 139 | savebases[x] = 1 |
|
104 | 140 | |
|
141 | filenodes = _collectfilenodes(repo, striprev) | |
|
142 | ||
|
143 | extranodes = _collectextranodes(repo, filenodes, striprev) | |
|
144 | ||
|
105 | 145 | # create a changegroup for all the branches we need to keep |
|
106 | 146 | if backup == "all": |
|
107 | 147 | _bundle(repo, [node], cl.heads(), node, 'backup') |
|
108 | if saveheads: | |
|
109 |
chgrpfile = _bundle(repo, savebases.keys(), saveheads, node, 'temp' |
|
|
148 | if saveheads or extranodes: | |
|
149 | chgrpfile = _bundle(repo, savebases.keys(), saveheads, node, 'temp', | |
|
150 | extranodes) | |
|
110 | 151 | |
|
111 | filenodes = _collectfilenodes(repo, striprev) | |
|
112 | 152 | _stripall(repo, striprev, filenodes) |
|
113 | 153 | |
|
114 | 154 | change = cl.read(node) |
|
115 | 155 | cl.strip(striprev, striprev) |
|
116 | 156 | repo.manifest.strip(repo.manifest.rev(change[0]), striprev) |
|
117 | if saveheads: | |
|
157 | if saveheads or extranodes: | |
|
118 | 158 | ui.status("adding branch\n") |
|
119 | 159 | f = open(chgrpfile, "rb") |
|
120 | 160 | gen = changegroup.readbundle(f, chgrpfile) |
|
121 | repo.addchangegroup(gen, 'strip', 'bundle:' + chgrpfile) | |
|
161 | repo.addchangegroup(gen, 'strip', 'bundle:' + chgrpfile, True) | |
|
122 | 162 | f.close() |
|
123 | 163 | if backup != "strip": |
|
124 | 164 | os.unlink(chgrpfile) |
@@ -1238,20 +1238,17 b' class revlog(object):' | |||
|
1238 | 1238 | return node |
|
1239 | 1239 | |
|
1240 | 1240 | def strip(self, rev, minlink): |
|
1241 |
if self.count() == 0 |
|
|
1241 | if self.count() == 0: | |
|
1242 | 1242 | return |
|
1243 | 1243 | |
|
1244 | 1244 | if isinstance(self.index, lazyindex): |
|
1245 | 1245 | self._loadindexmap() |
|
1246 | 1246 | |
|
1247 | # When stripping away a revision, we need to make sure it | |
|
1248 | # does not actually belong to an older changeset. | |
|
1249 | # The minlink parameter defines the oldest revision | |
|
1250 | # we're allowed to strip away. | |
|
1251 | while minlink > self.index[rev][4]: | |
|
1252 | rev += 1 | |
|
1253 | if rev >= self.count(): | |
|
1254 | return | |
|
1247 | for rev in xrange(0, self.count()): | |
|
1248 | if self.index[rev][4] >= minlink: | |
|
1249 | break | |
|
1250 | else: | |
|
1251 | return | |
|
1255 | 1252 | |
|
1256 | 1253 | # first truncate the files on disk |
|
1257 | 1254 | end = self.start(rev) |
General Comments 0
You need to be logged in to leave comments.
Login now