##// END OF EJS Templates
mergestate: handle additional record types specially...
Siddharth Agarwal -
r27027:a01ecbcf default
parent child Browse files
Show More
@@ -0,0 +1,25
1 # Extension to write out fake unsupported records into the merge state
2 #
3 #
4
5 from __future__ import absolute_import
6
7 from mercurial import (
8 cmdutil,
9 merge,
10 )
11
12 cmdtable = {}
13 command = cmdutil.command(cmdtable)
14
15 @command('fakemergerecord',
16 [('X', 'mandatory', None, 'add a fake mandatory record'),
17 ('x', 'advisory', None, 'add a fake advisory record')], '')
18 def fakemergerecord(ui, repo, *pats, **opts):
19 ms = merge.mergestate.read(repo)
20 records = ms._makerecords()
21 if opts.get('mandatory'):
22 records.append(('X', 'mandatory record'))
23 if opts.get('advisory'):
24 records.append(('x', 'advisory record'))
25 ms._writerecords(records)
@@ -62,6 +62,8 class mergestate(object):
62 62 (experimental)
63 63 m: the external merge driver defined for this merge plus its run state
64 64 (experimental)
65 X: unsupported mandatory record type (used in tests)
66 x: unsupported advisory record type (used in tests)
65 67
66 68 Merge driver run states (experimental):
67 69 u: driver-resolved files unmarked -- needs to be run next time we're about
@@ -231,6 +233,13 class mergestate(object):
231 233 `type` is a single character, `length` is a 4 byte integer, and
232 234 `content` is an arbitrary byte sequence of length `length`.
233 235
236 Mercurial versions prior to 3.7 have a bug where if there are
237 unsupported mandatory merge records, attempting to clear out the merge
238 state with hg update --clean or similar aborts. The 't' record type
239 works around that by writing out what those versions treat as an
240 advisory record, but later versions interpret as special: the first
241 character is the 'real' record type and everything onwards is the data.
242
234 243 Returns list of records [(TYPE, data), ...]."""
235 244 records = []
236 245 try:
@@ -245,6 +254,8 class mergestate(object):
245 254 off += 4
246 255 record = data[off:(off + length)]
247 256 off += length
257 if rtype == 't':
258 rtype, record = record[0], record[1:]
248 259 records.append((rtype, record))
249 260 f.close()
250 261 except IOError as err:
@@ -326,10 +337,16 class mergestate(object):
326 337 f.close()
327 338
328 339 def _writerecordsv2(self, records):
329 """Write current state on disk in a version 2 file"""
340 """Write current state on disk in a version 2 file
341
342 See the docstring for _readrecordsv2 for why we use 't'."""
343 # these are the records that all version 2 clients can read
344 whitelist = 'LOF'
330 345 f = self._repo.vfs(self.statepathv2, 'w')
331 346 for key, data in records:
332 347 assert len(key) == 1
348 if key not in whitelist:
349 key, data = 't', '%s%s' % (key, data)
333 350 format = '>sI%is' % len(data)
334 351 f.write(_pack(format, key, len(data), data))
335 352 f.close()
@@ -75,10 +75,45 edit the history
75 75 warning: conflicts while merging e! (edit, then use 'hg resolve --mark')
76 76 Fix up the change and run hg histedit --continue
77 77
78 insert unsupported advisory merge record
79 $ hg --config extensions.fakemergerecord=$TESTDIR/fakemergerecord.py fakemergerecord -x
80 $ hg debugmergestate
81 * version 2 records
82 local: 8f7551c7e4a2f2efe0bc8c741baf7f227d65d758
83 other: e860deea161a2f77de56603b340ebbb4536308ae
84 unrecognized entry: x advisory record
85 file: e (record type "F", state "u", hash 58e6b3a414a1e090dfc6029add0f3555ccba127f)
86 local path: e (flags "")
87 ancestor path: e (node 0000000000000000000000000000000000000000)
88 other path: e (node 6b67ccefd5ce6de77e7ead4f5292843a0255329f)
89 $ hg resolve -l
90 U e
78 91
79 abort the edit
92 insert unsupported mandatory merge record
93 $ hg --config extensions.fakemergerecord=$TESTDIR/fakemergerecord.py fakemergerecord -X
94 $ hg debugmergestate
95 * version 2 records
96 local: 8f7551c7e4a2f2efe0bc8c741baf7f227d65d758
97 other: e860deea161a2f77de56603b340ebbb4536308ae
98 file: e (record type "F", state "u", hash 58e6b3a414a1e090dfc6029add0f3555ccba127f)
99 local path: e (flags "")
100 ancestor path: e (node 0000000000000000000000000000000000000000)
101 other path: e (node 6b67ccefd5ce6de77e7ead4f5292843a0255329f)
102 unrecognized entry: X mandatory record
103 $ hg resolve -l
104 abort: unsupported merge state records: X
105 (see https://mercurial-scm.org/wiki/MergeStateRecords for more information)
106 [255]
107 $ hg resolve -ma
108 abort: unsupported merge state records: X
109 (see https://mercurial-scm.org/wiki/MergeStateRecords for more information)
110 [255]
111
112 abort the edit (should clear out merge state)
80 113 $ hg histedit --abort 2>&1 | fixbundle
81 114 2 files updated, 0 files merged, 0 files removed, 0 files unresolved
115 $ hg debugmergestate
116 no merge state found
82 117
83 118 log after abort
84 119 $ hg resolve -l
@@ -68,11 +68,49 Conflicting rebase:
68 68 unresolved conflicts (see hg resolve, then hg rebase --continue)
69 69 [1]
70 70
71 Abort:
71 Insert unsupported advisory merge record:
72
73 $ hg --config extensions.fakemergerecord=$TESTDIR/fakemergerecord.py fakemergerecord -x
74 $ hg debugmergestate
75 * version 2 records
76 local: 3e046f2ecedb793b97ed32108086edd1a162f8bc
77 other: 46f0b057b5c061d276b91491c22151f78698abd2
78 unrecognized entry: x advisory record
79 file: common (record type "F", state "u", hash 94c8c21d08740f5da9eaa38d1f175c592692f0d1)
80 local path: common (flags "")
81 ancestor path: common (node de0a666fdd9c1a0b0698b90d85064d8bd34f74b6)
82 other path: common (node 2f6411de53677f6f1048fef5bf888d67a342e0a5)
83 $ hg resolve -l
84 U common
85
86 Insert unsupported mandatory merge record:
87
88 $ hg --config extensions.fakemergerecord=$TESTDIR/fakemergerecord.py fakemergerecord -X
89 $ hg debugmergestate
90 * version 2 records
91 local: 3e046f2ecedb793b97ed32108086edd1a162f8bc
92 other: 46f0b057b5c061d276b91491c22151f78698abd2
93 file: common (record type "F", state "u", hash 94c8c21d08740f5da9eaa38d1f175c592692f0d1)
94 local path: common (flags "")
95 ancestor path: common (node de0a666fdd9c1a0b0698b90d85064d8bd34f74b6)
96 other path: common (node 2f6411de53677f6f1048fef5bf888d67a342e0a5)
97 unrecognized entry: X mandatory record
98 $ hg resolve -l
99 abort: unsupported merge state records: X
100 (see https://mercurial-scm.org/wiki/MergeStateRecords for more information)
101 [255]
102 $ hg resolve -ma
103 abort: unsupported merge state records: X
104 (see https://mercurial-scm.org/wiki/MergeStateRecords for more information)
105 [255]
106
107 Abort (should clear out unsupported merge state):
72 108
73 109 $ hg rebase --abort
74 110 saved backup bundle to $TESTTMP/a/.hg/strip-backup/3e046f2ecedb-6beef7d5-backup.hg (glob)
75 111 rebase aborted
112 $ hg debugmergestate
113 no merge state found
76 114
77 115 $ hg tglog
78 116 @ 4:draft 'L2'
@@ -249,9 +249,69 resolve <file> should do nothing if 'fil
249 249 $ cat file1
250 250 resolved
251 251
252 insert unsupported advisory merge record
253
254 $ hg --config extensions.fakemergerecord=$TESTDIR/fakemergerecord.py fakemergerecord -x
255 $ hg debugmergestate
256 * version 2 records
257 local: 57653b9f834a4493f7240b0681efcb9ae7cab745
258 other: dc77451844e37f03f5c559e3b8529b2b48d381d1
259 unrecognized entry: x advisory record
260 file: file1 (record type "F", state "r", hash 60b27f004e454aca81b0480209cce5081ec52390)
261 local path: file1 (flags "")
262 ancestor path: file1 (node 2ed2a3912a0b24502043eae84ee4b279c18b90dd)
263 other path: file1 (node 6f4310b00b9a147241b071a60c28a650827fb03d)
264 file: file2 (record type "F", state "u", hash cb99b709a1978bd205ab9dfd4c5aaa1fc91c7523)
265 local path: file2 (flags "")
266 ancestor path: file2 (node 2ed2a3912a0b24502043eae84ee4b279c18b90dd)
267 other path: file2 (node 6f4310b00b9a147241b071a60c28a650827fb03d)
268 $ hg resolve -l
269 R file1
270 U file2
271
272 insert unsupported mandatory merge record
273
274 $ hg --config extensions.fakemergerecord=$TESTDIR/fakemergerecord.py fakemergerecord -X
275 $ hg debugmergestate
276 * version 2 records
277 local: 57653b9f834a4493f7240b0681efcb9ae7cab745
278 other: dc77451844e37f03f5c559e3b8529b2b48d381d1
279 file: file1 (record type "F", state "r", hash 60b27f004e454aca81b0480209cce5081ec52390)
280 local path: file1 (flags "")
281 ancestor path: file1 (node 2ed2a3912a0b24502043eae84ee4b279c18b90dd)
282 other path: file1 (node 6f4310b00b9a147241b071a60c28a650827fb03d)
283 file: file2 (record type "F", state "u", hash cb99b709a1978bd205ab9dfd4c5aaa1fc91c7523)
284 local path: file2 (flags "")
285 ancestor path: file2 (node 2ed2a3912a0b24502043eae84ee4b279c18b90dd)
286 other path: file2 (node 6f4310b00b9a147241b071a60c28a650827fb03d)
287 unrecognized entry: X mandatory record
288 $ hg resolve -l
289 abort: unsupported merge state records: X
290 (see https://mercurial-scm.org/wiki/MergeStateRecords for more information)
291 [255]
292 $ hg resolve -ma
293 abort: unsupported merge state records: X
294 (see https://mercurial-scm.org/wiki/MergeStateRecords for more information)
295 [255]
296 $ hg summary
297 parent: 2:57653b9f834a
298 append baz to files
299 parent: 1:dc77451844e3
300 append bar to files
301 branch: default
302 warning: merge state has unsupported record types: X
303 commit: 2 modified, 2 unknown (merge)
304 update: 2 new changesets (update)
305 phases: 5 draft
306
307 update --clean shouldn't abort on unsupported records
308
309 $ hg up -qC 1
310 $ hg debugmergestate
311 no merge state found
312
252 313 test crashed merge with empty mergestate
253 314
254 $ hg up -qC 1
255 315 $ mkdir .hg/merge
256 316 $ touch .hg/merge/state
257 317
General Comments 0
You need to be logged in to leave comments. Login now