##// END OF EJS Templates
dirstate: avoid a race with multiple commits in the same process...
Greg Ward -
r13704:a464763e default
parent child Browse files
Show More
@@ -0,0 +1,119
1 # reproduce issue2264, issue2516
2
3 create test repo
4 $ cat <<EOF >> $HGRCPATH
5 > [extensions]
6 > transplant =
7 > graphlog =
8 > EOF
9 $ hg init repo
10 $ cd repo
11 $ template="{rev} {desc|firstline} [{branch}]\n"
12
13 # we need to start out with two changesets on the default branch
14 # in order to avoid the cute little optimization where transplant
15 # pulls rather than transplants
16 add initial changesets
17 $ echo feature1 > file1
18 $ hg ci -Am"feature 1"
19 adding file1
20 $ echo feature2 >> file2
21 $ hg ci -Am"feature 2"
22 adding file2
23
24 # The changes to 'bugfix' are enough to show the bug: in fact, with only
25 # those changes, it's a very noisy crash ("RuntimeError: nothing
26 # committed after transplant"). But if we modify a second file in the
27 # transplanted changesets, the bug is much more subtle: transplant
28 # silently drops the second change to 'bugfix' on the floor, and we only
29 # see it when we run 'hg status' after transplanting. Subtle data loss
30 # bugs are worse than crashes, so reproduce the subtle case here.
31 commit bug fixes on bug fix branch
32 $ hg branch fixes
33 marked working directory as branch fixes
34 $ echo fix1 > bugfix
35 $ echo fix1 >> file1
36 $ hg ci -Am"fix 1"
37 adding bugfix
38 $ echo fix2 > bugfix
39 $ echo fix2 >> file1
40 $ hg ci -Am"fix 2"
41 $ hg glog --template="$template"
42 @ 3 fix 2 [fixes]
43 |
44 o 2 fix 1 [fixes]
45 |
46 o 1 feature 2 [default]
47 |
48 o 0 feature 1 [default]
49
50 transplant bug fixes onto release branch
51 $ hg update 0
52 1 files updated, 0 files merged, 2 files removed, 0 files unresolved
53 $ hg branch release
54 marked working directory as branch release
55 $ hg transplant 2 3
56 applying [0-9a-f]{12} (re)
57 [0-9a-f]{12} transplanted to [0-9a-f]{12} (re)
58 applying [0-9a-f]{12} (re)
59 [0-9a-f]{12} transplanted to [0-9a-f]{12} (re)
60 $ hg glog --template="$template"
61 @ 5 fix 2 [release]
62 |
63 o 4 fix 1 [release]
64 |
65 | o 3 fix 2 [fixes]
66 | |
67 | o 2 fix 1 [fixes]
68 | |
69 | o 1 feature 2 [default]
70 |/
71 o 0 feature 1 [default]
72
73 $ hg status
74 $ hg status --rev 0:4
75 M file1
76 A bugfix
77 $ hg status --rev 4:5
78 M bugfix
79 M file1
80
81 now test that we fixed the bug for all scripts/extensions
82 $ cat > $TESTTMP/committwice.py <<__EOF__
83 > from mercurial import ui, hg, match, node
84 >
85 > def replacebyte(fn, b):
86 > f = open("file1", "rb+")
87 > f.seek(0, 0)
88 > f.write(b)
89 > f.close()
90 >
91 > repo = hg.repository(ui.ui(), '.')
92 > assert len(repo) == 6, \
93 > "initial: len(repo) == %d, expected 6" % len(repo)
94 > try:
95 > wlock = repo.wlock()
96 > lock = repo.lock()
97 > m = match.exact(repo.root, '', ['file1'])
98 > replacebyte("file1", "x")
99 > n = repo.commit(text="x", user="test", date=(0, 0), match=m)
100 > print "commit 1: len(repo) == %d" % len(repo)
101 > replacebyte("file1", "y")
102 > n = repo.commit(text="y", user="test", date=(0, 0), match=m)
103 > print "commit 2: len(repo) == %d" % len(repo)
104 > finally:
105 > lock.release()
106 > wlock.release()
107 > __EOF__
108 $ $PYTHON $TESTTMP/committwice.py
109 commit 1: len(repo) == 7
110 commit 2: len(repo) == 8
111
112 Do a size-preserving modification outside of that process
113 $ echo abcd > bugfix
114 $ hg status
115 M bugfix
116 $ hg log --template "{rev} {desc} {files}\n" -r5:
117 5 fix 2 bugfix file1
118 6 x file1
119 7 y file1
@@ -49,6 +49,7 class dirstate(object):
49 49 self._rootdir = os.path.join(root, '')
50 50 self._dirty = False
51 51 self._dirtypl = False
52 self._lastnormal = set() # files believed to be normal
52 53 self._ui = ui
53 54
54 55 @propertycache
@@ -285,6 +286,12 class dirstate(object):
285 286 if f in self._copymap:
286 287 del self._copymap[f]
287 288
289 # Right now, this file is clean: but if some code in this
290 # process modifies it without changing its size before the clock
291 # ticks over to the next second, then it won't be clean anymore.
292 # So make sure that status() will look harder at it.
293 self._lastnormal.add(f)
294
288 295 def normallookup(self, f):
289 296 '''Mark a file normal, but possibly dirty.'''
290 297 if self._pl[1] != nullid and f in self._map:
@@ -308,6 +315,7 class dirstate(object):
308 315 self._map[f] = ('n', 0, -1, -1)
309 316 if f in self._copymap:
310 317 del self._copymap[f]
318 self._lastnormal.discard(f)
311 319
312 320 def otherparent(self, f):
313 321 '''Mark as coming from the other parent, always dirty.'''
@@ -319,6 +327,7 class dirstate(object):
319 327 self._map[f] = ('n', 0, -2, -1)
320 328 if f in self._copymap:
321 329 del self._copymap[f]
330 self._lastnormal.discard(f)
322 331
323 332 def add(self, f):
324 333 '''Mark a file added.'''
@@ -327,6 +336,7 class dirstate(object):
327 336 self._map[f] = ('a', 0, -1, -1)
328 337 if f in self._copymap:
329 338 del self._copymap[f]
339 self._lastnormal.discard(f)
330 340
331 341 def remove(self, f):
332 342 '''Mark a file removed.'''
@@ -343,6 +353,7 class dirstate(object):
343 353 self._map[f] = ('r', 0, size, 0)
344 354 if size == 0 and f in self._copymap:
345 355 del self._copymap[f]
356 self._lastnormal.discard(f)
346 357
347 358 def merge(self, f):
348 359 '''Mark a file merged.'''
@@ -352,6 +363,7 class dirstate(object):
352 363 self._map[f] = ('m', s.st_mode, s.st_size, int(s.st_mtime))
353 364 if f in self._copymap:
354 365 del self._copymap[f]
366 self._lastnormal.discard(f)
355 367
356 368 def forget(self, f):
357 369 '''Forget a file.'''
@@ -361,6 +373,7 class dirstate(object):
361 373 del self._map[f]
362 374 except KeyError:
363 375 self._ui.warn(_("not in dirstate: %s\n") % f)
376 self._lastnormal.discard(f)
364 377
365 378 def _normalize(self, path, knownpath):
366 379 norm_path = os.path.normcase(path)
@@ -640,6 +653,7 class dirstate(object):
640 653 radd = removed.append
641 654 dadd = deleted.append
642 655 cadd = clean.append
656 lastnormal = self._lastnormal.__contains__
643 657
644 658 lnkkind = stat.S_IFLNK
645 659
@@ -672,6 +686,18 class dirstate(object):
672 686 elif (time != int(st.st_mtime)
673 687 and (mode & lnkkind != lnkkind or self._checklink)):
674 688 ladd(fn)
689 elif lastnormal(fn):
690 # If previously in this process we recorded that
691 # this file is clean, think twice: intervening code
692 # may have modified the file in the same second
693 # without changing its size. So force caller to
694 # check file contents. Because we're not updating
695 # self._map, this only affects the current process.
696 # That should be OK because this mainly affects
697 # multiple commits in the same process, and each
698 # commit by definition makes the committed files
699 # clean.
700 ladd(fn)
675 701 elif listclean:
676 702 cadd(fn)
677 703 elif state == 'm':
General Comments 0
You need to be logged in to leave comments. Login now